- My blog
+ My blog
${headingElem}>
`} />
{ distinctYears.map(year => (
- {year}
+
+ ${year}
+ ${subHeadingElem}>
+ `} />
{ posts.filter(matchesYear(year)).sort(sortByPubDateDescending).map(post => (
-
diff --git a/website/src/components/LinksFeed.astro b/website/src/components/LinksFeed.astro
new file mode 100644
index 0000000..a428395
--- /dev/null
+++ b/website/src/components/LinksFeed.astro
@@ -0,0 +1,77 @@
+---
+import FormattedDate from '../components/FormattedDate.astro';
+import LINKS from '../data/links.ts';
+
+export interface Props {
+ headingLevel?: 1 | 2 | 3 | 4 | 5 | 6,
+ hideAuthor?: boolean,
+};
+
+const { headingLevel = 2, hideAuthor = false } = Astro.props;
+
+const headingElem = `h${headingLevel}`;
+const subHeadingElem = `h${headingLevel + 1}`
+
+const distinctYears: string[] = LINKS
+ .map(link => link.isoDateAdded.slice(0,4))
+ .reduce((acc, curr) => acc.includes(curr) ? acc : [...acc, curr], [])
+ .sort((a, b) => Number.parseInt(b) - Number.parseInt(a));
+
+function matchesYear(year: string) {
+ return (link: (typeof LINKS)[number]) => link.isoDateAdded.slice(0,4) === year;
+}
+
+function sortByDateAddedDescending(link1: (typeof LINKS)[number], link2: (typeof LINKS)[number]) {
+ const year1 = Number.parseInt(link1.isoDateAdded.slice(0, 4));
+ const year2 = Number.parseInt(link2.isoDateAdded.slice(0, 4));
+ const month1 = Number.parseInt(link1.isoDateAdded.slice(5, 7));
+ const month2 = Number.parseInt(link2.isoDateAdded.slice(5, 7));
+ const day1 = Number.parseInt(link1.isoDateAdded.slice(8, 10));
+ const day2 = Number.parseInt(link2.isoDateAdded.slice(8, 10));
+
+ if (year1 !== year2) {
+ return year2 - year1;
+ } else if (month1 !== month2) {
+ return month2 - month1;
+ } else {
+ return day2 - day1;
+ }
+}
+
+const canonicalLinksUrl = new URL('links', Astro.site)
+---
+
+
+
+ My links
+ ${headingElem}>
+ `} />
+
+
+
+ { distinctYears.map(year => (
+
+ ${year}
+ ${subHeadingElem}>
+ `} />
+
+ { LINKS.filter(matchesYear(year)).sort(sortByDateAddedDescending).map(link => (
+ -
+ {link.title}.
+ {link.description}
+ Added:
+
+ )) }
+
+ )) }
+
diff --git a/website/src/components/Navbar.astro b/website/src/components/Navbar.astro
index c94944a..479f8b5 100644
--- a/website/src/components/Navbar.astro
+++ b/website/src/components/Navbar.astro
@@ -9,5 +9,8 @@
-
Blog
+ -
+ Links
+
diff --git a/website/src/data/links.ts b/website/src/data/links.ts
index cd1e746..6816149 100644
--- a/website/src/data/links.ts
+++ b/website/src/data/links.ts
@@ -7,7 +7,7 @@ type Link = {
isoDateAdded: `20${Digit}${Digit}-${'0'|'1'}${Digit}-${'0'|'1'|'2'|'3'}${Digit}`,
};
-const links: Link[] = [
+const LINKS: Link[] = [
{
href: 'https://dl.acm.org/doi/pdf/10.1145/3613904.3642596',
title: 'Is Stack Overflow Obsolete? An Empirical Study of the Characteristics of ChatGPT Answers to Stack Overflow Questions',
@@ -28,4 +28,4 @@ const links: Link[] = [
},
];
-export default links;
+export default LINKS;
diff --git a/website/src/pages/index.astro b/website/src/pages/index.astro
index 4e9d87b..835b4d3 100644
--- a/website/src/pages/index.astro
+++ b/website/src/pages/index.astro
@@ -1,5 +1,6 @@
---
import BlogFeed from '../components/BlogFeed.astro';
+import LinksFeed from '../components/LinksFeed.astro';
import Me from '../components/Me.astro';
import { SITE_TITLE, SITE_DESCRIPTION } from '../consts';
import Page from '../layouts/Page.astro';
@@ -8,4 +9,5 @@ import Page from '../layouts/Page.astro';
+
diff --git a/website/src/pages/links.astro b/website/src/pages/links.astro
new file mode 100644
index 0000000..1368e00
--- /dev/null
+++ b/website/src/pages/links.astro
@@ -0,0 +1,8 @@
+---
+import Page from '../layouts/Page.astro';
+import LinksFeed from '../components/LinksFeed.astro';
+---
+
+
+
+