Compare commits
5 Commits
6a9ec1fe9f
...
c9f15011af
Author | SHA1 | Date | |
---|---|---|---|
c9f15011af | |||
cb6e5dc794 | |||
0610a6acfa | |||
6790083006 | |||
b560341068 |
1
Frontend/.gitignore
vendored
1
Frontend/.gitignore
vendored
@ -1,5 +1,6 @@
|
|||||||
node_modules
|
node_modules
|
||||||
.next/
|
.next/
|
||||||
|
.fleet/
|
||||||
.vscode/
|
.vscode/
|
||||||
.env*.local
|
.env*.local
|
||||||
next-env.d.ts
|
next-env.d.ts
|
4
Frontend/.prettierrc
Normal file
4
Frontend/.prettierrc
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
{
|
||||||
|
"trailingComma": "es5",
|
||||||
|
"tabWidth": 4
|
||||||
|
}
|
@ -3,12 +3,13 @@ import { minecrafter } from "@/font/fonts";
|
|||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import { Metadata } from "next";
|
import { Metadata } from "next";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Page metadata.
|
* Page metadata.
|
||||||
*/
|
*/
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
title: "Docs",
|
title: "Docs",
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -16,34 +17,34 @@ export const metadata: Metadata = {
|
|||||||
*
|
*
|
||||||
* @returns the page jsx
|
* @returns the page jsx
|
||||||
*/
|
*/
|
||||||
const DocsPage = (): JSX.Element => (
|
const DocsPage = (): ReactElement => (
|
||||||
<main className="h-[64vh] flex flex-col gap-3 justify-center items-center">
|
<main className="h-[64vh] flex flex-col gap-3 justify-center items-center">
|
||||||
{/* Creeper */}
|
{/* Creeper */}
|
||||||
<div className="absolute left-28 bottom-16 pointer-events-none">
|
<div className="absolute left-28 bottom-16 pointer-events-none">
|
||||||
<Creeper />
|
<Creeper />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<h1
|
<h1
|
||||||
className={cn(
|
className={cn(
|
||||||
"text-6xl text-minecraft-green-3 pointer-events-none",
|
"text-6xl text-minecraft-green-3 pointer-events-none",
|
||||||
minecrafter.className
|
minecrafter.className
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
Documentation
|
Documentation
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
{/* Content */}
|
{/* Content */}
|
||||||
<h2 className="text-xl">
|
<h2 className="text-xl">
|
||||||
This page is still under construction, however we do have a{" "}
|
This page is still under construction, however we do have a{" "}
|
||||||
<Link
|
<Link
|
||||||
className="text-minecraft-green-4"
|
className="text-minecraft-green-4"
|
||||||
href="https://git.rainnny.club/Rainnny/RESTfulMC/wiki"
|
href="https://git.rainnny.club/Rainnny/RESTfulMC/wiki"
|
||||||
>
|
>
|
||||||
Wiki
|
Wiki
|
||||||
</Link>
|
</Link>
|
||||||
!
|
!
|
||||||
</h2>
|
</h2>
|
||||||
</main>
|
</main>
|
||||||
);
|
);
|
||||||
export default DocsPage;
|
export default DocsPage;
|
||||||
|
@ -1,17 +1,18 @@
|
|||||||
import FeaturedContent from "@/components/landing/featured-content";
|
import FeaturedContent from "@/components/landing/featured-content";
|
||||||
import Hero from "@/components/landing/hero";
|
import Hero from "@/components/landing/hero";
|
||||||
import StatisticCounters from "@/components/landing/statistic-counters";
|
import StatisticCounters from "@/components/landing/statistic-counters";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The landing page.
|
* The landing page.
|
||||||
*
|
*
|
||||||
* @returns the page jsx
|
* @returns the page jsx
|
||||||
*/
|
*/
|
||||||
const LandingPage = (): JSX.Element => (
|
const LandingPage = (): ReactElement => (
|
||||||
<main className="px-3">
|
<main className="px-3">
|
||||||
<Hero />
|
<Hero />
|
||||||
<FeaturedContent />
|
<FeaturedContent />
|
||||||
<StatisticCounters />
|
<StatisticCounters />
|
||||||
</main>
|
</main>
|
||||||
);
|
);
|
||||||
export default LandingPage;
|
export default LandingPage;
|
||||||
|
@ -7,62 +7,63 @@ import { PageProps } from "@/types/page";
|
|||||||
import { Metadata } from "next";
|
import { Metadata } from "next";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { CachedPlayer, getPlayer, type RestfulMCAPIError } from "restfulmc-lib";
|
import { CachedPlayer, getPlayer, type RestfulMCAPIError } from "restfulmc-lib";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The page to lookup a player.
|
* The page to lookup a player.
|
||||||
*
|
*
|
||||||
* @returns the page jsx
|
* @returns the page jsx
|
||||||
*/
|
*/
|
||||||
const PlayerPage = async ({ params }: PageProps): Promise<JSX.Element> => {
|
const PlayerPage = async ({ params }: PageProps): Promise<ReactElement> => {
|
||||||
let error: string | undefined = undefined; // The error to display
|
let error: string | undefined = undefined; // The error to display
|
||||||
let result: CachedPlayer | undefined = undefined; // The player to display
|
let result: CachedPlayer | undefined = undefined; // The player to display
|
||||||
const query: string | undefined = trimQuery(params.slug?.[0]); // The query to search for
|
const query: string | undefined = trimQuery(params.slug?.[0]); // The query to search for
|
||||||
|
|
||||||
// Try and get the player to display
|
// Try and get the player to display
|
||||||
try {
|
try {
|
||||||
result = query ? await getPlayer(query) : undefined;
|
result = query ? await getPlayer(query) : undefined;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
error = (err as RestfulMCAPIError).message; // Set the error message
|
error = (err as RestfulMCAPIError).message; // Set the error message
|
||||||
}
|
}
|
||||||
|
|
||||||
// Render the page
|
// Render the page
|
||||||
return (
|
return (
|
||||||
<main className="px-3 h-screen flex justify-center items-center">
|
<main className="px-3 h-screen flex justify-center items-center">
|
||||||
<div className="mt-0 sm:mt-[45rem] xl:mt-0 flex flex-col xl:flex-row xl:gap-24 2xl:gap-48 transition-all transform-gpu">
|
<div className="mt-0 sm:mt-[45rem] xl:mt-0 flex flex-col xl:flex-row xl:gap-24 2xl:gap-48 transition-all transform-gpu">
|
||||||
{/* Banner */}
|
{/* Banner */}
|
||||||
<Image
|
<Image
|
||||||
className="hidden sm:flex xl:my-auto h-[28rem] pointer-events-none"
|
className="hidden sm:flex xl:my-auto h-[28rem] pointer-events-none"
|
||||||
src="/media/players.webp"
|
src="/media/players.webp"
|
||||||
alt="Minecraft Players"
|
alt="Minecraft Players"
|
||||||
width={632}
|
width={632}
|
||||||
height={632}
|
height={632}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* Search */}
|
{/* Search */}
|
||||||
<div className="pb-16 xl:pb-0 flex flex-col gap-7">
|
<div className="pb-16 xl:pb-0 flex flex-col gap-7">
|
||||||
<h1
|
<h1
|
||||||
className={cn(
|
className={cn(
|
||||||
"mt-20 text-6xl text-minecraft-green-3 text-center pointer-events-none",
|
"mt-20 text-6xl text-minecraft-green-3 text-center pointer-events-none",
|
||||||
minecrafter.className
|
minecrafter.className
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
Player Lookup
|
Player Lookup
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
<div className="flex flex-col gap-5 px-10 xs:px-0">
|
<div className="flex flex-col gap-5 px-10 xs:px-0">
|
||||||
{/* Error */}
|
{/* Error */}
|
||||||
{error && <p className="text-red-500">{error}</p>}
|
{error && <p className="text-red-500">{error}</p>}
|
||||||
|
|
||||||
{/* Search */}
|
{/* Search */}
|
||||||
<PlayerSearch query={query} />
|
<PlayerSearch query={query} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Player Result */}
|
{/* Player Result */}
|
||||||
{result && <PlayerResult player={result} />}
|
{result && <PlayerResult player={result} />}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -73,35 +74,35 @@ const PlayerPage = async ({ params }: PageProps): Promise<JSX.Element> => {
|
|||||||
* @returns the generated metadata
|
* @returns the generated metadata
|
||||||
*/
|
*/
|
||||||
export const generateMetadata = async ({
|
export const generateMetadata = async ({
|
||||||
params,
|
params,
|
||||||
}: PageProps): Promise<Metadata> => {
|
}: PageProps): Promise<Metadata> => {
|
||||||
const query: string | undefined = trimQuery(params.slug?.[0]); // The query to embed for
|
const query: string | undefined = trimQuery(params.slug?.[0]); // The query to embed for
|
||||||
|
|
||||||
// Try and get the player to display
|
// Try and get the player to display
|
||||||
if (query) {
|
if (query) {
|
||||||
try {
|
try {
|
||||||
const player: CachedPlayer = await getPlayer(query); // Get the player to embed
|
const player: CachedPlayer = await getPlayer(query); // Get the player to embed
|
||||||
return Embed({
|
return Embed({
|
||||||
title: `${player.username}'s Profile`,
|
title: `${player.username}'s Profile`,
|
||||||
description: `UUID: ${player.uniqueId}\n\nClick to view data about this player.`,
|
description: `UUID: ${player.uniqueId}\n\nClick to view data about this player.`,
|
||||||
thumbnail: player.skin.parts.HEAD,
|
thumbnail: player.skin.parts.HEAD,
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
const code: number = (err as RestfulMCAPIError).code; // Get the error status code
|
const code: number = (err as RestfulMCAPIError).code; // Get the error status code
|
||||||
if (code === 400) {
|
if (code === 400) {
|
||||||
return Embed({
|
return Embed({
|
||||||
title: "Invalid Player",
|
title: "Invalid Player",
|
||||||
description: "The player you searched for is invalid.",
|
description: "The player you searched for is invalid.",
|
||||||
});
|
});
|
||||||
} else if (code === 404) {
|
} else if (code === 404) {
|
||||||
return Embed({
|
return Embed({
|
||||||
title: "Player Not Found",
|
title: "Player Not Found",
|
||||||
description: "The player you searched for was not found.",
|
description: "The player you searched for was not found.",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return {};
|
return {};
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -111,11 +112,11 @@ export const generateMetadata = async ({
|
|||||||
* @returns the trimmed query
|
* @returns the trimmed query
|
||||||
*/
|
*/
|
||||||
const trimQuery = (query: string | undefined): string | undefined => {
|
const trimQuery = (query: string | undefined): string | undefined => {
|
||||||
// Limit the query to 36 chars
|
// Limit the query to 36 chars
|
||||||
if (query && query.length > 36) {
|
if (query && query.length > 36) {
|
||||||
query = query.substr(0, 36);
|
query = query.substring(0, 36);
|
||||||
}
|
}
|
||||||
return query;
|
return query;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default PlayerPage;
|
export default PlayerPage;
|
||||||
|
@ -3,28 +3,29 @@
|
|||||||
import { minecrafter } from "@/font/fonts";
|
import { minecrafter } from "@/font/fonts";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import CountUp from "react-countup";
|
import CountUp from "react-countup";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Props for the counter.
|
* Props for the counter.
|
||||||
*/
|
*/
|
||||||
type CounterProps = {
|
type CounterProps = {
|
||||||
/**
|
/**
|
||||||
* The name of the counter.
|
* The name of the counter.
|
||||||
*/
|
*/
|
||||||
name: string;
|
name: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The amount to count up to.
|
* The amount to count up to.
|
||||||
*/
|
*/
|
||||||
amount: number;
|
amount: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The optional duration of the count up.
|
* The optional duration of the count up.
|
||||||
* <p>
|
* <p>
|
||||||
* Uses the default duration if not provided.
|
* Uses the default duration if not provided.
|
||||||
* </p>
|
* </p>
|
||||||
*/
|
*/
|
||||||
duration?: number | undefined;
|
duration?: number | undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -34,16 +35,19 @@ type CounterProps = {
|
|||||||
* @param duration the optional duration of the count up
|
* @param duration the optional duration of the count up
|
||||||
* @returns the counter jsx
|
* @returns the counter jsx
|
||||||
*/
|
*/
|
||||||
const Counter = ({ name, amount, duration }: CounterProps): JSX.Element => (
|
const Counter = ({ name, amount, duration }: CounterProps): ReactElement => (
|
||||||
<div className="flex flex-col items-center gap-4">
|
<div className="flex flex-col items-center gap-4">
|
||||||
<h1
|
<h1
|
||||||
className={cn("text-6xl text-minecraft-green-3", minecrafter.className)}
|
className={cn(
|
||||||
>
|
"text-6xl text-minecraft-green-3",
|
||||||
{name}
|
minecrafter.className
|
||||||
</h1>
|
)}
|
||||||
<h2 className="text-4xl font-semibold uppercase">
|
>
|
||||||
<CountUp start={0} end={amount} duration={duration} />
|
{name}
|
||||||
</h2>
|
</h1>
|
||||||
</div>
|
<h2 className="text-4xl font-semibold uppercase">
|
||||||
|
<CountUp start={0} end={amount} duration={duration} />
|
||||||
|
</h2>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
export default Counter;
|
export default Counter;
|
||||||
|
@ -1,16 +1,17 @@
|
|||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A creeper image.
|
* A creeper image.
|
||||||
*
|
*
|
||||||
* @returns the creeper jsx
|
* @returns the creeper jsx
|
||||||
*/
|
*/
|
||||||
const Creeper = (): JSX.Element => (
|
const Creeper = (): ReactElement => (
|
||||||
<Image
|
<Image
|
||||||
src="/media/creeper.png"
|
src="/media/creeper.png"
|
||||||
alt="A Minecraft Creeper"
|
alt="A Minecraft Creeper"
|
||||||
width={216}
|
width={216}
|
||||||
height={216}
|
height={216}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
export default Creeper;
|
export default Creeper;
|
||||||
|
@ -1,20 +1,23 @@
|
|||||||
import { Metadata } from "next";
|
import { Metadata } from "next";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Props for an embed.
|
||||||
|
*/
|
||||||
type EmbedProps = {
|
type EmbedProps = {
|
||||||
/**
|
/**
|
||||||
* The title of the embed.
|
* The title of the embed.
|
||||||
*/
|
*/
|
||||||
title: string;
|
title: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The description of the embed.
|
* The description of the embed.
|
||||||
*/
|
*/
|
||||||
description: string;
|
description: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The optional thumbnail image of the embed.
|
* The optional thumbnail image of the embed.
|
||||||
*/
|
*/
|
||||||
thumbnail?: string;
|
thumbnail?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -24,24 +27,24 @@ type EmbedProps = {
|
|||||||
* @returns the embed jsx
|
* @returns the embed jsx
|
||||||
*/
|
*/
|
||||||
const Embed = ({
|
const Embed = ({
|
||||||
title,
|
title,
|
||||||
description,
|
description,
|
||||||
thumbnail = "",
|
thumbnail = "",
|
||||||
}: EmbedProps): Metadata => {
|
}: EmbedProps): Metadata => {
|
||||||
return {
|
return {
|
||||||
title: title,
|
title: title,
|
||||||
openGraph: {
|
openGraph: {
|
||||||
title: `${title}`,
|
title: `${title}`,
|
||||||
description: description,
|
description: description,
|
||||||
images: [
|
images: [
|
||||||
{
|
{
|
||||||
url: thumbnail,
|
url: thumbnail,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
twitter: {
|
twitter: {
|
||||||
card: "summary",
|
card: "summary",
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
export default Embed;
|
export default Embed;
|
||||||
|
@ -1,2 +1,4 @@
|
|||||||
const Footer = (): JSX.Element => <footer>FOOTER</footer>;
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
|
const Footer = (): ReactElement => <footer>FOOTER</footer>;
|
||||||
export default Footer;
|
export default Footer;
|
||||||
|
@ -2,30 +2,38 @@ import MinecraftButton from "@/components/minecraft-button";
|
|||||||
import { Skeleton } from "@/components/ui/skeleton";
|
import { Skeleton } from "@/components/ui/skeleton";
|
||||||
import { StarIcon } from "@heroicons/react/24/outline";
|
import { StarIcon } from "@heroicons/react/24/outline";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { Suspense } from "react";
|
import { ReactElement, Suspense } from "react";
|
||||||
|
|
||||||
const GitHubStarButton = async (): Promise<JSX.Element> => {
|
/**
|
||||||
return (
|
* The button to display the amount
|
||||||
<Link
|
* of stars the GitHub repository has.
|
||||||
href="https://github.com/Rainnny7/RESTfulMC"
|
*
|
||||||
rel="noopener noreferrer"
|
* @returns the component jsx
|
||||||
target="_blank"
|
*/
|
||||||
>
|
const GitHubStarButton = async (): Promise<ReactElement> => {
|
||||||
<MinecraftButton className="flex gap-1.5 items-center group/star">
|
return (
|
||||||
{/* Star Count */}
|
<Link
|
||||||
<Suspense fallback={<Skeleton className="w-4 h-5 rounded-md" />}>
|
href="https://github.com/Rainnny7/RESTfulMC"
|
||||||
<GitHubStarCount />
|
rel="noopener noreferrer"
|
||||||
</Suspense>
|
target="_blank"
|
||||||
|
>
|
||||||
|
<MinecraftButton className="flex gap-1.5 items-center group/star">
|
||||||
|
{/* Star Count */}
|
||||||
|
<Suspense
|
||||||
|
fallback={<Skeleton className="w-4 h-5 rounded-md" />}
|
||||||
|
>
|
||||||
|
<GitHubStarCount />
|
||||||
|
</Suspense>
|
||||||
|
|
||||||
<StarIcon
|
<StarIcon
|
||||||
className="group-hover/star:text-orange-400 delay-0 transition-all transform-gpu"
|
className="group-hover/star:text-orange-400 delay-0 transition-all transform-gpu"
|
||||||
width={22}
|
width={22}
|
||||||
height={22}
|
height={22}
|
||||||
/>
|
/>
|
||||||
<span>Star on GitHub</span>
|
<span>Star on GitHub</span>
|
||||||
</MinecraftButton>
|
</MinecraftButton>
|
||||||
</Link>
|
</Link>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -33,11 +41,11 @@ const GitHubStarButton = async (): Promise<JSX.Element> => {
|
|||||||
*
|
*
|
||||||
* @returns the star count jsx
|
* @returns the star count jsx
|
||||||
*/
|
*/
|
||||||
const GitHubStarCount = async (): Promise<JSX.Element> => {
|
const GitHubStarCount = async (): Promise<ReactElement> => {
|
||||||
const stars: number = await getStarCount(); // Get the repo star count
|
const stars: number = await getStarCount(); // Get the repo star count
|
||||||
return (
|
return (
|
||||||
<code className="px-1 rounded-md bg-minecraft-green-3/80">{stars}</code>
|
<code className="px-1 rounded-md bg-minecraft-green-3/80">{stars}</code>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -47,12 +55,12 @@ const GitHubStarCount = async (): Promise<JSX.Element> => {
|
|||||||
* @returns the star count
|
* @returns the star count
|
||||||
*/
|
*/
|
||||||
const getStarCount = async (): Promise<number> => {
|
const getStarCount = async (): Promise<number> => {
|
||||||
const response: Response = await fetch(
|
const response: Response = await fetch(
|
||||||
"https://api.github.com/repos/Rainnny7/RESTfulMC",
|
"https://api.github.com/repos/Rainnny7/RESTfulMC",
|
||||||
{ next: { revalidate: 300 } } // Revalidate every 5 minutes
|
{ next: { revalidate: 300 } } // Revalidate every 5 minutes
|
||||||
);
|
);
|
||||||
const json: any = await response.json(); // Get the JSON response
|
const json: any = await response.json(); // Get the JSON response
|
||||||
return json.stargazers_count; // Return the stars
|
return json.stargazers_count; // Return the stars
|
||||||
};
|
};
|
||||||
|
|
||||||
export default GitHubStarButton;
|
export default GitHubStarButton;
|
||||||
|
@ -3,20 +3,23 @@ import { minecrafter } from "@/font/fonts";
|
|||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import { FeaturedItemProps } from "@/types/config";
|
import { FeaturedItemProps } from "@/types/config";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The featured content component.
|
* The featured content component.
|
||||||
*
|
*
|
||||||
* @returns the featured content jsx
|
* @returns the featured content jsx
|
||||||
*/
|
*/
|
||||||
const FeaturedContent = (): JSX.Element => (
|
const FeaturedContent = (): ReactElement => (
|
||||||
<div className="flex justify-center items-center">
|
<div className="flex justify-center items-center">
|
||||||
<div className="max-w-2xl flex flex-wrap justify-center gap-5">
|
<div className="max-w-2xl flex flex-wrap justify-center gap-5">
|
||||||
{config.featuredItems.map((item, index) => (
|
{config.featuredItems.map(
|
||||||
<FeaturedItem key={index} {...item} />
|
(item: FeaturedItemProps, index: number) => (
|
||||||
))}
|
<FeaturedItem key={index} {...item} />
|
||||||
</div>
|
)
|
||||||
</div>
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -26,25 +29,28 @@ const FeaturedContent = (): JSX.Element => (
|
|||||||
* @returns the item jsx
|
* @returns the item jsx
|
||||||
*/
|
*/
|
||||||
const FeaturedItem = ({
|
const FeaturedItem = ({
|
||||||
name,
|
name,
|
||||||
description,
|
description,
|
||||||
image,
|
image,
|
||||||
href,
|
href,
|
||||||
}: FeaturedItemProps): JSX.Element => (
|
}: FeaturedItemProps): ReactElement => (
|
||||||
<Link
|
<Link
|
||||||
className="pt-28 w-[19rem] h-80 flex flex-col gap-1 items-center bg-center bg-cover bg-no-repeat rounded-3xl text-center backdrop-blur-md hover:scale-[1.01] transition-all transform-gpu"
|
className="pt-28 w-[19rem] h-80 flex flex-col gap-1 items-center bg-center bg-cover bg-no-repeat rounded-3xl text-center backdrop-blur-md hover:scale-[1.01] transition-all transform-gpu"
|
||||||
href={href}
|
href={href}
|
||||||
style={{
|
style={{
|
||||||
backgroundImage: `url(${image})`,
|
backgroundImage: `url(${image})`,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<h1
|
<h1
|
||||||
className={cn("text-3xl font-semibold text-white", minecrafter.className)}
|
className={cn(
|
||||||
>
|
"text-3xl font-semibold text-white",
|
||||||
{name}
|
minecrafter.className
|
||||||
</h1>
|
)}
|
||||||
<h2 className="text-md max-w-[15rem]">{description}</h2>
|
>
|
||||||
</Link>
|
{name}
|
||||||
|
</h1>
|
||||||
|
<h2 className="text-md max-w-[15rem]">{description}</h2>
|
||||||
|
</Link>
|
||||||
);
|
);
|
||||||
|
|
||||||
export default FeaturedContent;
|
export default FeaturedContent;
|
||||||
|
@ -4,40 +4,43 @@ import config from "@/config";
|
|||||||
import { minecrafter } from "@/font/fonts";
|
import { minecrafter } from "@/font/fonts";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The hero content.
|
* The hero content.
|
||||||
*
|
*
|
||||||
* @returns the hero jsx
|
* @returns the hero jsx
|
||||||
*/
|
*/
|
||||||
const Hero = (): JSX.Element => (
|
const Hero = (): ReactElement => (
|
||||||
<div className="pt-56 pb-40 flex flex-col gap-8 justify-center items-center">
|
<div className="pt-56 pb-40 flex flex-col gap-8 justify-center items-center">
|
||||||
<div className="flex flex-col gap-4 items-center text-center">
|
<div className="flex flex-col gap-4 items-center text-center">
|
||||||
{/* Title */}
|
{/* Title */}
|
||||||
<h1
|
<h1
|
||||||
className={cn(
|
className={cn(
|
||||||
"text-5xl sm:text-6xl text-minecraft-green-3",
|
"text-5xl sm:text-6xl text-minecraft-green-3",
|
||||||
minecrafter.className
|
minecrafter.className
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
{config.siteName}
|
{config.siteName}
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
{/* Subtitle */}
|
{/* Subtitle */}
|
||||||
<h2 className="text-xl">{config.metadata.description}</h2>
|
<h2 className="text-xl">{config.metadata.description}</h2>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Links */}
|
{/* Links */}
|
||||||
<div className="flex gap-5 xs:gap-10">
|
<div className="flex gap-5 xs:gap-10">
|
||||||
<Link href="/docs">
|
<Link href="/docs">
|
||||||
<MinecraftButton className="w-44 h-12">Get Started</MinecraftButton>
|
<MinecraftButton className="w-44 h-12">
|
||||||
</Link>
|
Get Started
|
||||||
|
</MinecraftButton>
|
||||||
|
</Link>
|
||||||
|
|
||||||
{/* Star on Github <3 */}
|
{/* Star on Github <3 */}
|
||||||
<div className="md:hidden">
|
<div className="md:hidden">
|
||||||
<GitHubStarButton />
|
<GitHubStarButton />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
export default Hero;
|
export default Hero;
|
||||||
|
@ -1,18 +1,19 @@
|
|||||||
import Counter from "@/components/counter";
|
import Counter from "@/components/counter";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The statistic counters component.
|
* The statistic counters component.
|
||||||
*
|
*
|
||||||
* @returns the counters jsx
|
* @returns the counters jsx
|
||||||
*/
|
*/
|
||||||
const StatisticCounters = (): JSX.Element => (
|
const StatisticCounters = (): ReactElement => (
|
||||||
<div className="py-56 flex justify-center items-center">
|
<div className="py-56 flex justify-center items-center">
|
||||||
<div className="grid grid-flow-row grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 2xl:grid-cols-4 gap-24">
|
<div className="grid grid-flow-row grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 2xl:grid-cols-4 gap-24">
|
||||||
<Counter name="Testing" amount={1_000_000} />
|
<Counter name="Testing" amount={1_000_000} />
|
||||||
<Counter name="Testing" amount={1_000_000} />
|
<Counter name="Testing" amount={1_000_000} />
|
||||||
<Counter name="Testing" amount={1_000_000} />
|
<Counter name="Testing" amount={1_000_000} />
|
||||||
<Counter name="Testing" amount={1_000_000} />
|
<Counter name="Testing" amount={1_000_000} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
export default StatisticCounters;
|
export default StatisticCounters;
|
||||||
|
@ -1,19 +1,20 @@
|
|||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
|
import { ButtonHTMLAttributes, ReactElement, ReactNode } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Props for this button.
|
* Props for this button.
|
||||||
*/
|
*/
|
||||||
type MinecraftButtonProps = {
|
type MinecraftButtonProps = {
|
||||||
/**
|
/**
|
||||||
* The class name to apply to this button.
|
* The class name to apply to this button.
|
||||||
*/
|
*/
|
||||||
className?: string;
|
className?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The children of this button.
|
* The children of this button.
|
||||||
*/
|
*/
|
||||||
children: React.ReactNode;
|
children: ReactNode;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -22,27 +23,27 @@ type MinecraftButtonProps = {
|
|||||||
* @returns the button jsx
|
* @returns the button jsx
|
||||||
*/
|
*/
|
||||||
const MinecraftButton = ({
|
const MinecraftButton = ({
|
||||||
className,
|
className,
|
||||||
children,
|
children,
|
||||||
...props
|
...props
|
||||||
}: React.ButtonHTMLAttributes<HTMLButtonElement> &
|
}: ButtonHTMLAttributes<HTMLButtonElement> &
|
||||||
MinecraftButtonProps): JSX.Element => (
|
MinecraftButtonProps): ReactElement => (
|
||||||
<Button
|
<Button
|
||||||
className={cn(
|
className={cn(
|
||||||
"before:absolute before:-inset-x-5 before:rotate-90 before:w-9 before:h-1 before:bg-minecraft-green-1", // Left Green Bar
|
"before:absolute before:-inset-x-5 before:rotate-90 before:w-9 before:h-1 before:bg-minecraft-green-1", // Left Green Bar
|
||||||
"after:absolute after:right-[-1.24rem] after:rotate-90 after:w-9 after:h-1 after:bg-minecraft-green-1", // Right Green Bar
|
"after:absolute after:right-[-1.24rem] after:rotate-90 after:w-9 after:h-1 after:bg-minecraft-green-1", // Right Green Bar
|
||||||
"relative h-full px-5 bg-minecraft-green-2 hover:opacity-85 hover:bg-minecraft-green-2 rounded-none tracking-wide font-semibold uppercase transition-all transform-gpu", // Styling
|
"relative h-full px-5 bg-minecraft-green-2 hover:opacity-85 hover:bg-minecraft-green-2 rounded-none tracking-wide font-semibold uppercase transition-all transform-gpu", // Styling
|
||||||
className
|
className
|
||||||
)}
|
)}
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
style={{
|
style={{
|
||||||
// Above and below the button shadow
|
// Above and below the button shadow
|
||||||
boxShadow:
|
boxShadow:
|
||||||
"inset 0 -4px 0 hsl(var(--minecraft-green-1)), inset 0 4px 0 hsl(var(--minecraft-green-3))",
|
"inset 0 -4px 0 hsl(var(--minecraft-green-1)), inset 0 4px 0 hsl(var(--minecraft-green-3))",
|
||||||
}}
|
}}
|
||||||
{...props}
|
{...props}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</Button>
|
</Button>
|
||||||
);
|
);
|
||||||
export default MinecraftButton;
|
export default MinecraftButton;
|
||||||
|
@ -7,66 +7,69 @@ import { cn } from "@/lib/utils";
|
|||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The navbar for the site.
|
* The navbar for the site.
|
||||||
*
|
*
|
||||||
* @returns the navbar jsx
|
* @returns the navbar jsx
|
||||||
*/
|
*/
|
||||||
const Navbar = (): JSX.Element => {
|
const Navbar = (): ReactElement => {
|
||||||
const path: string = usePathname(); // Get the current path
|
const path: string = usePathname(); // Get the current path
|
||||||
return (
|
return (
|
||||||
<nav className="fixed inset-x-0 flex h-16 sm:px-12 justify-center sm:justify-between items-center bg-navbar-background z-50">
|
<nav className="fixed inset-x-0 flex h-16 sm:px-12 justify-center sm:justify-between items-center bg-navbar-background z-50">
|
||||||
{/* Left */}
|
{/* Left */}
|
||||||
<div className="flex gap-3 xs:gap-7 lg:gap-12 items-center transition-all transform-gpu">
|
<div className="flex gap-3 xs:gap-7 lg:gap-12 items-center transition-all transform-gpu">
|
||||||
{/* App Branding */}
|
{/* App Branding */}
|
||||||
<Link
|
<Link
|
||||||
className={cn(
|
className={cn(
|
||||||
"text-3xl text-minecraft-green-3 hover:opacity-85 transition-all transform-gpu",
|
"text-3xl text-minecraft-green-3 hover:opacity-85 transition-all transform-gpu",
|
||||||
minecrafter.className
|
minecrafter.className
|
||||||
)}
|
)}
|
||||||
href="/"
|
href="/"
|
||||||
>
|
>
|
||||||
{/* Small Screens */}
|
{/* Small Screens */}
|
||||||
<Image
|
<Image
|
||||||
className="lg:hidden"
|
className="lg:hidden"
|
||||||
src="/media/logo.webp"
|
src="/media/logo.webp"
|
||||||
alt="Site Logo"
|
alt="Site Logo"
|
||||||
width={42}
|
width={42}
|
||||||
height={42}
|
height={42}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* Large Screens */}
|
{/* Large Screens */}
|
||||||
<span className="hidden lg:flex">{config.siteName}</span>
|
<span className="hidden lg:flex">{config.siteName}</span>
|
||||||
</Link>
|
</Link>
|
||||||
|
|
||||||
{/* Links */}
|
{/* Links */}
|
||||||
<div className="flex gap-7">
|
<div className="flex gap-7">
|
||||||
{Object.entries(config.navbarLinks).map((link, index) => {
|
{Object.entries(config.navbarLinks).map(
|
||||||
const url: string = link[1]; // The href of the link
|
(link: [string, string], index: number) => {
|
||||||
let active: boolean = path.startsWith(url); // Is this the active link?
|
const url: string = link[1]; // The href of the link
|
||||||
return (
|
let active: boolean = path.startsWith(url); // Is this the active link?
|
||||||
<Link
|
return (
|
||||||
key={index}
|
<Link
|
||||||
className={cn(
|
key={index}
|
||||||
"font-semibold uppercase hover:text-minecraft-green-4 transition-all transform-gpu",
|
className={cn(
|
||||||
active && "text-minecraft-green-4"
|
"font-semibold uppercase hover:text-minecraft-green-4 transition-all transform-gpu",
|
||||||
)}
|
active && "text-minecraft-green-4"
|
||||||
href={url}
|
)}
|
||||||
>
|
href={url}
|
||||||
{link[0]}
|
>
|
||||||
</Link>
|
{link[0]}
|
||||||
);
|
</Link>
|
||||||
})}
|
);
|
||||||
</div>
|
}
|
||||||
</div>
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Social Buttons - Right */}
|
{/* Social Buttons - Right */}
|
||||||
<div className="hidden md:flex">
|
<div className="hidden md:flex">
|
||||||
{/* Star on Github <3 */}
|
{/* Star on Github <3 */}
|
||||||
<GitHubStarButton />
|
<GitHubStarButton />
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
export default Navbar;
|
export default Navbar;
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { CachedPlayer, SkinPart } from "restfulmc-lib";
|
import { CachedPlayer, SkinPart } from "restfulmc-lib";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The result of a player search.
|
* The result of a player search.
|
||||||
@ -10,62 +11,68 @@ import { CachedPlayer, SkinPart } from "restfulmc-lib";
|
|||||||
* @returns the player result jsx
|
* @returns the player result jsx
|
||||||
*/
|
*/
|
||||||
const PlayerResult = ({
|
const PlayerResult = ({
|
||||||
player: {
|
player: {
|
||||||
uniqueId,
|
uniqueId,
|
||||||
username,
|
username,
|
||||||
skin: { parts },
|
skin: { parts },
|
||||||
legacy,
|
legacy,
|
||||||
},
|
},
|
||||||
}: {
|
}: {
|
||||||
player: CachedPlayer;
|
player: CachedPlayer;
|
||||||
}): JSX.Element => (
|
}): ReactElement => (
|
||||||
<div className="px-2 py-3 flex flex-col gap-3 items-center bg-muted rounded-xl divide-y divide-zinc-700">
|
<div className="px-2 py-3 flex flex-col gap-3 items-center bg-muted rounded-xl divide-y divide-zinc-700">
|
||||||
{/* Details */}
|
{/* Details */}
|
||||||
<div className="flex gap-5 items-center">
|
<div className="flex gap-5 items-center">
|
||||||
{/* Player Head */}
|
{/* Player Head */}
|
||||||
<Image
|
<Image
|
||||||
className="w-24 h-24 sm:w-28 sm:h-28 md:w-32 md:h-32"
|
className="w-24 h-24 sm:w-28 sm:h-28 md:w-32 md:h-32"
|
||||||
src={parts.HEAD}
|
src={parts.HEAD}
|
||||||
alt={`${username}'s Head`}
|
alt={`${username}'s Head`}
|
||||||
width={128}
|
width={128}
|
||||||
height={128}
|
height={128}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* Name, Unique ID, and Badges */}
|
{/* Name, Unique ID, and Badges */}
|
||||||
<div className="flex flex-col gap-1.5">
|
<div className="flex flex-col gap-1.5">
|
||||||
<h1 className="text-xl font-bold text-minecraft-green-3">{username}</h1>
|
<h1 className="text-xl font-bold text-minecraft-green-3">
|
||||||
<code className="text-xs xs:text-sm text-zinc-300">{uniqueId}</code>
|
{username}
|
||||||
|
</h1>
|
||||||
|
<code className="text-xs xs:text-sm text-zinc-300">
|
||||||
|
{uniqueId}
|
||||||
|
</code>
|
||||||
|
|
||||||
{/* Legacy Badge */}
|
{/* Legacy Badge */}
|
||||||
{legacy && <p className="text-sm font-semibold uppercase">Legacy</p>}
|
{legacy && (
|
||||||
</div>
|
<p className="text-sm font-semibold uppercase">Legacy</p>
|
||||||
</div>
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Skin Parts */}
|
{/* Skin Parts */}
|
||||||
<div className="pt-3 w-[90%] flex flex-col gap-3">
|
<div className="pt-3 w-[90%] flex flex-col gap-3">
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<h1 className="font-semibold uppercase">Skin Parts</h1>
|
<h1 className="font-semibold uppercase">Skin Parts</h1>
|
||||||
|
|
||||||
{/* Skin Parts */}
|
{/* Skin Parts */}
|
||||||
<div className="flex gap-5">
|
<div className="flex gap-5">
|
||||||
{Object.entries(parts)
|
{Object.entries(parts)
|
||||||
.filter(
|
.filter(
|
||||||
([part]) =>
|
([part]) =>
|
||||||
part === SkinPart.HEAD ||
|
part === SkinPart.HEAD ||
|
||||||
part === SkinPart.FACE ||
|
part === SkinPart.FACE ||
|
||||||
part === SkinPart.BODY_FLAT
|
part === SkinPart.BODY_FLAT
|
||||||
)
|
)
|
||||||
.map(([part, url], index) => (
|
.map(([part, url], index) => (
|
||||||
<Link key={index} href={url} target="_blank">
|
<Link key={index} href={url} target="_blank">
|
||||||
<img
|
<img
|
||||||
className="h-20 sm:h-24 md:h-28 hover:scale-[1.02] transition-all transform-gpu"
|
className="h-20 sm:h-24 md:h-28 hover:scale-[1.02] transition-all transform-gpu"
|
||||||
src={url}
|
src={url}
|
||||||
alt={`${username}'s ${part}`}
|
alt={`${username}'s ${part}`}
|
||||||
/>
|
/>
|
||||||
</Link>
|
</Link>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
export default PlayerResult;
|
export default PlayerResult;
|
||||||
|
@ -3,6 +3,7 @@ import { Input } from "@/components/ui/input";
|
|||||||
|
|
||||||
import { Label } from "@/components/ui/label";
|
import { Label } from "@/components/ui/label";
|
||||||
import { redirect } from "next/navigation";
|
import { redirect } from "next/navigation";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A component for searching for a player.
|
* A component for searching for a player.
|
||||||
@ -11,31 +12,31 @@ import { redirect } from "next/navigation";
|
|||||||
* @returns the search component jsx
|
* @returns the search component jsx
|
||||||
*/
|
*/
|
||||||
const PlayerSearch = ({
|
const PlayerSearch = ({
|
||||||
query,
|
query,
|
||||||
}: {
|
}: {
|
||||||
query: string | undefined;
|
query: string | undefined;
|
||||||
}): JSX.Element => {
|
}): ReactElement => {
|
||||||
const handleRedirect = async (form: FormData) => {
|
const handleRedirect = async (form: FormData): Promise<void> => {
|
||||||
"use server";
|
"use server";
|
||||||
redirect(`/player/${form.get("query")}`);
|
redirect(`/player/${form.get("query")}`);
|
||||||
};
|
};
|
||||||
return (
|
return (
|
||||||
<form
|
<form
|
||||||
className="flex flex-col gap-7 justify-center items-center"
|
className="flex flex-col gap-7 justify-center items-center"
|
||||||
action={handleRedirect}
|
action={handleRedirect}
|
||||||
>
|
>
|
||||||
<div className="w-full flex flex-col gap-3">
|
<div className="w-full flex flex-col gap-3">
|
||||||
<Label htmlFor="query">Username or UUID</Label>
|
<Label htmlFor="query">Username or UUID</Label>
|
||||||
<Input
|
<Input
|
||||||
type="search"
|
type="search"
|
||||||
name="query"
|
name="query"
|
||||||
placeholder="Query..."
|
placeholder="Query..."
|
||||||
defaultValue={query}
|
defaultValue={query}
|
||||||
maxLength={36}
|
maxLength={36}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<MinecraftButton type="submit">Search</MinecraftButton>
|
<MinecraftButton type="submit">Search</MinecraftButton>
|
||||||
</form>
|
</form>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
export default PlayerSearch;
|
export default PlayerSearch;
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
import { Config } from "@/types/config";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The configuration for this app.
|
* The configuration for this app.
|
||||||
*/
|
*/
|
||||||
const config: Config = require("@/configJson");
|
import config from "@/configJson";
|
||||||
export default config;
|
export default config;
|
||||||
|
@ -3,76 +3,76 @@
|
|||||||
@tailwind utilities;
|
@tailwind utilities;
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
:root {
|
:root {
|
||||||
--background: 30 5% 9%;
|
--background: 30 5% 9%;
|
||||||
--foreground: 0 0% 98%;
|
--foreground: 0 0% 98%;
|
||||||
|
|
||||||
--card: 240 10% 3.9%;
|
--card: 240 10% 3.9%;
|
||||||
--card-foreground: 0 0% 98%;
|
--card-foreground: 0 0% 98%;
|
||||||
|
|
||||||
--popover: 240 10% 3.9%;
|
--popover: 240 10% 3.9%;
|
||||||
--popover-foreground: 0 0% 98%;
|
--popover-foreground: 0 0% 98%;
|
||||||
|
|
||||||
--primary: 0 0% 98%;
|
--primary: 0 0% 98%;
|
||||||
--primary-foreground: 240 5.9% 10%;
|
--primary-foreground: 240 5.9% 10%;
|
||||||
|
|
||||||
--secondary: 240 3.7% 15.9%;
|
--secondary: 240 3.7% 15.9%;
|
||||||
--secondary-foreground: 0 0% 98%;
|
--secondary-foreground: 0 0% 98%;
|
||||||
|
|
||||||
--muted: 20 4% 14%;
|
--muted: 20 4% 14%;
|
||||||
--muted-foreground: 240 5% 64.9%;
|
--muted-foreground: 240 5% 64.9%;
|
||||||
|
|
||||||
--accent: 240 3.7% 15.9%;
|
--accent: 240 3.7% 15.9%;
|
||||||
--accent-foreground: 0 0% 98%;
|
--accent-foreground: 0 0% 98%;
|
||||||
|
|
||||||
--destructive: 0 62.8% 30.6%;
|
--destructive: 0 62.8% 30.6%;
|
||||||
--destructive-foreground: 0 0% 98%;
|
--destructive-foreground: 0 0% 98%;
|
||||||
|
|
||||||
--border: 240 3.7% 15.9%;
|
--border: 240 3.7% 15.9%;
|
||||||
--input: 240 3.7% 15.9%;
|
--input: 240 3.7% 15.9%;
|
||||||
--ring: 240 4.9% 83.9%;
|
--ring: 240 4.9% 83.9%;
|
||||||
|
|
||||||
--radius: 0.5rem;
|
--radius: 0.5rem;
|
||||||
|
|
||||||
/* Navbar */
|
/* Navbar */
|
||||||
--navbar-background: 0 0% 7%;
|
--navbar-background: 0 0% 7%;
|
||||||
|
|
||||||
/* Minecraft Colors (Dark -> Light) */
|
/* Minecraft Colors (Dark -> Light) */
|
||||||
--minecraft-green-1: 108 56% 25%;
|
--minecraft-green-1: 108 56% 25%;
|
||||||
--minecraft-green-2: 107 55% 34%;
|
--minecraft-green-2: 107 55% 34%;
|
||||||
--minecraft-green-3: 104 51% 43%;
|
--minecraft-green-3: 104 51% 43%;
|
||||||
--minecraft-green-4: 103 50% 53%;
|
--minecraft-green-4: 103 50% 53%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
* {
|
* {
|
||||||
@apply border-border;
|
@apply border-border;
|
||||||
|
|
||||||
/* Scrollbar (Firefox) */
|
/* Scrollbar (Firefox) */
|
||||||
scrollbar-color: hsl(var(--minecraft-green-2)) hsl(var(--background));
|
scrollbar-color: hsl(var(--minecraft-green-2)) hsl(var(--background));
|
||||||
}
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
@apply bg-background text-foreground;
|
@apply bg-background text-foreground;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Scrollbar (Chrome & Safari) */
|
/* Scrollbar (Chrome & Safari) */
|
||||||
@layer base {
|
@layer base {
|
||||||
::-webkit-scrollbar {
|
::-webkit-scrollbar {
|
||||||
@apply w-1.5;
|
@apply w-1.5;
|
||||||
}
|
}
|
||||||
|
|
||||||
::-webkit-scrollbar-track {
|
::-webkit-scrollbar-track {
|
||||||
@apply bg-inherit;
|
@apply bg-inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
::-webkit-scrollbar-thumb {
|
::-webkit-scrollbar-thumb {
|
||||||
@apply bg-minecraft-green-2 rounded-3xl;
|
@apply bg-minecraft-green-2 rounded-3xl;
|
||||||
}
|
}
|
||||||
|
|
||||||
::-webkit-scrollbar-thumb:hover {
|
::-webkit-scrollbar-thumb:hover {
|
||||||
@apply bg-opacity-80;
|
@apply bg-opacity-80;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,7 @@ import ThemeProvider from "@/provider/theme-provider";
|
|||||||
import type { Metadata, Viewport } from "next";
|
import type { Metadata, Viewport } from "next";
|
||||||
import PlausibleProvider from "next-plausible";
|
import PlausibleProvider from "next-plausible";
|
||||||
import "./globals.css";
|
import "./globals.css";
|
||||||
|
import { ReactElement, ReactNode } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Site metadata & viewport.
|
* Site metadata & viewport.
|
||||||
@ -22,32 +23,32 @@ export const viewport: Viewport = config.viewport;
|
|||||||
* @returns the layout jsx
|
* @returns the layout jsx
|
||||||
*/
|
*/
|
||||||
const RootLayout = ({
|
const RootLayout = ({
|
||||||
children,
|
children,
|
||||||
}: Readonly<{
|
}: Readonly<{
|
||||||
children: React.ReactNode;
|
children: ReactNode;
|
||||||
}>): JSX.Element => {
|
}>): ReactElement => {
|
||||||
const analyticsDomain: string | undefined = config.analyticsDomain;
|
const analyticsDomain: string | undefined = config.analyticsDomain;
|
||||||
return (
|
return (
|
||||||
<html lang="en" className={cn("scroll-smooth", notoSans.className)}>
|
<html lang="en" className={cn("scroll-smooth", notoSans.className)}>
|
||||||
<head>
|
<head>
|
||||||
{analyticsDomain && (
|
{analyticsDomain && (
|
||||||
<PlausibleProvider
|
<PlausibleProvider
|
||||||
domain={analyticsDomain}
|
domain={analyticsDomain}
|
||||||
customDomain="https://analytics.rainnny.club"
|
customDomain="https://analytics.rainnny.club"
|
||||||
selfHosted
|
selfHosted
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</head>
|
</head>
|
||||||
<body className="relative min-h-screen">
|
<body className="relative min-h-screen">
|
||||||
<ThemeProvider attribute="class" defaultTheme="dark">
|
<ThemeProvider attribute="class" defaultTheme="dark">
|
||||||
<TooltipProvider>
|
<TooltipProvider>
|
||||||
<Navbar />
|
<Navbar />
|
||||||
{children}
|
{children}
|
||||||
<Footer />
|
{/*<Footer />*/}
|
||||||
</TooltipProvider>
|
</TooltipProvider>
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
export default RootLayout;
|
export default RootLayout;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { clsx, type ClassValue } from "clsx";
|
import { clsx, type ClassValue } from "clsx";
|
||||||
import { twMerge } from "tailwind-merge";
|
import { twMerge } from "tailwind-merge";
|
||||||
|
|
||||||
export function cn(...inputs: ClassValue[]) {
|
export function cn(...inputs: ClassValue[]): string {
|
||||||
return twMerge(clsx(inputs));
|
return twMerge(clsx(inputs));
|
||||||
}
|
}
|
||||||
|
@ -1,28 +1,32 @@
|
|||||||
import Creeper from "@/components/creeper";
|
import Creeper from "@/components/creeper";
|
||||||
import { minecrafter } from "@/font/fonts";
|
import { minecrafter } from "@/font/fonts";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
|
import { ReactElement } from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The 404 page.
|
* The 404 page.
|
||||||
*
|
*
|
||||||
* @returns the page jsx
|
* @returns the page jsx
|
||||||
*/
|
*/
|
||||||
const NotFoundPage = (): JSX.Element => (
|
const NotFoundPage = (): ReactElement => (
|
||||||
<main className="h-[84vh] flex flex-col gap-3 justify-center items-center pointer-events-none">
|
<main className="h-[84vh] flex flex-col gap-3 justify-center items-center text-center pointer-events-none">
|
||||||
{/* Creeper */}
|
{/* Creeper */}
|
||||||
<Creeper />
|
<Creeper />
|
||||||
|
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<h1
|
<h1
|
||||||
className={cn("text-6xl text-minecraft-green-3", minecrafter.className)}
|
className={cn(
|
||||||
>
|
"text-5xl sm:text-6xl text-minecraft-green-3",
|
||||||
We're Sssssorry
|
minecrafter.className
|
||||||
</h1>
|
)}
|
||||||
|
>
|
||||||
|
We're Sssssorry
|
||||||
|
</h1>
|
||||||
|
|
||||||
{/* Error */}
|
{/* Error */}
|
||||||
<h2 className="text-2xl">
|
<h2 className="text-2xl">
|
||||||
The page you were looking for could not be found.
|
The page you were looking for could not be found.
|
||||||
</h2>
|
</h2>
|
||||||
</main>
|
</main>
|
||||||
);
|
);
|
||||||
export default NotFoundPage;
|
export default NotFoundPage;
|
||||||
|
Reference in New Issue
Block a user