* feat: update prettier configuration for print width * chore: apply code formatting to entire repository * fix: remove build files * fix: format issue --------- Co-authored-by: Meier Lukas <meierschlumpf@gmail.com>
171 lines
5.3 KiB
TypeScript
171 lines
5.3 KiB
TypeScript
import Image from "next/image";
|
|
import {
|
|
Accordion,
|
|
AccordionControl,
|
|
AccordionItem,
|
|
AccordionPanel,
|
|
AspectRatio,
|
|
Avatar,
|
|
Card,
|
|
Center,
|
|
Flex,
|
|
Group,
|
|
List,
|
|
ListItem,
|
|
Stack,
|
|
Text,
|
|
Title,
|
|
} from "@mantine/core";
|
|
import { IconLanguage, IconLibrary, IconUsers } from "@tabler/icons-react";
|
|
import { setStaticParamsLocale } from "next-international/server";
|
|
|
|
import { getScopedI18n, getStaticParams } from "@homarr/translation/server";
|
|
|
|
import { createMetaTitle } from "~/metadata";
|
|
import { getPackageAttributesAsync } from "~/versions/package-reader";
|
|
import contributorsData from "../../../../../../../static-data/contributors.json";
|
|
import translatorsData from "../../../../../../../static-data/translators.json";
|
|
import logo from "../../../../../public/logo/logo.png";
|
|
import classes from "./about.module.css";
|
|
|
|
export async function generateMetadata() {
|
|
const t = await getScopedI18n("management");
|
|
|
|
return {
|
|
title: createMetaTitle(t("metaTitle")),
|
|
};
|
|
}
|
|
|
|
interface PageProps {
|
|
params: {
|
|
locale: string;
|
|
};
|
|
}
|
|
|
|
export default async function AboutPage({ params: { locale } }: PageProps) {
|
|
setStaticParamsLocale(locale);
|
|
const t = await getScopedI18n("management.page.about");
|
|
const attributes = await getPackageAttributesAsync();
|
|
return (
|
|
<div>
|
|
<Center w="100%">
|
|
<Group py="lg">
|
|
<Image src={logo} width={100} height={100} alt="" />
|
|
<Stack gap={0}>
|
|
<Title order={1} tt="uppercase">
|
|
Homarr
|
|
</Title>
|
|
<Title order={2}>{t("version", { version: attributes.version })}</Title>
|
|
</Stack>
|
|
</Group>
|
|
</Center>
|
|
<Text mb="xl">{t("text")}</Text>
|
|
|
|
<Accordion defaultValue="contributors" variant="filled" radius="md">
|
|
<AccordionItem value="contributors">
|
|
<AccordionControl icon={<IconUsers size="1rem" />}>
|
|
<Stack gap={0}>
|
|
<Text>{t("accordion.contributors.title")}</Text>
|
|
<Text size="sm" c="dimmed">
|
|
{t("accordion.contributors.subtitle", {
|
|
count: contributorsData.length,
|
|
})}
|
|
</Text>
|
|
</Stack>
|
|
</AccordionControl>
|
|
<AccordionPanel>
|
|
<Flex wrap="wrap" gap="xs">
|
|
{contributorsData.map((contributor) => (
|
|
<GenericContributorLinkCard
|
|
key={contributor.login}
|
|
link={`https://github.com/${contributor.login}`}
|
|
image={contributor.avatar_url}
|
|
name={contributor.login}
|
|
/>
|
|
))}
|
|
</Flex>
|
|
</AccordionPanel>
|
|
</AccordionItem>
|
|
<AccordionItem value="translators">
|
|
<AccordionControl icon={<IconLanguage size="1rem" />}>
|
|
<Stack gap={0}>
|
|
<Text>{t("accordion.translators.title")}</Text>
|
|
<Text size="sm" c="dimmed">
|
|
{t("accordion.translators.subtitle", {
|
|
count: translatorsData.length,
|
|
})}
|
|
</Text>
|
|
</Stack>
|
|
</AccordionControl>
|
|
<AccordionPanel>
|
|
<Flex wrap="wrap" gap="xs">
|
|
{translatorsData.map((translator) => (
|
|
<GenericContributorLinkCard
|
|
key={translator.username}
|
|
link={`https://crowdin.com/profile/${translator.username}`}
|
|
image={translator.avatarUrl}
|
|
name={translator.username}
|
|
/>
|
|
))}
|
|
</Flex>
|
|
</AccordionPanel>
|
|
</AccordionItem>
|
|
<AccordionItem value="libraries">
|
|
<AccordionControl icon={<IconLibrary size="1rem" />}>
|
|
<Stack gap={0}>
|
|
<Text>{t("accordion.libraries.title")}</Text>
|
|
<Text size="sm" c="dimmed">
|
|
{t("accordion.libraries.subtitle", {
|
|
count: Object.keys(attributes.dependencies).length,
|
|
})}
|
|
</Text>
|
|
</Stack>
|
|
</AccordionControl>
|
|
<AccordionPanel>
|
|
<List>
|
|
{Object.entries(attributes.dependencies)
|
|
.sort(([key1], [key2]) => key1.localeCompare(key2))
|
|
.map(([key, value]) => (
|
|
<ListItem key={key}>
|
|
{value.includes("workspace:") ? (
|
|
<Text>{key}</Text>
|
|
) : (
|
|
<a href={`https://www.npmjs.com/package/${key}`}>{key}</a>
|
|
)}
|
|
</ListItem>
|
|
))}
|
|
</List>
|
|
</AccordionPanel>
|
|
</AccordionItem>
|
|
</Accordion>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
interface GenericContributorLinkCardProps {
|
|
name: string;
|
|
link: string;
|
|
image: string;
|
|
}
|
|
|
|
const GenericContributorLinkCard = ({ name, image, link }: GenericContributorLinkCardProps) => {
|
|
return (
|
|
<AspectRatio ratio={1}>
|
|
<Card className={classes.contributorCard} component="a" href={link} target="_blank" w={100}>
|
|
<Stack align="center">
|
|
<Avatar src={image} alt={name} size={40} display="block" />
|
|
<Text lineClamp={1} size="sm">
|
|
{name}
|
|
</Text>
|
|
</Stack>
|
|
</Card>
|
|
</AspectRatio>
|
|
);
|
|
};
|
|
|
|
export function generateStaticParams() {
|
|
return getStaticParams();
|
|
}
|
|
|
|
export const dynamic = "force-static";
|