From fe9b50b30eebad03fadf1312520f9912bdcf8665 Mon Sep 17 00:00:00 2001 From: Aidan Date: Thu, 9 Oct 2025 04:12:05 -0400 Subject: [PATCH] feat (v1.0.0): initial refactor and redesign --- .gitignore | 5 +- README.md | 14 +- app/about/page.tsx | 367 +-- app/ai/claude/components/LoadingSkeleton.tsx | 177 -- app/ai/claude/components/PageHeader.tsx | 26 - app/ai/claude/components/RecentSessions.tsx | 37 - app/ai/claude/components/StatsGrid.tsx | 34 - app/ai/claude/components/TokenComposition.tsx | 30 - .../claude/components/TokenTypeBreakdown.tsx | 27 - app/ai/claude/components/utils.ts | 191 -- app/ai/claude/page.tsx | 85 - app/ai/components/AIStack.tsx | 58 +- app/ai/components/FavoriteModels.tsx | 54 +- app/ai/components/FavoriteTools.tsx | 83 +- app/ai/components/TopPick.tsx | 45 +- app/ai/data.tsx | 26 +- app/ai/page.tsx | 19 +- app/ai/theme.ts | 141 + app/ai/types.ts | 7 +- .../{claude => usage}/components/Activity.tsx | 122 +- app/ai/usage/components/LoadingSkeleton.tsx | 242 ++ .../components/ModelUsageCard.tsx | 24 +- app/ai/usage/components/PageHeader.tsx | 72 + app/ai/usage/components/ProviderFilter.tsx | 71 + app/ai/usage/components/RecentSessions.tsx | 55 + app/ai/usage/components/SegmentedControl.tsx | 71 + app/ai/usage/components/StatsGrid.tsx | 48 + app/ai/usage/components/TimeRangeFilter.tsx | 47 + app/ai/usage/components/TokenComposition.tsx | 75 + app/ai/usage/components/TokenType.tsx | 60 + app/ai/{claude => usage}/components/types.ts | 0 app/ai/usage/components/utils.ts | 119 + app/ai/usage/page.tsx | 200 ++ app/api/now-playing/route.ts | 26 + app/contact/page.tsx | 93 +- app/device/[slug]/page.tsx | 52 + app/device/bonito/page.tsx | 199 -- app/device/cheetah/page.tsx | 249 -- app/device/komodo/page.tsx | 247 -- app/device/layout.tsx | 15 + app/docs/DocsPageClient.tsx | 311 +++ app/docs/layout.tsx | 19 + app/docs/page.tsx | 11 + app/domains/[domain]/page.tsx | 49 + app/domains/page.tsx | 159 +- app/globals.css | 25 +- app/i18n.ts | 21 - app/layout.tsx | 52 +- app/manifesto/page.tsx | 21 +- app/page.tsx | 74 +- app/robots.ts | 2 +- app/sitemap.ts | 28 +- bunfig.toml | 2 + components/Footer.tsx | 21 - components/Header.tsx | 473 ---- components/I18nProvider.tsx | 8 - components/device/DeviceHero.tsx | 81 + components/device/DevicePageShell.tsx | 283 ++ components/docs/APIEndpointDoc.tsx | 257 ++ components/docs/CodeBlock.tsx | 198 ++ components/docs/DocsSearch.tsx | 144 + components/docs/DocsSidebar.tsx | 210 ++ components/docs/FunctionDoc.tsx | 296 ++ components/docs/TypeDoc.tsx | 246 ++ components/docs/TypeLink.tsx | 126 + components/domains/DomainCard.tsx | 98 + components/domains/DomainDetails.tsx | 180 ++ components/domains/DomainFilters.tsx | 198 ++ components/domains/DomainTimeline.tsx | 88 + components/icons/DynadotIcon.tsx | 18 + components/icons/GoogleIcon.tsx | 12 + components/icons/KowalskiIcon.tsx | 25 + components/icons/NameIcon.tsx | 19 + components/navigation/Footer.tsx | 310 +++ components/navigation/Header.tsx | 495 ++++ components/navigation/footer-config.ts | 47 + components/navigation/header-config.ts | 165 ++ components/navigation/index.ts | 4 + components/{ => objects}/AnimatedTitle.tsx | 2 +- components/objects/Link.tsx | 32 +- components/objects/PageHeader.tsx | 26 + components/objects/RandomFooterMsg.tsx | 101 +- components/objects/footerMessages.ts | 54 + components/ui/Card.tsx | 72 + components/ui/CardGrid.tsx | 41 + components/ui/PaginatedCardList.tsx | 95 + components/ui/Section.tsx | 32 + components/ui/Surface.tsx | 20 + components/widgets/FeaturedRepos.tsx | 13 +- components/widgets/GitHubStatsImage.tsx | 41 + components/widgets/NowPlaying.tsx | 7 +- i18n.ts | 20 - lib/config/featured-repos.ts | 41 + lib/devices/config.ts | 61 + lib/devices/data.ts | 465 ++++ lib/devices/index.ts | 13 + lib/docs/loader.ts | 48 + lib/docs/parser.ts | 614 +++++ lib/docs/search.ts | 344 +++ lib/docs/types.ts | 225 ++ lib/domains/config.ts | 178 ++ lib/domains/data.ts | 222 ++ lib/domains/utils.ts | 590 ++++ lib/github.ts | 486 ++++ lib/services/ai.service.ts | 919 +++++++ lib/services/device.service.ts | 454 ++++ lib/services/domain.service.ts | 419 +++ lib/services/index.ts | 3 + lib/theme/colors.ts | 306 +++ lib/theme/effects.ts | 310 +++ lib/theme/index.ts | 253 ++ lib/theme/surfaces.ts | 382 +++ lib/types/ai.ts | 362 +++ lib/types/common.ts | 83 + lib/types/device.ts | 517 ++++ lib/types/domain.ts | 380 +++ lib/types/index.ts | 6 + lib/types/navigation.ts | 71 + lib/types/service.ts | 376 +++ lib/utils/formatting.ts | 317 +++ lib/utils/index.ts | 3 + lib/utils/styles.ts | 56 + lib/utils/validation.ts | 302 +++ package.json | 46 +- public/data/cc.json | 2389 +++++++++++------ public/data/domains.json | 72 - public/data/featured.json | 38 - public/img/jm21.png | Bin 0 -> 212860 bytes public/locales/en-US.json | 116 - public/manifest.json | 4 +- tailwind.config.ts | 10 +- tools/ccombine.ts | 1105 +++++++- tsconfig.json | 1 + typedoc.json | 30 + 134 files changed, 17792 insertions(+), 3670 deletions(-) delete mode 100644 app/ai/claude/components/LoadingSkeleton.tsx delete mode 100644 app/ai/claude/components/PageHeader.tsx delete mode 100644 app/ai/claude/components/RecentSessions.tsx delete mode 100644 app/ai/claude/components/StatsGrid.tsx delete mode 100644 app/ai/claude/components/TokenComposition.tsx delete mode 100644 app/ai/claude/components/TokenTypeBreakdown.tsx delete mode 100644 app/ai/claude/components/utils.ts delete mode 100644 app/ai/claude/page.tsx create mode 100644 app/ai/theme.ts rename app/ai/{claude => usage}/components/Activity.tsx (61%) create mode 100644 app/ai/usage/components/LoadingSkeleton.tsx rename app/ai/{claude => usage}/components/ModelUsageCard.tsx (77%) create mode 100644 app/ai/usage/components/PageHeader.tsx create mode 100644 app/ai/usage/components/ProviderFilter.tsx create mode 100644 app/ai/usage/components/RecentSessions.tsx create mode 100644 app/ai/usage/components/SegmentedControl.tsx create mode 100644 app/ai/usage/components/StatsGrid.tsx create mode 100644 app/ai/usage/components/TimeRangeFilter.tsx create mode 100644 app/ai/usage/components/TokenComposition.tsx create mode 100644 app/ai/usage/components/TokenType.tsx rename app/ai/{claude => usage}/components/types.ts (100%) create mode 100644 app/ai/usage/components/utils.ts create mode 100644 app/ai/usage/page.tsx create mode 100644 app/device/[slug]/page.tsx delete mode 100644 app/device/bonito/page.tsx delete mode 100644 app/device/cheetah/page.tsx delete mode 100644 app/device/komodo/page.tsx create mode 100644 app/device/layout.tsx create mode 100644 app/docs/DocsPageClient.tsx create mode 100644 app/docs/layout.tsx create mode 100644 app/docs/page.tsx create mode 100644 app/domains/[domain]/page.tsx delete mode 100644 app/i18n.ts create mode 100644 bunfig.toml delete mode 100644 components/Footer.tsx delete mode 100644 components/Header.tsx delete mode 100644 components/I18nProvider.tsx create mode 100644 components/device/DeviceHero.tsx create mode 100644 components/device/DevicePageShell.tsx create mode 100644 components/docs/APIEndpointDoc.tsx create mode 100644 components/docs/CodeBlock.tsx create mode 100644 components/docs/DocsSearch.tsx create mode 100644 components/docs/DocsSidebar.tsx create mode 100644 components/docs/FunctionDoc.tsx create mode 100644 components/docs/TypeDoc.tsx create mode 100644 components/docs/TypeLink.tsx create mode 100644 components/domains/DomainCard.tsx create mode 100644 components/domains/DomainDetails.tsx create mode 100644 components/domains/DomainFilters.tsx create mode 100644 components/domains/DomainTimeline.tsx create mode 100644 components/icons/DynadotIcon.tsx create mode 100644 components/icons/GoogleIcon.tsx create mode 100644 components/icons/KowalskiIcon.tsx create mode 100644 components/icons/NameIcon.tsx create mode 100644 components/navigation/Footer.tsx create mode 100644 components/navigation/Header.tsx create mode 100644 components/navigation/footer-config.ts create mode 100644 components/navigation/header-config.ts create mode 100644 components/navigation/index.ts rename components/{ => objects}/AnimatedTitle.tsx (95%) create mode 100644 components/objects/PageHeader.tsx create mode 100644 components/objects/footerMessages.ts create mode 100644 components/ui/Card.tsx create mode 100644 components/ui/CardGrid.tsx create mode 100644 components/ui/PaginatedCardList.tsx create mode 100644 components/ui/Section.tsx create mode 100644 components/ui/Surface.tsx create mode 100644 components/widgets/GitHubStatsImage.tsx delete mode 100644 i18n.ts create mode 100644 lib/config/featured-repos.ts create mode 100644 lib/devices/config.ts create mode 100644 lib/devices/data.ts create mode 100644 lib/devices/index.ts create mode 100644 lib/docs/loader.ts create mode 100644 lib/docs/parser.ts create mode 100644 lib/docs/search.ts create mode 100644 lib/docs/types.ts create mode 100644 lib/domains/config.ts create mode 100644 lib/domains/data.ts create mode 100644 lib/domains/utils.ts create mode 100644 lib/github.ts create mode 100644 lib/services/ai.service.ts create mode 100644 lib/services/device.service.ts create mode 100644 lib/services/domain.service.ts create mode 100644 lib/services/index.ts create mode 100644 lib/theme/colors.ts create mode 100644 lib/theme/effects.ts create mode 100644 lib/theme/index.ts create mode 100644 lib/theme/surfaces.ts create mode 100644 lib/types/ai.ts create mode 100644 lib/types/common.ts create mode 100644 lib/types/device.ts create mode 100644 lib/types/domain.ts create mode 100644 lib/types/index.ts create mode 100644 lib/types/navigation.ts create mode 100644 lib/types/service.ts create mode 100644 lib/utils/formatting.ts create mode 100644 lib/utils/index.ts create mode 100644 lib/utils/styles.ts create mode 100644 lib/utils/validation.ts delete mode 100644 public/data/domains.json delete mode 100644 public/data/featured.json create mode 100644 public/img/jm21.png delete mode 100644 public/locales/en-US.json create mode 100644 typedoc.json diff --git a/.gitignore b/.gitignore index 5e89f8c..e8cbf5a 100644 --- a/.gitignore +++ b/.gitignore @@ -48,4 +48,7 @@ bun.lockb .vscode/ # webstorm -.idea/ \ No newline at end of file +.idea/ + +# docs +public/docs \ No newline at end of file diff --git a/README.md b/README.md index 29ae9d3..7a8c28a 100644 --- a/README.md +++ b/README.md @@ -17,10 +17,16 @@ Just create a `.env` file with the below variables, run `docker compose -d --bui ## Environment Variables -| Variable | Required? | Description | -|----------------------|-----------|-------------------------------------------------------------------------------------| -| `LISTENBRAINZ_TOKEN` | No | Get this from your ListenBrainz [user settings](https://listenbrainz.org/settings/) | -| `LASTFM_API_KEY` | Yes | Get this from your Last.fm [API account](https://www.last.fm/api/account/create) | +| Variable | Required? | Description | +|------------------------|-----------|----------------------------------------------------------------------------------------------------------| +| `LASTFM_API_KEY` | Yes | Get this from your Last.fm [API account](https://www.last.fm/api/account/create) | +| `LISTENBRAINZ_TOKEN` | No | Get this from your ListenBrainz [user settings](https://listenbrainz.org/settings/) | +| `GITHUB_PROJECTS_USER` | No | GitHub username to display in the footer projects list (defaults to `ihatenodejs`) | +| `GITHUB_USERNAME` | No | Fallback GitHub username if `GITHUB_PROJECTS_USER` is not set | +| `GITHUB_PROJECTS_PAT` | No | GitHub personal access token used to increase API limits for the footer projects list | +| `GITHUB_PAT` | No | Fallback GitHub personal access token if `GITHUB_PROJECTS_PAT` is not set | +| `PORT` | No | Server port (defaults to `3000`) | +| `NODE_ENV` | No | Environment mode (`production` or `development`, automatically set by deployment platform) | ## MusicBrainz diff --git a/app/about/page.tsx b/app/about/page.tsx index 476d8e7..bf32410 100644 --- a/app/about/page.tsx +++ b/app/about/page.tsx @@ -1,232 +1,159 @@ -"use client" - -import Header from '@/components/Header' -import Footer from '@/components/Footer' import Link from '@/components/objects/Link' import Button from '@/components/objects/Button' import FeaturedRepos from '@/components/widgets/FeaturedRepos' -import Image from 'next/image' -import { useState } from 'react' +import GitHubStatsImage from '@/components/widgets/GitHubStatsImage' +import PageHeader from '@/components/objects/PageHeader' +import { Card } from '@/components/ui/Card' +import { CardGrid } from '@/components/ui/CardGrid' import { SiGoogle } from 'react-icons/si' import { TbUserHeart } from 'react-icons/tb' -import { useTranslation } from 'react-i18next' -import { cn } from '@/lib/utils' +import { getFeaturedReposWithMetrics } from '@/lib/github' -export default function About() { - const { t } = useTranslation() - const [imageError, setImageError] = useState(false) - const mainStrings: string[][] = [ - t('about.projects', { returnObjects: true }) as string[], - t('about.hobbies', { returnObjects: true }) as string[], - t('about.devices', { returnObjects: true }) as string[], - t('about.contributions', { returnObjects: true }) as string[], - t('about.featuredProjects', { returnObjects: true }) as string[] - ] +const getGitHubUsername = () => { + return process.env.GITHUB_PROJECTS_USER ?? process.env.GITHUB_USERNAME ?? 'ihatenodejs' +} - const mainSections = [ - t('about.sections.projects'), - t('about.sections.hobbies'), - t('about.sections.devices'), - t('about.sections.contributions'), - t('about.sections.featuredProjects') - ] - return ( -
-
-
-
-
- +interface ContentSection { + title: string + content: React.ReactElement +} + +export default async function About() { + const featuredProjects = await getFeaturedReposWithMetrics() + const githubUsername = getGitHubUsername() + + const sections: ContentSection[] = [ + { + title: "Projects", + content: ( + <> +

+ I have worked on countless projects over the past five years, for the most part. I started learning to code with Python when I was seven and my interest has only evolved from there. I got into web development due to my uncle, who taught my how to write my first lines of HTML. +

+

+ Recently, I have been involved in developing several projects, especially with TypeScript, which is my new favorite language as of a year ago. My biggest project currently is p0ntus, a free service provider for privacy-focused individuals. +

+

+ You will also come to find that I have an addiction to Docker! Almost every project I've made is able to be run in Docker. +

+

+ Me and my developer friends operate an organization called ABOCN, where we primarily maintain a Telegram bot called Kowalski. You can find it on Telegram as @KowalskiNodeBot. +

+

+ I have learned system administration from the past three years of learning Linux for practical use and fun. I currently operate four servers running in the cloud, ran out of Canada, Germany, and the United States. +

+

+ I own a channel called PontusHub on Telegram, where I post updates about my projects, along with commentary and info about my projects related to the Android rooting community. +

+ + ) + }, + { + title: "Hobbies", + content: ( + <> +

+ When I'm not programming, I can typically be found distro hopping or flashing a new ROM to my phone. I also spend a lot of time spreading Next.js and TypeScript propaganda to JavaScript developers. +

+

+ I consider maintaining my devices as a hobby as well, as I devote a lot of time to it. I genuinely enjoy installing Arch, Gentoo, and NixOS frequently, and flashing new ROMs to the phones I own. +

+

+ I am frequently active on my Forgejo server and GitHub, and aim to make daily contributions. I am a big fan of open source software and public domain software (which most of my repos are licensed under). In fact, the website you're currently on is free and open source. It's even under the public domain! +

+

+ When I touch grass, I prefer to walk on the streets, especially in Boston, Massachusetts. I also used to swim competitively, though it has turned into to a casual hobby over time. +

+

+ Editing Wikipedia has also been a good pastime for me, and I have been editing for a year and a half now. As of writing, I have made 6.1k edits to the English Wikipedia. I am also an AfC reviewer, new page reviewer, and rollbacker. You can find me on Wikipedia as OnlyNano. +

+ + ) + }, + { + title: "Devices", + content: ( + <> +

Mobile Devices

+

+ I use a Google Pixel 9 Pro XL (komodo) as my daily driver. It runs Android 16 and is proudly rooted with KernelSU-Next. +

+

+ My previous phone, the Google Pixel 7 Pro (cheetah), is still in use as my secondary WiFi-only device. It runs Android 16 and is proudly rooted with KernelSU-Next. +

+

+ I also have a Google Pixel 3a XL (bonito) which I use as a tertiary device. It runs LineageOS 22.2 and is rooted with Magisk. +

+
+ + +
-

{t('about.title')}

-
+

Laptops

+

+ I currently daily-drive with a 16-inch MacBook Pro with an M4 Max, 64GB of memory, 2TB of storage, 16 core CPU, and a 40 core GPU. +

+

+ I use a Lenovo Thinkpad T470s with macOS Sequoia (using OpenCore) as my "side piece," if you will. I've had it for about a year now, and it's been a great experience. +

+

+ I also own two MacBook Airs (2015 and 2013 base models) and an HP Chromebook, used as secondary devices. The 2013 runs unsupported macOS Sequoia Beta, the 2015 runs Xubuntu, and the Chromebook runs Arch Linux. +

+ + ) + }, + { + title: "Contributions", + content: ( + <> +

+ Most of my repositories have migrated to p0ntus git. My username is aidan. You can find me on GitHub as {githubUsername}. +

+ + + ) + }, + { + title: "Featured Projects", + content: ( + <> +

+ Here's just four of my top projects. Star and fork counts are fetched in real-time from both GitHub and Forgejo APIs. +

+ + + ) + } + ] -
- {mainStrings.map((section, index) => { - if (mainSections[index] === t('about.sections.featuredProjects')) { - return ( -
-

{mainSections[index]}

- {section.map((text, index) => ( -

- {text} -

- ))} - -
- ) - } else if (mainSections[index] === t('about.sections.contributions')) { - return ( -
-

{mainSections[index]}

- {section.map((text, index) => ( -

- {text.split(/(ihatenodejs|p0ntus git|aidan)/).map((part, i) => { - if (part === 'ihatenodejs') { - return ihatenodejs - } - if (part === 'p0ntus git') { - return p0ntus git - } - if (part === 'aidan') { - return aidan - } - return part - })} -

- ))} - {!imageError && ( -
- ihatenodejs's Stats setImageError(true)} - loading="eager" - priority - unoptimized - className="max-w-full h-auto" - /> - ihatenodejs's Top Languages setImageError(true)} - loading="eager" - priority - unoptimized - className="max-w-full h-auto" - /> -
- )} -
- ) - } else if (mainSections[index] === t('about.sections.devices')) { - return ( -
-

{mainSections[index]}

- {Object.entries(section).map(([key, value], index) => ( -
-

{key}

- {(value as unknown as string[]).map((text: string, index: number) => ( -

- {text.split(/(KernelSU-Next|LineageOS 22.2|Android 16|Xubuntu)/).map((part, i) => { - if (part === 'KernelSU-Next') { - return KernelSU-Next - } - if (part === 'LineageOS 22.2') { - return LineageOS 22.2 - } - if (part === 'Android 16') { - return Android 16 - } - if (part === 'OpenCore') { - return OpenCore - } - if (part === 'Xubuntu') { - return Xubuntu - } - return part - })} -

- ))} - {key === "Mobile Devices" && ( -
- - - -
- )} -
- ))} -
- ) - } else if (mainSections[index] === t('about.sections.hobbies')) { - return ( -
-

{mainSections[index]}

- {section.map((text, index) => ( -

- {text.split(/(my Forgejo server|my phone|AfC|OnlyNano)/).map((part, i) => { - if (part === 'my Forgejo server') { - return my Forgejo server - } - if (part === 'my phone') { - return my phone - } - if (part === 'AfC') { - return AfC - } - if (part === 'OnlyNano') { - return OnlyNano - } - return part - })} -

- ))} -
- ) - } else if (mainSections[index] === t('about.sections.projects')) { - return ( -
-

{mainSections[index]}

- {section.map((text, index) => ( -

- {text.split(/(p0ntus|PontusHub|ABOCN|Kowalski|@KowalskiNodeBot)/).map((part, i) => { - if (part === 'p0ntus') { - return p0ntus - } - if (part === 'PontusHub') { - return PontusHub - } - if (part === 'ABOCN') { - return ABOCN - } - if (part === 'Kowalski') { - return Kowalski - } - if (part === '@KowalskiNodeBot') { - return @KowalskiNodeBot - } - return part - })} -

- ))} -
- ) - } else { - return ( -
-

{mainSections[index]}

- {section.map((text, index) => ( -

- {text} -

- ))} -
- ) - } - })} -
-
-