diff --git a/.eslintrc.json b/.eslintrc.json index a026f2c1..e43ebf23 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -3,7 +3,7 @@ "extends": [ "eslint:recommended", "plugin:@typescript-eslint/recommended", - "next" + "next/core-web-vitals" ], "plugins": [ "@typescript-eslint", diff --git a/.github/workflows/cypress.yml b/.github/workflows/cypress.yml index 1651e912..388b3432 100644 --- a/.github/workflows/cypress.yml +++ b/.github/workflows/cypress.yml @@ -13,6 +13,7 @@ jobs: with: build: yarn build start: yarn start + quiet: true command-prefix: percy exec -- env: PERCY_TOKEN: "${{ secrets.PERCY_TOKEN }}" diff --git a/.gitignore b/.gitignore index 26d39d6c..524f958c 100644 --- a/.gitignore +++ b/.gitignore @@ -30,5 +30,8 @@ yarn-error.log* .env.production.local # Cypress +cypress/downloads cypress/screenshots cypress/videos + +.contentlayer diff --git a/app/%5Ftest/page.tsx b/app/%5Ftest/page.tsx new file mode 100644 index 00000000..563f9f93 --- /dev/null +++ b/app/%5Ftest/page.tsx @@ -0,0 +1,41 @@ +import { allTests } from 'contentlayer/generated' +import { notFound } from 'next/navigation' +import type { Metadata } from 'next/types' +import { useMDXComponent } from 'next-contentlayer/hooks' + +import { FancyH1 } from '@/components/headings' +import mdxComponents from '@/components/mdx-components' + +export const generateMetadata = (): Metadata => { + const testPage = allTests[0] + + return { + title: testPage.title, + description: testPage.description, + openGraph: { + type: 'article', + title: testPage.title, + description: testPage.description, + images: { + url: testPage.image, + }, + tags: testPage.tags, + }, + } +} + +export default function Test() { + const testPage = allTests[0] + if (!testPage) notFound() + + const MDXContent = useMDXComponent(testPage.body.code) + + return ( + <> + {testPage.title} +
+ +
+ + ) +} diff --git a/public/img/apple-touch-icon.png b/app/apple-icon.png similarity index 100% rename from public/img/apple-touch-icon.png rename to app/apple-icon.png diff --git a/app/blog/[slug]/layout.tsx b/app/blog/[slug]/layout.tsx new file mode 100644 index 00000000..8ec90f81 --- /dev/null +++ b/app/blog/[slug]/layout.tsx @@ -0,0 +1,14 @@ +import Link from 'next/link' + +export default function PostLayout({ children }) { + return ( +
+ {children} +
+ + ← Back to Blog + +
+
+ ) +} diff --git a/app/blog/[slug]/page.tsx b/app/blog/[slug]/page.tsx new file mode 100644 index 00000000..1eff6f79 --- /dev/null +++ b/app/blog/[slug]/page.tsx @@ -0,0 +1,54 @@ +import { allPosts } from 'contentlayer/generated' +import dayjs from 'dayjs' +import utc from 'dayjs/plugin/utc' +import type { Metadata } from 'next' +import { notFound } from 'next/navigation' +import { useMDXComponent } from 'next-contentlayer/hooks' + +import { H1 } from '@/components/headings' +import mdxComponents from '@/components/mdx-components' + +dayjs.extend(utc) + +export const generateStaticParams = async () => allPosts.map((post) => ({ slug: post.slug })) + +export const generateMetadata = ({ params }: { params: { slug: string } }): Metadata => { + const post = allPosts.find((post) => post.slug === decodeURIComponent(params.slug)) + if (!post) throw new Error(`Post not found for slug: ${decodeURIComponent(params.slug)}`) + + return { + title: post.title, + description: post.description, + openGraph: { + title: post.title, + type: 'article', + publishedTime: post.publishedAt, + authors: 'Ryan Rishi', + url: `https://ryanrishi.com/blog/${decodeURIComponent(params.slug)}`, + images: [ + { url: `https://ryanrishi.com/${post.image}` }, + ], + tags: post.tags, + }, + twitter: { + title: post.title, + images: `https://ryanrishi.com/${post.image}`, + }, + } +} + +export default function Post({ params }: { params: { slug: string } }) { + const post = allPosts.find((post) => post.slug === params.slug) + if (!post) notFound() + + const MDXContent = useMDXComponent(post.body.code) + + return ( + <> +

{post.title}

+

{dayjs.utc(post.publishedAt).format('MMMM D, YYYY')}

+ + + + ) +} diff --git a/app/blog/page.tsx b/app/blog/page.tsx new file mode 100644 index 00000000..2bbb9269 --- /dev/null +++ b/app/blog/page.tsx @@ -0,0 +1,51 @@ +import { allPosts } from 'contentlayer/generated' +import { compareDesc } from 'date-fns' +import dayjs from 'dayjs' +import utc from 'dayjs/plugin/utc' +import { Metadata } from 'next' +import Link from 'next/link' + +import { FancyH1,H1 } from '@/components/headings' + +dayjs.extend(utc) + +export const metadata: Metadata = { + title: 'Blog', + openGraph: { + title: 'Blog', + }, + twitter: { + title: 'Blog', + }, +} + +export default function BlogIndex() { + const posts = allPosts.sort((a, b) => compareDesc(new Date(a.publishedAt), new Date(b.publishedAt))) + + return ( + <> + Blog + {posts.map((post, i) => ( +
+

+ + {post.title} + +

+

{dayjs.utc(post.publishedAt).format('MMMM D, YYYY')}

+ +
+

{post.description}

+ + Read more » + +
+
+ ))} + + ) +} diff --git a/app/contact/page.tsx b/app/contact/page.tsx new file mode 100644 index 00000000..319ac345 --- /dev/null +++ b/app/contact/page.tsx @@ -0,0 +1,93 @@ +import classNames from 'classnames' +import { Metadata } from 'next/types' + +import { FancyH1 } from '@/components/headings' + +export const metadata: Metadata = { + title: 'Contact', + openGraph: { + title: 'Contact', + }, + twitter: { + title: 'Contact', + }, +} + +const sharedInputClassNames = 'block w-full border rounded py-3 px-4 mb-3 mt-2 leading-tight appearance-none text-slate-800 bg-slate-100 dark:text-slate-100 dark:bg-slate-700 border-slate-300 hover:border-slate-400 transition' + +function Label({ htmlFor, children }) { + return ( + + ) +} + +export default function Contact() { + return ( +
+ Contact +
+
+
+ +
+
+
+
+ +
+
+
+
+