feat/memorium-go #3
| @@ -17,13 +17,12 @@ const locales = { | |||||||
| const DEFAULT_LAYOUT = '@layouts/Post.astro'; | const DEFAULT_LAYOUT = '@layouts/Post.astro'; | ||||||
|  |  | ||||||
| function setDefaultLayout() { | function setDefaultLayout() { | ||||||
|   return function (_, file) { |   return function(_, file) { | ||||||
|     const { frontmatter } = file.data.astro; |     const { frontmatter } = file.data.astro; | ||||||
|     if (!frontmatter.layout) frontmatter.layout = DEFAULT_LAYOUT; |     if (!frontmatter.layout) frontmatter.layout = DEFAULT_LAYOUT; | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
| // https://astro.build/config |  | ||||||
| export default defineConfig({ | export default defineConfig({ | ||||||
|   site: "https://max-richter.dev", |   site: "https://max-richter.dev", | ||||||
|   trailingSlash: "never", |   trailingSlash: "never", | ||||||
| @@ -39,7 +38,7 @@ export default defineConfig({ | |||||||
|       }), |       }), | ||||||
|     ], |     ], | ||||||
|     server: { |     server: { | ||||||
|      watch: { |       watch: { | ||||||
|         // Customize watch behavior to reduce file watchers |         // Customize watch behavior to reduce file watchers | ||||||
|         ignored: ['**/node_modules/**', '**/dist/**', '**/.git/**'], |         ignored: ['**/node_modules/**', '**/dist/**', '**/.git/**'], | ||||||
|         usePolling: process.env.NODE_ENV === 'production', |         usePolling: process.env.NODE_ENV === 'production', | ||||||
|   | |||||||
							
								
								
									
										28
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								package.json
									
									
									
									
									
								
							| @@ -11,32 +11,32 @@ | |||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@astrojs/check": "^0.9.4", |     "@astrojs/check": "^0.9.4", | ||||||
|     "@astrojs/mdx": "^4.2.6", |     "@astrojs/mdx": "^4.3.1", | ||||||
|     "@astrojs/svelte": "^7.0.13", |     "@astrojs/svelte": "^7.1.0", | ||||||
|     "@astrojs/tailwind": "^6.0.2", |     "@astrojs/tailwind": "^6.0.2", | ||||||
|     "astro": "^5.7.13", |     "astro": "^5.12.0", | ||||||
|     "astro-i18n-aut": "^0.7.3", |     "astro-i18n-aut": "^0.7.3", | ||||||
|     "exifreader": "^4.30.1", |     "exifreader": "^4.31.1", | ||||||
|     "svelte": "^5.28.6", |     "svelte": "^5.36.10", | ||||||
|     "svelte-gestures": "^5.1.4", |     "svelte-gestures": "^5.1.4", | ||||||
|     "tailwindcss": "^4.1.6", |     "tailwindcss": "^4.1.11", | ||||||
|     "thumbhash": "^0.1.1", |     "thumbhash": "^0.1.1", | ||||||
|     "typescript": "^5.8.3" |     "typescript": "^5.8.3" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@astrojs/sitemap": "^3.4.0", |     "@astrojs/sitemap": "^3.4.1", | ||||||
|     "@iconify-json/tabler": "^1.2.17", |     "@iconify-json/tabler": "^1.2.19", | ||||||
|     "@types/markdown-it": "^14.1.2", |     "@types/markdown-it": "^14.1.2", | ||||||
|     "@unocss/preset-icons": "^66.1.1", |     "@unocss/preset-icons": "^66.3.3", | ||||||
|     "@unocss/reset": "^66.1.1", |     "@unocss/reset": "^66.3.3", | ||||||
|     "astro-font": "^1.1.0", |     "astro-font": "^1.1.0", | ||||||
|     "markdown-it": "^14.1.0", |     "markdown-it": "^14.1.0", | ||||||
|     "ogl": "^1.0.11", |     "ogl": "^1.0.11", | ||||||
|     "prettier": "^3.5.3", |     "prettier": "^3.6.2", | ||||||
|     "prettier-plugin-astro": "^0.14.1", |     "prettier-plugin-astro": "^0.14.1", | ||||||
|     "sharp": "^0.34.1", |     "sharp": "^0.34.3", | ||||||
|     "unocss": "^66.1.1", |     "unocss": "^66.3.3", | ||||||
|     "unplugin-icons": "^22.1.0", |     "unplugin-icons": "^22.1.0", | ||||||
|     "vite-plugin-glsl": "^1.4.1" |     "vite-plugin-glsl": "^1.5.1" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1809
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1809
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										4
									
								
								pnpm-workspace.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								pnpm-workspace.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | onlyBuiltDependencies: | ||||||
|  |   - esbuild | ||||||
|  |   - exifreader | ||||||
|  |   - sharp | ||||||
| @@ -24,8 +24,8 @@ const paths = [ | |||||||
|     text: t("nav.photos"), |     text: t("nav.photos"), | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     link: translatePath("/videos"), |     link: translatePath("/resources"), | ||||||
|     text: t("nav.videos"), |     text: t("nav.resources"), | ||||||
|   }, |   }, | ||||||
| ]; | ]; | ||||||
| --- | --- | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| --- | --- | ||||||
| title: Videos | title: Resources | ||||||
| menu: nav | menu: nav | ||||||
| --- | --- | ||||||
|   | |||||||
| @@ -53,10 +53,15 @@ const allowedExif = [ | |||||||
| ]; | ]; | ||||||
|  |  | ||||||
| export async function getExifData(image: ImageMetadata) { | export async function getExifData(image: ImageMetadata) { | ||||||
|  |   if (image.format === "svg") return undefined; // SVGs don't have EXIF data") | ||||||
|   const sharp = await getSharp(); |   const sharp = await getSharp(); | ||||||
|   if (!sharp) return; |   if (!sharp) return; | ||||||
|  |   const imagePath = (image as ImageMetadata & { fsPath: string }).fsPath; | ||||||
|   try { |   try { | ||||||
|     const tags = await ExifReader.load((image as ImageMetadata & { fsPath: string }).fsPath, { async: true }); |  | ||||||
|  |     const buffer = await sharp(imagePath).toBuffer(); | ||||||
|  |  | ||||||
|  |     const tags = await ExifReader.load(buffer, { async: true }); | ||||||
|  |  | ||||||
|     const out: Record<string, any> = {}; |     const out: Record<string, any> = {}; | ||||||
|     let hasExif = false; |     let hasExif = false; | ||||||
| @@ -70,7 +75,7 @@ export async function getExifData(image: ImageMetadata) { | |||||||
|     return hasExif ? out : undefined; |     return hasExif ? out : undefined; | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|  |  | ||||||
|     console.log("Error reading EXIF data", error); |     console.log(`Error reading EXIF data from ${imagePath}`, error); | ||||||
|     return undefined |     return undefined | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								src/helpers/memorium.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/helpers/memorium.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | export async function listResource(id: string): Promise<any[]> { | ||||||
|  |   try { | ||||||
|  |     const response = await fetch( | ||||||
|  |       `http://localhost:8080/resources?name=${id}`, | ||||||
|  |     ); | ||||||
|  |     return await response.json(); | ||||||
|  |   } catch (error) { | ||||||
|  |     return [] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -20,7 +20,7 @@ export const ui = { | |||||||
|     'home.subtitle': 'Trained Media Designer, Blender Nerd, Developer and Hardware Tinkerer.', |     'home.subtitle': 'Trained Media Designer, Blender Nerd, Developer and Hardware Tinkerer.', | ||||||
|     'nav.blog': 'Blog', |     'nav.blog': 'Blog', | ||||||
|     'nav.projects': 'Projects', |     'nav.projects': 'Projects', | ||||||
|     'nav.videos': 'Videos', |     'nav.resources': 'Resources', | ||||||
|     'nav.photos': 'Photos', |     'nav.photos': 'Photos', | ||||||
|     'toc.title': 'Table of Contents', |     'toc.title': 'Table of Contents', | ||||||
|     "resume": "Resume", |     "resume": "Resume", | ||||||
| @@ -38,7 +38,7 @@ export const ui = { | |||||||
|     'home.subtitle': 'Ausgebildeter Mediengestalter, Blender Nerd, Entwickler und Hardware Bastler.', |     'home.subtitle': 'Ausgebildeter Mediengestalter, Blender Nerd, Entwickler und Hardware Bastler.', | ||||||
|     'nav.blog': 'Blog', |     'nav.blog': 'Blog', | ||||||
|     'nav.projects': 'Projekte', |     'nav.projects': 'Projekte', | ||||||
|     'nav.videos': 'Videos', |     'nav.resources': 'Resources', | ||||||
|     'nav.photos': 'Fotos', |     'nav.photos': 'Fotos', | ||||||
|     "resume": "Lebenslauf", |     "resume": "Lebenslauf", | ||||||
|     'toc.title': 'Inhaltsverzeichnis', |     'toc.title': 'Inhaltsverzeichnis', | ||||||
|   | |||||||
| @@ -16,11 +16,10 @@ const { frontmatter, headings } = Astro.props; | |||||||
| const t = useTranslations(Astro.url); | const t = useTranslations(Astro.url); | ||||||
| const { title, url, date: dateString, links, toc, cover } = frontmatter; | const { title, url, date: dateString, links, toc, cover } = frontmatter; | ||||||
| const collection = url?.split("/")[2]; | const collection = url?.split("/")[2]; | ||||||
| const date = new Date(dateString); |  | ||||||
| const path = useTranslatedPath(Astro.url); |  | ||||||
|  |  | ||||||
| //@ts-ignore | //@ts-ignore | ||||||
| const backlinkContent = t(`nav.${collection}`).toLowerCase(); | const backlinkContent = t(`nav.${collection}`).toLowerCase(); | ||||||
|  | const date = new Date(dateString); | ||||||
|  | const path = useTranslatedPath(Astro.url); | ||||||
| --- | --- | ||||||
|  |  | ||||||
| <Layout title={title} image={cover}> | <Layout title={title} image={cover}> | ||||||
|   | |||||||
							
								
								
									
										64
									
								
								src/pages/resources/index.astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								src/pages/resources/index.astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import HeroCard from "@components/HeroCard.astro"; | ||||||
|  |  | ||||||
|  | const collection = "resources"; | ||||||
|  |  | ||||||
|  | const wiki = { | ||||||
|  |   id: "wiki", | ||||||
|  |   collection, | ||||||
|  |   body: "My knowledge base", | ||||||
|  |   data: { | ||||||
|  |     title: "Wiki", | ||||||
|  |     icon: "🧠", | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const articles = { | ||||||
|  |   id: "articles", | ||||||
|  |   collection, | ||||||
|  |   body: "Articles saved", | ||||||
|  |   data: { | ||||||
|  |     title: "Articles", | ||||||
|  |     icon: "📰", | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const recipes = { | ||||||
|  |   id: "recipes", | ||||||
|  |   collection, | ||||||
|  |   body: "Recipes", | ||||||
|  |   data: { | ||||||
|  |     title: "Recipes", | ||||||
|  |     icon: "🍲", | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const movies = { | ||||||
|  |   id: "movies", | ||||||
|  |   collection, | ||||||
|  |   body: "Movies", | ||||||
|  |   data: { | ||||||
|  |     title: "Movies", | ||||||
|  |     icon: "🎥", | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const series = { | ||||||
|  |   id: "series", | ||||||
|  |   collection, | ||||||
|  |   body: "Series", | ||||||
|  |   data: { | ||||||
|  |     title: "Series", | ||||||
|  |     icon: "📺", | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   <HeroCard post={wiki} /> | ||||||
|  |   <HeroCard post={recipes} /> | ||||||
|  |   <HeroCard post={articles} /> | ||||||
|  |   <HeroCard post={movies} /> | ||||||
|  |   <HeroCard post={series} /> | ||||||
|  | </Layout> | ||||||
							
								
								
									
										38
									
								
								src/pages/resources/movies/[movieName].astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								src/pages/resources/movies/[movieName].astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import * as memorium from "@helpers/memorium"; | ||||||
|  |  | ||||||
|  | export async function getStaticPaths() { | ||||||
|  |   const movieReviews = await memorium.listResource("Media/movies/*"); | ||||||
|  |  | ||||||
|  |   const paths = movieReviews.map((review: any) => { | ||||||
|  |     return { | ||||||
|  |       params: { | ||||||
|  |         movieName: review.identifier | ||||||
|  |           .replace("Media/movies/", "") | ||||||
|  |           .replace(/\.md$/, ""), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |   return paths; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const reviews = await memorium.listResource( | ||||||
|  |   //@ts-ignore | ||||||
|  |   `Media/movies/${Astro.params.movieName}.md`, | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | if (reviews.length === 0) { | ||||||
|  |   return new Response(null, { | ||||||
|  |     status: 404, | ||||||
|  |     statusText: "Not found", | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | const review = reviews[0]; | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   <h1>{review.itemReviewed?.name}</h1> | ||||||
|  |   <p>{review.reviewBody}</p> | ||||||
|  |   <!-- <pre><code>{JSON.stringify(review, null, 2)}</code></pre> --> | ||||||
|  | </Layout> | ||||||
							
								
								
									
										27
									
								
								src/pages/resources/movies/index.astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								src/pages/resources/movies/index.astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import HeroCard from "@components/HeroCard.astro"; | ||||||
|  | import * as memorium from "@helpers/memorium"; | ||||||
|  |  | ||||||
|  | const movieReviews = await memorium.listResource("Media/movies/*"); | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   { | ||||||
|  |     movieReviews.map((review: any) => ( | ||||||
|  |       <HeroCard | ||||||
|  |         post={{ | ||||||
|  |           collection: "resources/movies", | ||||||
|  |           id: review.identifier | ||||||
|  |             .replace("Media/movies/", "") | ||||||
|  |             .replace(/\.md$/, ""), | ||||||
|  |           data: { | ||||||
|  |             title: review.itemReviewed.name, | ||||||
|  |             description: review.reviewBody, | ||||||
|  |           }, | ||||||
|  |           body: review.reviewBody, | ||||||
|  |         }} | ||||||
|  |       /> | ||||||
|  |     )) | ||||||
|  |   } | ||||||
|  | </Layout> | ||||||
							
								
								
									
										71
									
								
								src/pages/resources/recipes/[recipeName].astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/pages/resources/recipes/[recipeName].astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import { useTranslatedPath } from "@i18n/utils"; | ||||||
|  | import markdownToText from "@helpers/markdownToText"; | ||||||
|  | import * as memorium from "@helpers/memorium"; | ||||||
|  |  | ||||||
|  | const path = useTranslatedPath(Astro.url); | ||||||
|  |  | ||||||
|  | const collection = "resources/recipes"; | ||||||
|  |  | ||||||
|  | export async function getStaticPaths() { | ||||||
|  |  | ||||||
|  |     const recipes = await memorium.listResource("Recipes/*"); | ||||||
|  |  | ||||||
|  |     const paths = recipes.map((recipe: any) => { | ||||||
|  |       return { | ||||||
|  |         params: { | ||||||
|  |           recipeName: recipe.identifier | ||||||
|  |             .replace("Recipes/", "") | ||||||
|  |             .replace(/\.md$/, ""), | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     return paths; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const recipes = await memorium.listResource( | ||||||
|  |   //@ts-ignore | ||||||
|  |   `Recipes/${Astro.params.recipeName}.md`, | ||||||
|  | ); | ||||||
|  | if (recipes.length === 0) { | ||||||
|  |   return new Response(null, { | ||||||
|  |     status: 404, | ||||||
|  |     statusText: "Not found", | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | const recipe = recipes[0]; | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   <div class="top-info flex items-center place-content-between m-y-2"> | ||||||
|  |     <a class="flex items-center gap-1 opacity-50" href={path("/" + collection)}> | ||||||
|  |       <span class="i-tabler-arrow-left"></span> back | ||||||
|  |     </a> | ||||||
|  |     <div class="date opacity-50"> | ||||||
|  |       { | ||||||
|  |         recipe.date?.toLocaleString("en-US", { | ||||||
|  |           month: "long", | ||||||
|  |           day: "numeric", | ||||||
|  |           year: "numeric", | ||||||
|  |         }) | ||||||
|  |       } | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  |  | ||||||
|  |   <h1>{recipe.name}</h1> | ||||||
|  |  | ||||||
|  |   <h3>Ingredients</h3> | ||||||
|  |   <ol> | ||||||
|  |     { | ||||||
|  |       recipe.recipeIngredient?.map((ingredient: any) => ( | ||||||
|  |         <li>{markdownToText(ingredient)}</li> | ||||||
|  |       )) | ||||||
|  |     } | ||||||
|  |   </ol> | ||||||
|  |  | ||||||
|  |   <h3>Instructions</h3> | ||||||
|  |   <p>{recipe.recipeInstructions}</p> | ||||||
|  | </Layout> | ||||||
							
								
								
									
										23
									
								
								src/pages/resources/recipes/index.astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/pages/resources/recipes/index.astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import HeroCard from "@components/HeroCard.astro"; | ||||||
|  | import * as memorium from "@helpers/memorium"; | ||||||
|  |  | ||||||
|  | const recipes = await memorium.listResource("Recipes/*"); | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   { | ||||||
|  |     recipes.map((recipe: any) => ( | ||||||
|  |       <HeroCard | ||||||
|  |         post={{ | ||||||
|  |           collection: "resources/recipes", | ||||||
|  |           id: recipe.identifier.replace("Recipes/", "").replace(/\.md$/, ""), | ||||||
|  |           data: { | ||||||
|  |             title: recipe.name, | ||||||
|  |           }, | ||||||
|  |         }} | ||||||
|  |       /> | ||||||
|  |     )) | ||||||
|  |   } | ||||||
|  | </Layout> | ||||||
							
								
								
									
										58
									
								
								src/pages/resources/series/[seriesName].astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/pages/resources/series/[seriesName].astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import { useTranslatedPath } from "@i18n/utils"; | ||||||
|  | import * as memorium from "@helpers/memorium"; | ||||||
|  |  | ||||||
|  | const collection = "resources/series"; | ||||||
|  |  | ||||||
|  | const path = useTranslatedPath(Astro.url); | ||||||
|  |  | ||||||
|  | export async function getStaticPaths() { | ||||||
|  |   const seriesReviews = await memorium.listResource("Media/series/*"); | ||||||
|  |  | ||||||
|  |   const paths = seriesReviews.map((review: any) => { | ||||||
|  |     return { | ||||||
|  |       params: { | ||||||
|  |         seriesName: review.identifier | ||||||
|  |           .replace("Media/series/", "") | ||||||
|  |           .replace(/\.md$/, ""), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   return paths; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const reviews = await memorium.listResource( | ||||||
|  |   //@ts-ignore | ||||||
|  |   `Media/series/${Astro.params.seriesName}.md`, | ||||||
|  | ); | ||||||
|  | if (reviews.length === 0) { | ||||||
|  |   return new Response(null, { | ||||||
|  |     status: 404, | ||||||
|  |     statusText: "Not found", | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | const review = reviews[0]; | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   <div class="top-info flex items-center place-content-between m-y-2"> | ||||||
|  |     <a class="flex items-center gap-1 opacity-50" href={path("/" + collection)}> | ||||||
|  |       <span class="i-tabler-arrow-left"></span> back | ||||||
|  |     </a> | ||||||
|  |     <div class="date opacity-50"> | ||||||
|  |       { | ||||||
|  |         review.date?.toLocaleString("en-US", { | ||||||
|  |           month: "long", | ||||||
|  |           day: "numeric", | ||||||
|  |           year: "numeric", | ||||||
|  |         }) | ||||||
|  |       } | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  |  | ||||||
|  |   <h1>{review.itemReviewed?.name}</h1> | ||||||
|  |   <p>{review.reviewBody}</p> | ||||||
|  | </Layout> | ||||||
							
								
								
									
										27
									
								
								src/pages/resources/series/index.astro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								src/pages/resources/series/index.astro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | --- | ||||||
|  | import Layout from "@layouts/Layout.astro"; | ||||||
|  | import HeroCard from "@components/HeroCard.astro"; | ||||||
|  | import * as memorium from "@helpers/memorium"; | ||||||
|  |  | ||||||
|  | const seriesReviewes = await memorium.listResource("Media/series/*"); | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <Layout title="Max Richter"> | ||||||
|  |   { | ||||||
|  |     seriesReviewes.map((review: any) => ( | ||||||
|  |       <HeroCard | ||||||
|  |         post={{ | ||||||
|  |           collection: "resources/series", | ||||||
|  |           id: review.identifier | ||||||
|  |             .replace("Media/series/", "") | ||||||
|  |             .replace(/\.md$/, ""), | ||||||
|  |           data: { | ||||||
|  |             title: review.itemReviewed.name, | ||||||
|  |             description: review.reviewBody, | ||||||
|  |           }, | ||||||
|  |           body: review.reviewBody, | ||||||
|  |         }} | ||||||
|  |       /> | ||||||
|  |     )) | ||||||
|  |   } | ||||||
|  | </Layout> | ||||||
		Reference in New Issue
	
	Block a user