Initial commit
This commit is contained in:
commit
84591f3a2d
23 changed files with 4795 additions and 0 deletions
21
.gitignore
vendored
Normal file
21
.gitignore
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
# build output
|
||||||
|
dist/
|
||||||
|
# generated types
|
||||||
|
.astro/
|
||||||
|
|
||||||
|
# dependencies
|
||||||
|
node_modules/
|
||||||
|
|
||||||
|
# logs
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
pnpm-debug.log*
|
||||||
|
|
||||||
|
|
||||||
|
# environment variables
|
||||||
|
.env
|
||||||
|
.env.production
|
||||||
|
|
||||||
|
# macOS-specific files
|
||||||
|
.DS_Store
|
1
README.md
Normal file
1
README.md
Normal file
|
@ -0,0 +1 @@
|
||||||
|
# Personal site using Astro SSR + React + Railway
|
26
astro.config.mjs
Normal file
26
astro.config.mjs
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
import { defineConfig } from "astro/config";
|
||||||
|
import mdx from "@astrojs/mdx";
|
||||||
|
import sitemap from "@astrojs/sitemap";
|
||||||
|
import tailwind from "@astrojs/tailwind";
|
||||||
|
import node from "@astrojs/node";
|
||||||
|
|
||||||
|
import react from "@astrojs/react";
|
||||||
|
|
||||||
|
// https://astro.build/config
|
||||||
|
export default defineConfig({
|
||||||
|
site: "https://atri.dad/",
|
||||||
|
output: "server",
|
||||||
|
adapter: node({
|
||||||
|
mode: "standalone",
|
||||||
|
}),
|
||||||
|
server: {
|
||||||
|
host: "0.0.0.0",
|
||||||
|
port: process.env.PORT || 3000,
|
||||||
|
},
|
||||||
|
integrations: [mdx(), sitemap(), tailwind(), react()],
|
||||||
|
vite: {
|
||||||
|
ssr: {
|
||||||
|
noExternal: ["react-icons"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
29
package.json
Normal file
29
package.json
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
{
|
||||||
|
"name": "atri-dad",
|
||||||
|
"type": "module",
|
||||||
|
"version": "0.0.1",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "astro dev",
|
||||||
|
"start": "node dist/server/entry.mjs",
|
||||||
|
"build": "astro build",
|
||||||
|
"preview": "astro preview",
|
||||||
|
"astro": "astro"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@astrojs/mdx": "^0.19.1",
|
||||||
|
"@astrojs/node": "^5.1.3",
|
||||||
|
"@astrojs/react": "^2.1.3",
|
||||||
|
"@astrojs/rss": "^2.4.1",
|
||||||
|
"@astrojs/sitemap": "^1.3.1",
|
||||||
|
"@astrojs/tailwind": "^3.1.2",
|
||||||
|
"@tailwindcss/typography": "^0.5.9",
|
||||||
|
"@types/react": "^18.0.21",
|
||||||
|
"@types/react-dom": "^18.0.6",
|
||||||
|
"astro": "^2.4.5",
|
||||||
|
"daisyui": "^2.51.6",
|
||||||
|
"react": "^18.0.0",
|
||||||
|
"react-dom": "^18.0.0",
|
||||||
|
"react-icons": "^4.8.0",
|
||||||
|
"tailwindcss": "^3.3.2"
|
||||||
|
}
|
||||||
|
}
|
4262
pnpm-lock.yaml
generated
Normal file
4262
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load diff
BIN
public/favicon.ico
Normal file
BIN
public/favicon.ico
Normal file
Binary file not shown.
After Width: | Height: | Size: 15 KiB |
11
sandbox.config.json
Normal file
11
sandbox.config.json
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
{
|
||||||
|
"infiniteLoopProtection": true,
|
||||||
|
"hardReloadOnChange": false,
|
||||||
|
"view": "browser",
|
||||||
|
"template": "node",
|
||||||
|
"container": {
|
||||||
|
"port": 3000,
|
||||||
|
"startScript": "start",
|
||||||
|
"node": "16"
|
||||||
|
}
|
||||||
|
}
|
43
src/components/BaseHead.astro
Normal file
43
src/components/BaseHead.astro
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
---
|
||||||
|
// Import the global.css file here so that it is included on
|
||||||
|
// all pages through the use of the <BaseHead /> component.
|
||||||
|
import "../styles/global.css";
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
title: string;
|
||||||
|
description: string;
|
||||||
|
image?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const canonicalURL = new URL(Astro.url.pathname, Astro.site);
|
||||||
|
|
||||||
|
const { title, description, image = "/placeholder-social.jpg" } = Astro.props;
|
||||||
|
---
|
||||||
|
|
||||||
|
<!-- Global Metadata -->
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<meta name="viewport" content="width=device-width,initial-scale=1" />
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
|
<meta name="generator" content={Astro.generator} />
|
||||||
|
|
||||||
|
<!-- Canonical URL -->
|
||||||
|
<link rel="canonical" href={canonicalURL} />
|
||||||
|
|
||||||
|
<!-- Primary Meta Tags -->
|
||||||
|
<title>{title}</title>
|
||||||
|
<meta name="title" content={title} />
|
||||||
|
<meta name="description" content={description} />
|
||||||
|
|
||||||
|
<!-- Open Graph / Facebook -->
|
||||||
|
<meta property="og:type" content="website" />
|
||||||
|
<meta property="og:url" content={Astro.url} />
|
||||||
|
<meta property="og:title" content={title} />
|
||||||
|
<meta property="og:description" content={description} />
|
||||||
|
<meta property="og:image" content={new URL(image, Astro.url)} />
|
||||||
|
|
||||||
|
<!-- Twitter -->
|
||||||
|
<meta property="twitter:card" content="summary_large_image" />
|
||||||
|
<meta property="twitter:url" content={Astro.url} />
|
||||||
|
<meta property="twitter:title" content={title} />
|
||||||
|
<meta property="twitter:description" content={description} />
|
||||||
|
<meta property="twitter:image" content={new URL(image, Astro.url)} />
|
17
src/components/FormattedDate.astro
Normal file
17
src/components/FormattedDate.astro
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
---
|
||||||
|
export interface Props {
|
||||||
|
date: Date;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { date } = Astro.props;
|
||||||
|
---
|
||||||
|
|
||||||
|
<time datetime={date.toISOString()}>
|
||||||
|
{
|
||||||
|
date.toLocaleDateString('en-us', {
|
||||||
|
year: 'numeric',
|
||||||
|
month: 'short',
|
||||||
|
day: 'numeric',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</time>
|
35
src/components/Header.astro
Normal file
35
src/components/Header.astro
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
---
|
||||||
|
import HeaderLink from "./HeaderLink.astro";
|
||||||
|
import { SITE_TITLE } from "../consts";
|
||||||
|
---
|
||||||
|
|
||||||
|
<header class="navbar bg-base-100">
|
||||||
|
<div class="navbar-start">
|
||||||
|
<a class="btn btn-ghost normal-case text-xl" href="/">{SITE_TITLE}</a>
|
||||||
|
</div>
|
||||||
|
<div class="navbar-end">
|
||||||
|
<div class="dropdown dropdown-end">
|
||||||
|
<label tabindex="0" class="btn btn-ghost">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
class="h-5 w-5"
|
||||||
|
fill="none"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
stroke="currentColor"
|
||||||
|
><path
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
stroke-width="2"
|
||||||
|
d="M4 6h16M4 12h8m-8 6h16"></path></svg
|
||||||
|
>
|
||||||
|
</label>
|
||||||
|
<ul
|
||||||
|
tabindex="0"
|
||||||
|
class="menu menu-compact dropdown-content mt-3 p-2 shadow bg-base-100 rounded-box w-52"
|
||||||
|
>
|
||||||
|
<li><HeaderLink href="/">Home</HeaderLink></li>
|
||||||
|
<li><HeaderLink href="/blog">Blog</HeaderLink></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
24
src/components/HeaderLink.astro
Normal file
24
src/components/HeaderLink.astro
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
---
|
||||||
|
import type { HTMLAttributes } from 'astro/types';
|
||||||
|
|
||||||
|
type Props = HTMLAttributes<'a'>;
|
||||||
|
|
||||||
|
const { href, class: className, ...props } = Astro.props;
|
||||||
|
|
||||||
|
const { pathname } = Astro.url;
|
||||||
|
const isActive = href === pathname || href === pathname.replace(/\/$/, '');
|
||||||
|
---
|
||||||
|
|
||||||
|
<a href={href} class:list={[className, { active: isActive }]} {...props}>
|
||||||
|
<slot />
|
||||||
|
</a>
|
||||||
|
<style>
|
||||||
|
a {
|
||||||
|
display: inline-block;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
a.active {
|
||||||
|
font-weight: bolder;
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
</style>
|
5
src/consts.ts
Normal file
5
src/consts.ts
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
// Place any global data in this file.
|
||||||
|
// You can import this data from anywhere in your site by using the `import` keyword.
|
||||||
|
|
||||||
|
export const SITE_TITLE = "[atri.dad]";
|
||||||
|
export const SITE_DESCRIPTION = "It's me, hi, I'm the problem, it's me!";
|
7
src/content/blog/first-post.md
Normal file
7
src/content/blog/first-post.md
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
---
|
||||||
|
title: "Welcome to my site!"
|
||||||
|
description: "Welcome to my site!"
|
||||||
|
pubDate: "March 26 2023"
|
||||||
|
---
|
||||||
|
|
||||||
|
Hello and welcome to my small corner of the internet! I will post some content here about the tech I work with, and other things that excite me. **Thanks for stopping by**!
|
20
src/content/config.ts
Normal file
20
src/content/config.ts
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
import { defineCollection, z } from "astro:content";
|
||||||
|
|
||||||
|
const blog = defineCollection({
|
||||||
|
// Type-check frontmatter using a schema
|
||||||
|
schema: z.object({
|
||||||
|
title: z.string(),
|
||||||
|
description: z.string(),
|
||||||
|
// Transform string to Date object
|
||||||
|
pubDate: z
|
||||||
|
.string()
|
||||||
|
.or(z.date())
|
||||||
|
.transform((val) => new Date(val)),
|
||||||
|
updatedDate: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.transform((str) => (str ? new Date(str) : undefined)),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const collections = { blog };
|
2
src/env.d.ts
vendored
Normal file
2
src/env.d.ts
vendored
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
/// <reference path="../.astro/types.d.ts" />
|
||||||
|
/// <reference types="astro/client" />
|
35
src/layouts/BlogPost.astro
Normal file
35
src/layouts/BlogPost.astro
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
---
|
||||||
|
import type { CollectionEntry } from "astro:content";
|
||||||
|
import BaseHead from "../components/BaseHead.astro";
|
||||||
|
import Header from "../components/Header.astro";
|
||||||
|
import FormattedDate from "../components/FormattedDate.astro";
|
||||||
|
|
||||||
|
type Props = CollectionEntry<"blog">["data"];
|
||||||
|
|
||||||
|
const { title, description, pubDate, updatedDate } = Astro.props;
|
||||||
|
---
|
||||||
|
|
||||||
|
<html lang="en" data-theme="night">
|
||||||
|
<head>
|
||||||
|
<BaseHead title={title} description={description} />
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<Header />
|
||||||
|
<main class="prose prose-invert mx-auto p-4">
|
||||||
|
<article>
|
||||||
|
<h1 class="title">{title}</h1>
|
||||||
|
<FormattedDate date={pubDate} />
|
||||||
|
{
|
||||||
|
updatedDate && (
|
||||||
|
<div class="last-updated-on">
|
||||||
|
Last updated on <FormattedDate date={updatedDate} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
<hr />
|
||||||
|
<slot />
|
||||||
|
</article>
|
||||||
|
</main>
|
||||||
|
</body>
|
||||||
|
</html>
|
18
src/pages/blog/[...slug].astro
Normal file
18
src/pages/blog/[...slug].astro
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
---
|
||||||
|
import { CollectionEntry, getCollection } from "astro:content";
|
||||||
|
import BlogPost from "../../layouts/BlogPost.astro";
|
||||||
|
|
||||||
|
const posts = await getCollection("blog");
|
||||||
|
type Props = CollectionEntry<"blog">;
|
||||||
|
|
||||||
|
const { slug } = Astro.params;
|
||||||
|
const post = posts.find((page) => page.slug === slug);
|
||||||
|
if (!post) {
|
||||||
|
return Astro.redirect("/404");
|
||||||
|
}
|
||||||
|
const { Content } = await post.render();
|
||||||
|
---
|
||||||
|
|
||||||
|
<BlogPost {...post.data}>
|
||||||
|
<Content />
|
||||||
|
</BlogPost>
|
37
src/pages/blog/index.astro
Normal file
37
src/pages/blog/index.astro
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
---
|
||||||
|
import BaseHead from "../../components/BaseHead.astro";
|
||||||
|
import Header from "../../components/Header.astro";
|
||||||
|
import { SITE_TITLE, SITE_DESCRIPTION } from "../../consts";
|
||||||
|
import { getCollection } from "astro:content";
|
||||||
|
import FormattedDate from "../../components/FormattedDate.astro";
|
||||||
|
|
||||||
|
const posts = (await getCollection("blog")).sort(
|
||||||
|
(a, b) => a.data.pubDate.valueOf() - b.data.pubDate.valueOf()
|
||||||
|
);
|
||||||
|
---
|
||||||
|
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en" data-theme="night">
|
||||||
|
<head>
|
||||||
|
<BaseHead title={SITE_TITLE} description={SITE_DESCRIPTION} />
|
||||||
|
</head>
|
||||||
|
<body class="block h-[100%]">
|
||||||
|
<Header />
|
||||||
|
<main
|
||||||
|
class="prose prose-invert container flex flex-col items-center justify-center gap-3 sm:gap-6 px-4 py-16 text-center mx-auto min-h-[calc(100%-64px)]"
|
||||||
|
>
|
||||||
|
<section>
|
||||||
|
<ul>
|
||||||
|
{
|
||||||
|
posts.map((post) => (
|
||||||
|
<li>
|
||||||
|
<FormattedDate date={post.data.pubDate} />
|
||||||
|
<a href={`/blog/${post.slug}/`}>{post.data.title}</a>
|
||||||
|
</li>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</ul>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
</body>
|
||||||
|
</html>
|
158
src/pages/index.astro
Normal file
158
src/pages/index.astro
Normal file
|
@ -0,0 +1,158 @@
|
||||||
|
---
|
||||||
|
import BaseHead from "../components/BaseHead.astro";
|
||||||
|
import Header from "../components/Header.astro";
|
||||||
|
import { SITE_TITLE, SITE_DESCRIPTION } from "../consts";
|
||||||
|
|
||||||
|
import {
|
||||||
|
SiGithub,
|
||||||
|
SiLinkedin,
|
||||||
|
SiNextdotjs,
|
||||||
|
SiPrisma,
|
||||||
|
SiSpotify,
|
||||||
|
SiTwitch,
|
||||||
|
SiTwitter,
|
||||||
|
SiYoutube,
|
||||||
|
SiAstro,
|
||||||
|
SiSvelte,
|
||||||
|
SiPostgresql,
|
||||||
|
SiRedis,
|
||||||
|
SiRailway,
|
||||||
|
} from "react-icons/si";
|
||||||
|
---
|
||||||
|
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en" data-theme="night">
|
||||||
|
<head>
|
||||||
|
<BaseHead title={SITE_TITLE} description={SITE_DESCRIPTION} />
|
||||||
|
</head>
|
||||||
|
<body class="block h-[100%]">
|
||||||
|
<Header title={SITE_TITLE} />
|
||||||
|
<main
|
||||||
|
class="container flex flex-col items-center justify-center gap-3 sm:gap-6 px-4 py-16 text-center mx-auto min-h-[calc(100%-64px)]"
|
||||||
|
>
|
||||||
|
<h1 class="text-4xl font-extrabold text-white sm:text-8xl">
|
||||||
|
Hi, I'm <span class="text-pink-500">Atridad</span>
|
||||||
|
</h1>
|
||||||
|
|
||||||
|
<h2
|
||||||
|
class="text-2xl font-extrabold tracking-tight text-white sm:text-[2rem]"
|
||||||
|
>
|
||||||
|
I'm a <span class="text-green-500">persian</span> -
|
||||||
|
<span class="text-red-500">canadian</span>{" "}
|
||||||
|
<span
|
||||||
|
class="bg-gradient-to-r from-pink-500 to-blue-500 bg-clip-text text-transparent"
|
||||||
|
>
|
||||||
|
queer
|
||||||
|
</span>{" "}
|
||||||
|
software developer who loves making things for the web.
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<h2
|
||||||
|
class="text-xl font-extrabold tracking-tight text-white sm:text-[1.5rem]"
|
||||||
|
>
|
||||||
|
Pronouns: (he/<span class="text-pink-500">they</span>)
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<span>
|
||||||
|
<h2 class="mb-2 text-xl text-white sm:text-[1.5rem]">
|
||||||
|
Places I exist:
|
||||||
|
</h2>
|
||||||
|
<div
|
||||||
|
class="flex flex-row flex-wrap items-center justify-center gap-4 text-center"
|
||||||
|
>
|
||||||
|
<a
|
||||||
|
href={"https://www.linkedin.com/in/atridad/"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="Linkedin"
|
||||||
|
>
|
||||||
|
<SiLinkedin className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href={"https://twitter.com/atridadl/"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="Twitter"
|
||||||
|
>
|
||||||
|
<SiTwitter className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href={"https://github.com/atridadl"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="Github"
|
||||||
|
>
|
||||||
|
<SiGithub className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href={"https://open.spotify.com/user/31v3p4oq6im7fvpqhhbju222pbr4?si=d75830b95ed94d4f"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="Spotify"
|
||||||
|
>
|
||||||
|
<SiSpotify className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href={"https://www.twitch.tv/himbothyswaggins"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="Twitch.tv"
|
||||||
|
>
|
||||||
|
<SiTwitch className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href={"https://www.youtube.com/@himbothyswaggins"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="YouTube"
|
||||||
|
>
|
||||||
|
<SiYoutube className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span>
|
||||||
|
<h2 class="mb-2 text-xl text-white sm:text-[1.5rem]">Stuff I use:</h2>
|
||||||
|
<div
|
||||||
|
class="flex flex-row flex-wrap items-center justify-center gap-4 text-center"
|
||||||
|
>
|
||||||
|
<a href={"https://nextjs.org/"} target="_blank" aria-label="Next.js">
|
||||||
|
<SiNextdotjs className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a href={"https://astro.build/"} target="_blank" aria-label="Astro">
|
||||||
|
<SiAstro className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a
|
||||||
|
href={"https://kit.svelte.dev/"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="SvelteKit"
|
||||||
|
>
|
||||||
|
<SiSvelte className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a
|
||||||
|
href={"https://www.prisma.io/"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="Prisma"
|
||||||
|
>
|
||||||
|
<SiPrisma className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a
|
||||||
|
href={"https://www.postgresql.org/"}
|
||||||
|
target="_blank"
|
||||||
|
aria-label="PostgreSQL"
|
||||||
|
>
|
||||||
|
<SiPostgresql
|
||||||
|
className="text-2xl hover:text-pink-500 sm:text-4xl"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a href={"https://redis.io/"} target="_blank" aria-label="Redis">
|
||||||
|
<SiRedis className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a href={"https://railway.app/"} target="_blank" aria-label="Railway">
|
||||||
|
<SiRailway className="text-2xl hover:text-pink-500 sm:text-4xl" />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
</main>
|
||||||
|
</body>
|
||||||
|
</html>
|
16
src/pages/rss.xml.js
Normal file
16
src/pages/rss.xml.js
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
import rss from '@astrojs/rss';
|
||||||
|
import { getCollection } from 'astro:content';
|
||||||
|
import { SITE_TITLE, SITE_DESCRIPTION } from '../consts';
|
||||||
|
|
||||||
|
export async function get(context) {
|
||||||
|
const posts = await getCollection('blog');
|
||||||
|
return rss({
|
||||||
|
title: SITE_TITLE,
|
||||||
|
description: SITE_DESCRIPTION,
|
||||||
|
site: context.site,
|
||||||
|
items: posts.map((post) => ({
|
||||||
|
...post.data,
|
||||||
|
link: `/blog/${post.slug}/`,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
}
|
12
src/styles/global.css
Normal file
12
src/styles/global.css
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
@tailwind base;
|
||||||
|
@tailwind components;
|
||||||
|
@tailwind utilities;
|
||||||
|
|
||||||
|
html,
|
||||||
|
container,
|
||||||
|
body {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
overflow-y: auto;
|
||||||
|
position: fixed;
|
||||||
|
}
|
8
tailwind.config.cjs
Normal file
8
tailwind.config.cjs
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
/** @type {import('tailwindcss').Config} */
|
||||||
|
module.exports = {
|
||||||
|
content: ["./src/**/*.{astro,html,js,jsx,md,mdx,svelte,ts,tsx,vue}"],
|
||||||
|
theme: {
|
||||||
|
extend: {},
|
||||||
|
},
|
||||||
|
plugins: [require("@tailwindcss/typography"), require("daisyui")],
|
||||||
|
};
|
8
tsconfig.json
Normal file
8
tsconfig.json
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
{
|
||||||
|
"extends": "astro/tsconfigs/strict",
|
||||||
|
"compilerOptions": {
|
||||||
|
"strictNullChecks": true,
|
||||||
|
"jsx": "react-jsx",
|
||||||
|
"jsxImportSource": "react"
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue