design/feat: stringify about page, re-layout sections into cards, add hover effects (+home page)
This commit is contained in:
		
							parent
							
								
									b4330d8df8
								
							
						
					
					
						commit
						a3c7b47935
					
				
					 5 changed files with 225 additions and 96 deletions
				
			
		|  | @ -8,70 +8,79 @@ import FeaturedRepos from '@/components/widgets/FeaturedRepos' | ||||||
| import Image from 'next/image' | import Image from 'next/image' | ||||||
| import { useState } from 'react' | import { useState } from 'react' | ||||||
| import { User, Smartphone } from 'lucide-react' | import { User, Smartphone } from 'lucide-react' | ||||||
|  | import { useTranslation } from 'react-i18next' | ||||||
|  | import { cn } from '@/lib/utils' | ||||||
| 
 | 
 | ||||||
| export default function About() { | export default function About() { | ||||||
|  |   const { t } = useTranslation() | ||||||
|   const [imageError, setImageError] = useState(false) |   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 mainSections = [ | ||||||
|  |     t('about.sections.projects'), | ||||||
|  |     t('about.sections.hobbies'), | ||||||
|  |     t('about.sections.devices'), | ||||||
|  |     t('about.sections.contributions'), | ||||||
|  |     t('about.sections.featuredProjects') | ||||||
|  |   ] | ||||||
|   return ( |   return ( | ||||||
|     <div className="min-h-screen flex flex-col"> |     <div className="min-h-screen flex flex-col"> | ||||||
|       <Header /> |       <Header /> | ||||||
|       <main className="grow container mx-auto px-4 py-12"> |       <main className="text-center py-12"> | ||||||
|         <div className="max-w-2xl mx-auto text-center"> |  | ||||||
|         <div className='mb-6 flex justify-center'> |         <div className='mb-6 flex justify-center'> | ||||||
|           <User size={60} /> |           <User size={60} /> | ||||||
|         </div> |         </div> | ||||||
|         <h1 className="text-4xl font-bold my-2 text-center text-gray-200" style={{ textShadow: '0 0 10px rgba(255, 255, 255, 0.5)' }}> |         <h1 className="text-4xl font-bold my-2 text-center text-gray-200" style={{ textShadow: '0 0 10px rgba(255, 255, 255, 0.5)' }}> | ||||||
|             About Me |           {t('about.title')} | ||||||
|         </h1> |         </h1> | ||||||
|           <div className="px-6 pt-6"> | 
 | ||||||
|             <p className="text-gray-300 mb-4"> |         <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 p-4"> | ||||||
|               Hey there! I'm Aidan, a web developer and student, and this is my website. I'm passionate about web development (although I'm not great with design), especially with Next.js and APIs! I enjoy working with both backend and frontend. |           {mainStrings.map((section, index) => { | ||||||
|  |             if (mainSections[index] === t('about.sections.featuredProjects')) { | ||||||
|  |               return ( | ||||||
|  |                 <section key={index} className="p-8 border-2 border-gray-700 rounded-lg col-span-2 hover:border-gray-600 transition-colors duration-300"> | ||||||
|  |                   <h2 className="text-2xl font-semibold mb-4 text-gray-200">{mainSections[index]}</h2> | ||||||
|  |                   {section.map((text, index) => ( | ||||||
|  |                     <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  |                       {text} | ||||||
|                     </p> |                     </p> | ||||||
|             <h2 className="text-2xl font-semibold mb-4 mt-12 text-gray-200">Projects</h2> |                   ))} | ||||||
|             <p className="text-gray-300 mb-4"> |                   <FeaturedRepos className="mt-4" /> | ||||||
|               I have worked on countless projects over the past five years, for the most part. I have been learning to program in Python since I was seven and have evolved from there. I got into web development due to my uncle, who taught my how to write my first lines of HTML. |                 </section> | ||||||
|  |               ) | ||||||
|  |             } else if (mainSections[index] === t('about.sections.contributions')) { | ||||||
|  |               return ( | ||||||
|  |                 <section key={index} className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|  |                   <h2 className="text-2xl font-semibold mb-4 text-gray-200">{mainSections[index]}</h2> | ||||||
|  |                   {section.map((text, index) => ( | ||||||
|  |                     <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  |                       {text.split(/(ihatenodejs|LibreCloud Git|aidan)/).map((part, i) => { | ||||||
|  |                         if (part === 'ihatenodejs') { | ||||||
|  |                           return <Link key={i} href="https://github.com/ihatenodejs/">GitHub</Link> | ||||||
|  |                         } | ||||||
|  |                         if (part === 'LibreCloud Git') { | ||||||
|  |                           return <Link key={i} href="https://git.pontusmail.org/">LibreCloud Git</Link> | ||||||
|  |                         } | ||||||
|  |                         if (part === 'aidan') { | ||||||
|  |                           return <Link key={i} href="https://git.pontusmail.org/aidan/">aidan</Link> | ||||||
|  |                         } | ||||||
|  |                         return part | ||||||
|  |                       })} | ||||||
|                     </p> |                     </p> | ||||||
|             <p className="text-gray-300 mb-4"> |                   ))} | ||||||
|               Recently, I have been involved in developing several projects, especially with Node.js, my new favorite language as of a year ago. My biggest project is <Link href="https://librecloud.cc">LibreCloud</Link>, a free service provider for individuals. |  | ||||||
|             </p> |  | ||||||
|             <p className="text-gray-300 mb-4"> |  | ||||||
|               In terms of system administration, I have developed my skills over the past three years of learning Linux for fun. I currently operate three servers running in the cloud, which run out of Germany and the United States. |  | ||||||
|             </p> |  | ||||||
|             <h2 className="text-2xl font-semibold mb-4 mt-12 text-gray-200">Hobbies</h2> |  | ||||||
|             <p className="text-gray-300 mb-4"> |  | ||||||
|               When I'm not programming, I can typically be found installing another Linux distro on my laptop or flashing a new ROM to my phone. I am also a passionate writer and I like to write creatively in my free time. |  | ||||||
|             </p> |  | ||||||
|             <p className="text-gray-300 mb-4"> |  | ||||||
|               I consider maintaining my technology as a hobby as well, as I devote a lot of time to it. I currently run Gentoo Linux on my Thinkpad T470s, which does not use a single bin package. I am very proud of this laptop, despite it's constant need for compiling updates. |  | ||||||
|             </p> |  | ||||||
|             <p className="text-gray-300 mb-4"> |  | ||||||
|               I am almost always active on <Link href="https://git.pontusmail.org/">my Gitea instance</Link> and GitHub and make daily contributions to several of my repositories. 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! |  | ||||||
|             </p> |  | ||||||
|             <p className="text-gray-300 mb-4"> |  | ||||||
|               My Google Pixel 7 Pro (cheetah) runs LineageOS 22.1, and has been one of my favorite additions to my life. It is proudly rooted with KernelSU-Next. It has suffered one drop to it's back on a tile floor. |  | ||||||
|             </p> |  | ||||||
|           </div> |  | ||||||
|           <div className="mt-12"> |  | ||||||
|             <h2 className="text-2xl font-semibold mb-4 text-gray-200">Devices</h2> |  | ||||||
|             <p className="text-gray-300 mb-4">You can learn more about the devices I use daily with the pages below:</p> |  | ||||||
|             <Button |  | ||||||
|               href="/phone" |  | ||||||
|               label="My Phone" |  | ||||||
|               icon={Smartphone} |  | ||||||
|             /> |  | ||||||
|           </div> |  | ||||||
|           <div className="mt-12"> |  | ||||||
|             <h2 className="text-2xl font-semibold mb-4 text-gray-200">My Gitea/GitHub Contributions</h2> |  | ||||||
|             <p className="text-gray-300 mb-4">Most of my repositories have migrated to <Link href="https://git.pontusmail.org/">LibreCloud Git</Link>. My username is <Link href="https://git.pontusmail.org/aidan/">aidan</Link>.</p> |  | ||||||
|             <p className="text-gray-300 mb-4">You can find me on GitHub as <Link href="https://github.com/ihatenodejs/">ihatenodejs</Link>.</p> |  | ||||||
|                   {!imageError && ( |                   {!imageError && ( | ||||||
|               <div className="flex flex-col md:flex-row justify-center gap-4"> |                     <div className="flex flex-col justify-center items-center w-full mt-4 gap-4"> | ||||||
|                       <Image  |                       <Image  | ||||||
|                         src="https://github-readme-stats.vercel.app/api?username=ihatenodejs&theme=dark&show_icons=true&hide_border=true&count_private=true"  |                         src="https://github-readme-stats.vercel.app/api?username=ihatenodejs&theme=dark&show_icons=true&hide_border=true&count_private=true"  | ||||||
|                         alt="ihatenodejs's Stats" |                         alt="ihatenodejs's Stats" | ||||||
|                   width={500}  |                         width={420}  | ||||||
|                         height={200} |                         height={200} | ||||||
|                   className="w-full md:w-1/2"  |  | ||||||
|                         onError={() => setImageError(true)} |                         onError={() => setImageError(true)} | ||||||
|                         loading="eager" |                         loading="eager" | ||||||
|                         priority |                         priority | ||||||
|  | @ -80,9 +89,8 @@ export default function About() { | ||||||
|                       <Image  |                       <Image  | ||||||
|                         src="https://github-readme-stats.vercel.app/api/top-langs/?username=ihatenodejs&theme=dark&show_icons=true&hide_border=true&layout=compact"  |                         src="https://github-readme-stats.vercel.app/api/top-langs/?username=ihatenodejs&theme=dark&show_icons=true&hide_border=true&layout=compact"  | ||||||
|                         alt="ihatenodejs's Top Languages" |                         alt="ihatenodejs's Top Languages" | ||||||
|                   width={500}  |                         width={300} | ||||||
|                         height={200} |                         height={200} | ||||||
|                   className="w-full md:w-1/3"  |  | ||||||
|                         onError={() => setImageError(true)} |                         onError={() => setImageError(true)} | ||||||
|                         loading="eager" |                         loading="eager" | ||||||
|                         priority |                         priority | ||||||
|  | @ -90,12 +98,88 @@ export default function About() { | ||||||
|                       /> |                       /> | ||||||
|                     </div> |                     </div> | ||||||
|                   )} |                   )} | ||||||
|  |                 </section> | ||||||
|  |               ) | ||||||
|  |             } else if (mainSections[index] === t('about.sections.devices')) { | ||||||
|  |               return ( | ||||||
|  |                 <section key={index} className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|  |                   <h2 className="text-2xl font-semibold mb-4 text-gray-200">{mainSections[index]}</h2> | ||||||
|  |                   {Object.entries(section).map(([key, value], index) => ( | ||||||
|  |                     <div key={index}> | ||||||
|  |                       <h3 className={cn("text-xl font-semibold mb-2 text-gray-200", key === "Laptop" && "mt-4")}>{key}</h3> | ||||||
|  |                       {(value as unknown as string[]).map((text: string, index: number) => ( | ||||||
|  |                         <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  |                           {text.split(/(KernelSU-Next|LineageOS microG)/).map((part, i) => { | ||||||
|  |                             if (part === 'KernelSU-Next') { | ||||||
|  |                               return <Link key={i} href="https://github.com/KernelSU-Next/KernelSU-Next">KernelSU-Next</Link> | ||||||
|  |                             } | ||||||
|  |                             if (part === 'LineageOS microG') { | ||||||
|  |                               return <Link key={i} href="https://lineage.microg.org/">LineageOS microG</Link> | ||||||
|  |                             } | ||||||
|  |                             return part | ||||||
|  |                           })} | ||||||
|  |                         </p> | ||||||
|  |                       ))} | ||||||
|  |                       {key === "Phone" && ( | ||||||
|  |                         <Button | ||||||
|  |                           href="/phone" | ||||||
|  |                           label="My Phone" | ||||||
|  |                           icon={Smartphone} | ||||||
|  |                           className="mt-4" | ||||||
|  |                         /> | ||||||
|  |                       )} | ||||||
|                     </div> |                     </div> | ||||||
|           <div className="mt-12"> |                   ))} | ||||||
|             <h2 className="text-2xl font-semibold mb-4 text-gray-200">Featured Projects</h2> |                 </section> | ||||||
|             <p className="text-gray-300 mb-6">Here's just four of my top projects. Star and fork counts are manually updated and count both Gitea and GitHub.</p> |               ) | ||||||
|             <FeaturedRepos /> |             } else if (mainSections[index] === t('about.sections.hobbies')) { | ||||||
|           </div> |               return ( | ||||||
|  |                 <section key={index} className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|  |                   <h2 className="text-2xl font-semibold mb-4 text-gray-200">{mainSections[index]}</h2> | ||||||
|  |                   {section.map((text, index) => ( | ||||||
|  |                     <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  |                       {text.split(/(my Gitea instance|my phone)/).map((part, i) => { | ||||||
|  |                         if (part === 'my Gitea instance') { | ||||||
|  |                           return <Link key={i} href="https://git.pontusmail.org/">my Gitea instance</Link> | ||||||
|  |                         } | ||||||
|  |                         if (part === 'my phone') { | ||||||
|  |                           return <Link key={i} href="/phone">my phone</Link> | ||||||
|  |                         } | ||||||
|  |                         return part | ||||||
|  |                       })} | ||||||
|  |                     </p> | ||||||
|  |                   ))} | ||||||
|  |                 </section> | ||||||
|  |               ) | ||||||
|  |             } else if (mainSections[index] === t('about.sections.projects')) { | ||||||
|  |               return ( | ||||||
|  |                 <section key={index} className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|  |                   <h2 className="text-2xl font-semibold mb-4 text-gray-200">{mainSections[index]}</h2> | ||||||
|  |                   {section.map((text, index) => ( | ||||||
|  |                     <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  |                       {text.split(/(LibreCloud)/).map((part, i) => { | ||||||
|  |                         if (part === 'LibreCloud') { | ||||||
|  |                           return <Link key={i} href="https://librecloud.cc/">LibreCloud</Link> | ||||||
|  |                         } | ||||||
|  |                         return part | ||||||
|  |                       })} | ||||||
|  |                     </p> | ||||||
|  |                   ))} | ||||||
|  |                 </section> | ||||||
|  |               ) | ||||||
|  |             } else { | ||||||
|  |               return ( | ||||||
|  |                 <section key={index} className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|  |                   <h2 className="text-2xl font-semibold mb-4 text-gray-200">{mainSections[index]}</h2> | ||||||
|  |                   {section.map((text, index) => ( | ||||||
|  |                     <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  |                       {text} | ||||||
|  |                     </p> | ||||||
|  |                   ))} | ||||||
|  |                 </section> | ||||||
|  |               ) | ||||||
|  |             } | ||||||
|  |           })} | ||||||
|         </div> |         </div> | ||||||
|       </main> |       </main> | ||||||
|       <Footer /> |       <Footer /> | ||||||
|  |  | ||||||
							
								
								
									
										10
									
								
								app/page.tsx
									
										
									
									
									
								
							
							
						
						
									
										10
									
								
								app/page.tsx
									
										
									
									
									
								
							|  | @ -36,19 +36,19 @@ export default function Home() { | ||||||
|             alt="My Profile Picture" |             alt="My Profile Picture" | ||||||
|             width={150} |             width={150} | ||||||
|             height={150} |             height={150} | ||||||
|             className="rounded-full mx-auto mb-6 border-4 border-gray-700" |             className="rounded-full mx-auto mb-6 border-4 border-gray-700 hover:border-gray-600 transition-colors duration-300" | ||||||
|           /> |           /> | ||||||
|           <h1 className="text-4xl font-bold mb-2 text-gray-100 glow">{t('home.profile.name')}</h1> |           <h1 className="text-4xl font-bold mb-2 text-gray-100 glow">{t('home.profile.name')}</h1> | ||||||
|           <p className="text-gray-400 text-xl">{t('home.profile.description')}</p> |           <p className="text-gray-400 text-xl">{t('home.profile.description')}</p> | ||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
|         <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 p-4"> |         <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 p-4"> | ||||||
|           <div className="p-8 border-2 border-gray-700 rounded-lg"> |           <div className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|             <LastPlayed /> |             <LastPlayed /> | ||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           {mainSections.map((section, secIndex) => ( |           {mainSections.map((section, secIndex) => ( | ||||||
|             <section key={secIndex} className="p-8 border-2 border-gray-700 rounded-lg"> |             <section key={secIndex} className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|               <h2 className="text-2xl font-semibold mb-4 text-gray-200">{section}</h2> |               <h2 className="text-2xl font-semibold mb-4 text-gray-200">{section}</h2> | ||||||
|               {mainStrings[secIndex].map((text: string, index: number) => ( |               {mainStrings[secIndex].map((text: string, index: number) => ( | ||||||
|                 <p key={index} className="text-gray-300 leading-relaxed mt-2"> |                 <p key={index} className="text-gray-300 leading-relaxed mt-2"> | ||||||
|  | @ -67,7 +67,7 @@ export default function Home() { | ||||||
|             </section> |             </section> | ||||||
|           ))} |           ))} | ||||||
| 
 | 
 | ||||||
|           <section id="contact" className="p-8 border-2 border-gray-700 rounded-lg"> |           <section id="contact" className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|             <h2 className="text-2xl font-semibold mb-4 text-gray-200">{t('home.contact.title')}</h2> |             <h2 className="text-2xl font-semibold mb-4 text-gray-200">{t('home.contact.title')}</h2> | ||||||
|             <p className="text-gray-300 mb-6">{t('home.contact.description')}</p> |             <p className="text-gray-300 mb-6">{t('home.contact.description')}</p> | ||||||
|             <Button |             <Button | ||||||
|  | @ -77,7 +77,7 @@ export default function Home() { | ||||||
|             /> |             /> | ||||||
|           </section> |           </section> | ||||||
| 
 | 
 | ||||||
|           <section id="donation" className="p-8 border-2 border-gray-700 rounded-lg"> |           <section id="donation" className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||||
|             <h2 className="text-2xl font-semibold mb-4 text-gray-200">{t('home.donation.title')}</h2> |             <h2 className="text-2xl font-semibold mb-4 text-gray-200">{t('home.donation.title')}</h2> | ||||||
|             <p className="text-gray-300 mb-6">{t('home.donation.description')}</p> |             <p className="text-gray-300 mb-6">{t('home.donation.description')}</p> | ||||||
|              |              | ||||||
|  |  | ||||||
|  | @ -1,24 +1,30 @@ | ||||||
| import React from 'react' | import React from 'react' | ||||||
| import Link from 'next/link' | import Link from 'next/link' | ||||||
|  | import { cn } from '@/lib/utils' | ||||||
| 
 | 
 | ||||||
| interface ButtonProps { | interface ButtonProps { | ||||||
|   href: string; |   href: string | ||||||
|   label: string; |   label: string | ||||||
|   icon?: React.ElementType; |   icon?: React.ElementType | ||||||
|   target?: string; |   target?: string | ||||||
|  |   className?: string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const Button: React.FC<ButtonProps> = ({ href, label, icon, target }) => { | const Button: React.FC<ButtonProps> = ({ href, label, icon, target, className }) => { | ||||||
|   return ( |   return ( | ||||||
|     <Link |     <Link | ||||||
|       href={href} |       href={href} | ||||||
|       className="inline-flex items-center bg-gray-800 text-white font-bold py-2 px-4 rounded-sm shadow-md transition-all duration-300 ease-in-out hover:bg-gray-700 hover:shadow-lg hover:-translate-y-0.5 focus:outline-hidden focus:ring-2 focus:ring-offset-2 focus:ring-gray-500" |       className={className ? ( | ||||||
|  |         cn("inline-flex items-center bg-gray-800 text-white font-bold py-2 px-4 rounded-sm shadow-md transition-all duration-300 ease-in-out hover:bg-gray-700 hover:shadow-lg hover:-translate-y-0.5 focus:outline-hidden focus:ring-2 focus:ring-offset-2 focus:ring-gray-500", className) | ||||||
|  |       ) : ( | ||||||
|  |         "inline-flex items-center bg-gray-800 text-white font-bold py-2 px-4 rounded-sm shadow-md transition-all duration-300 ease-in-out hover:bg-gray-700 hover:shadow-lg hover:-translate-y-0.5 focus:outline-hidden focus:ring-2 focus:ring-offset-2 focus:ring-gray-500" | ||||||
|  |       )} | ||||||
|       target={target} |       target={target} | ||||||
|     > |     > | ||||||
|       {icon && React.createElement(icon, { size: 20, className: "mr-2" })} |       {icon && React.createElement(icon, { size: 20, className: "mr-2" })} | ||||||
|       {label} |       {label} | ||||||
|     </Link> |     </Link> | ||||||
|   ); |   ) | ||||||
| }; | } | ||||||
| 
 | 
 | ||||||
| export default Button | export default Button | ||||||
|  | @ -3,10 +3,11 @@ import { faGitAlt, faGithub } from '@fortawesome/free-brands-svg-icons' | ||||||
| import { faStar, faCodeBranch } from '@fortawesome/free-solid-svg-icons' | import { faStar, faCodeBranch } from '@fortawesome/free-solid-svg-icons' | ||||||
| import featuredProjects from '@/public/data/featured.json' | import featuredProjects from '@/public/data/featured.json' | ||||||
| import Link from 'next/link' | import Link from 'next/link' | ||||||
|  | import { cn } from '@/lib/utils' | ||||||
| 
 | 
 | ||||||
| export default function GitHubFeatured() { | export default function GitHubFeatured({ className }: { className?: string }) { | ||||||
|   return ( |   return ( | ||||||
|     <div className="grid grid-cols-1 md:grid-cols-2 gap-4"> |     <div className={cn("grid grid-cols-1 md:grid-cols-2 gap-4", className)}> | ||||||
|       {featuredProjects.map((project) => ( |       {featuredProjects.map((project) => ( | ||||||
|         <div key={project.id} className="bg-gray-800 p-6 rounded-lg shadow-md min-h-[200px] flex flex-col"> |         <div key={project.id} className="bg-gray-800 p-6 rounded-lg shadow-md min-h-[200px] flex flex-col"> | ||||||
|           <div className="flex-1"> |           <div className="flex-1"> | ||||||
|  |  | ||||||
|  | @ -76,5 +76,43 @@ | ||||||
|       "phone": "(802) 416-9516", |       "phone": "(802) 416-9516", | ||||||
|       "email": "aidan@p0ntus.com" |       "email": "aidan@p0ntus.com" | ||||||
|     } |     } | ||||||
|  |   }, | ||||||
|  |   "about": { | ||||||
|  |     "title": "About Me", | ||||||
|  |     "description": "Hey there! My name is Aidan, and I'm a systems administrator, full-stack developer, and student from the United States. I primarily work with Linux, Docker, Next.js, and Node.js.", | ||||||
|  |     "sections": { | ||||||
|  |       "projects": "Projects", | ||||||
|  |       "hobbies": "Hobbies", | ||||||
|  |       "devices": "Devices", | ||||||
|  |       "contributions": "Contributions", | ||||||
|  |       "featuredProjects": "Featured Projects" | ||||||
|  |     }, | ||||||
|  |     "projects": [ | ||||||
|  |       "I have worked on countless projects over the past five years, for the most part. I have been learning to program in Python since I was seven and have 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 Node.js, my new favorite language as of a year ago. My biggest project is LibreCloud, a free service provider for individuals.", | ||||||
|  |       "In terms of system administration, I have developed my skills over the past three years of learning Linux for fun. I currently operate three servers running in the cloud, which run out of Germany and the United States." | ||||||
|  |     ], | ||||||
|  |     "hobbies": [ | ||||||
|  |       "When I'm not programming, I can typically be found installing another Linux distro on my laptop or flashing a new ROM to my phone. I am also a passionate writer and I like to write creatively in my free time.", | ||||||
|  |       "I consider maintaining my technology as a hobby as well, as I devote a lot of time to it. I currently run Gentoo Linux on my Thinkpad T470s, which does not use a single bin package. I am very proud of this laptop, despite it's constant need for compiling updates.", | ||||||
|  |       "I am almost always active on my Gitea instance and GitHub and make daily contributions to several of my repositories. 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!", | ||||||
|  |       "My Google Pixel 7 Pro (cheetah) runs LineageOS 22.1, and has been one of my favorite additions to my life. It is proudly rooted with KernelSU-Next. It has suffered one drop to it's back on a tile floor." | ||||||
|  |     ], | ||||||
|  |     "devices": { | ||||||
|  |       "Phone": [ | ||||||
|  |         "I use a Google Pixel 7 Pro (cheetah) as my daily driver. It runs LineageOS microG and is proudly rooted with KernelSU-Next.", | ||||||
|  |         "It's back is shattered and missing volume buttons, but it continues to thrive and survive as my daily driver." | ||||||
|  |       ], | ||||||
|  |       "Laptop": [ | ||||||
|  |         "I use a Lenovo Thinkpad T470s running Arch Linux. I've had it for about half a year now, and it's been a great experience. I proudly use X11 and LXDE, with some Xfce backend components to make management easier." | ||||||
|  |       ] | ||||||
|  |     }, | ||||||
|  |     "contributions": [ | ||||||
|  |       "Most of my repositories have migrated to LibreCloud Git. My username is aidan.", | ||||||
|  |       "You can find me on GitHub as ihatenodejs." | ||||||
|  |     ], | ||||||
|  |     "featuredProjects": [ | ||||||
|  |       "Here's just four of my top projects. Star and fork counts are manually updated and count both Gitea and GitHub." | ||||||
|  |     ] | ||||||
|   } |   } | ||||||
| }  | }  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue