196 lines
8.9 KiB
TypeScript
196 lines
8.9 KiB
TypeScript
import Image from 'next/image';
|
||
import { Inter } from 'next/font/google';
|
||
import { format } from 'timeago.js'
|
||
import { promises as fs } from 'fs';
|
||
import path from 'path';
|
||
|
||
const inter = Inter({ subsets: ['latin'] });
|
||
|
||
interface Article {
|
||
date: string;
|
||
title: string;
|
||
url: string;
|
||
}
|
||
|
||
interface Dataset {
|
||
url: string;
|
||
name: string;
|
||
displayName: string;
|
||
articles: Article[];
|
||
}
|
||
|
||
export function MobileItem({dataset} : { dataset: Dataset}) {
|
||
return (
|
||
<div className="flex gap-x-2 pb-2 py-4 items-center justify-between border-b border-zinc-600">
|
||
<div className="flex flex-col">
|
||
<span className="font-light">{dataset.name}</span>
|
||
{dataset.articles.map((article) => (
|
||
<div className='py-1 flex flex-col'>
|
||
<span className="font-bold hover:underline">{article.title}</span>
|
||
<span className="font-light text-base">{format(article.date)}</span>{' '}
|
||
</div>
|
||
))}
|
||
</div>
|
||
<div className="flex flex-col justify-start">
|
||
<a
|
||
className="border border-zinc-900 font-light px-4 py-1 text-sm transition hover:bg-zinc-900 hover:text-white"
|
||
href={dataset.url}
|
||
target="_blank"
|
||
>
|
||
info
|
||
</a>
|
||
{/*
|
||
<button>
|
||
<svg
|
||
xmlns="http://www.w3.org/2000/svg"
|
||
viewBox="0 0 24 24"
|
||
fill="currentColor"
|
||
className="w-12 h-12 text-blue-400 hover:text-blue-300 transition mt-1"
|
||
>
|
||
<path
|
||
fillRule="evenodd"
|
||
d="M12 2.25c-5.385 0-9.75 4.365-9.75 9.75s4.365 9.75 9.75 9.75 9.75-4.365 9.75-9.75S17.385 2.25 12 2.25zm-.53 14.03a.75.75 0 001.06 0l3-3a.75.75 0 10-1.06-1.06l-1.72 1.72V8.25a.75.75 0 00-1.5 0v5.69l-1.72-1.72a.75.75 0 00-1.06 1.06l3 3z"
|
||
clipRule="evenodd"
|
||
/>
|
||
</svg>
|
||
</button> */}
|
||
</div>
|
||
</div>
|
||
);
|
||
}
|
||
|
||
export function DesktopItem({dataset} : { dataset: Dataset}) {
|
||
return (
|
||
<>
|
||
{dataset.articles.map((article, index) => (
|
||
<tr className={`${index === (dataset.articles.length - 1) ? 'border-b' : ''} border-zinc-400`}>
|
||
<td className="py-8 font-light">{index === 0 ? dataset.name : ''}</td>
|
||
<td>
|
||
<a className="py-8 font-bold hover:underline" href={article.url}>
|
||
{article.title}
|
||
</a>
|
||
</td>
|
||
<td className="py-8 font-light text-base min-w-[120px]">{format(article.date)}</td>
|
||
<td className="py-8">
|
||
{index === 0 && (
|
||
<a
|
||
className="border border-zinc-900 font-light px-[25px] py-2.5 text-sm transition hover:bg-zinc-900 hover:text-white"
|
||
href={dataset.url}
|
||
target="_blank"
|
||
>
|
||
info
|
||
</a>
|
||
)}
|
||
</td>
|
||
{/*
|
||
<td>
|
||
<button>
|
||
<svg
|
||
xmlns="http://www.w3.org/2000/svg"
|
||
viewBox="0 0 24 24"
|
||
fill="currentColor"
|
||
className="w-12 h-12 text-blue-400 hover:text-blue-300 transition mt-1"
|
||
>
|
||
<path
|
||
fillRule="evenodd"
|
||
d="M12 2.25c-5.385 0-9.75 4.365-9.75 9.75s4.365 9.75 9.75 9.75 9.75-4.365 9.75-9.75S17.385 2.25 12 2.25zm-.53 14.03a.75.75 0 001.06 0l3-3a.75.75 0 10-1.06-1.06l-1.72 1.72V8.25a.75.75 0 00-1.5 0v5.69l-1.72-1.72a.75.75 0 00-1.06 1.06l3 3z"
|
||
clipRule="evenodd"
|
||
/>
|
||
</svg>
|
||
</button>
|
||
</td>*/}
|
||
</tr>
|
||
))}
|
||
</>
|
||
);
|
||
}
|
||
|
||
export async function getStaticProps() {
|
||
const jsonDirectory = path.join(
|
||
process.cwd(),
|
||
'/datasets.json'
|
||
);
|
||
const datasetString = await fs.readFile(jsonDirectory, 'utf8');
|
||
const datasets = JSON.parse(datasetString)
|
||
return {
|
||
props: { datasets },
|
||
};
|
||
}
|
||
|
||
export default function Home( { datasets }: { datasets: Dataset[] }) {
|
||
return (
|
||
<>
|
||
<header className="max-w-5xl mx-auto mt-8 w-full">
|
||
<div className="border-b-2 pb-2.5 mx-2 border-zinc-800">
|
||
<h1>
|
||
<span className="sr-only">FiveThirtyEight</span>
|
||
<a className='flex gap-x-2 items-center' href="http://fivethirtyeight.com">
|
||
<img
|
||
width="197"
|
||
height="25"
|
||
alt="FiveThirtyEight"
|
||
src=""
|
||
/> by PortalJS
|
||
</a>
|
||
</h1>
|
||
</div>
|
||
</header>
|
||
<main
|
||
className={`flex min-h-screen flex-col items-center max-w-5xl mx-auto pt-20 px-2.5 ${inter.className}`}
|
||
>
|
||
<div>
|
||
<h1 className="text-[40px] font-bold text-zinc-800 text-center">
|
||
Our Data
|
||
</h1>
|
||
<p className="max-w-2xl text-lg text-center text-zinc-700">
|
||
We’re sharing the data and code behind some of our articles and
|
||
graphics. We hope you’ll use it to check our work and to create
|
||
stories and visualizations of your own.
|
||
</p>
|
||
</div>
|
||
<article className="w-full px-2 md:hidden py-4">{datasets.map(dataset => <MobileItem dataset={dataset} />)}</article>
|
||
<table className="w-full mt-10 mb-4 hidden md:table">
|
||
<thead className="border-b-4 pb-2 border-zinc-900">
|
||
<tr>
|
||
<th className="uppercase text-left font-light text-xs pb-3">
|
||
data set
|
||
</th>
|
||
<th className="uppercase text-left font-light text-xs pb-3">
|
||
related content
|
||
</th>
|
||
<th className="uppercase text-left font-light text-xs pb-3">
|
||
last updated
|
||
</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>{datasets.map(dataset => <DesktopItem dataset={dataset} />)}</tbody>
|
||
</table>
|
||
<p className="text-[13px] py-8">
|
||
Unless otherwise noted, our data sets are available under the{' '}
|
||
<a
|
||
className="text-blue-400 hover:underline"
|
||
href="http://creativecommons.org/licenses/by/4.0/"
|
||
>
|
||
Creative Commons Attribution 4.0 International license
|
||
</a>
|
||
, and the code is available under the{' '}
|
||
<a
|
||
className="text-blue-400 hover:underline"
|
||
href="http://opensource.org/licenses/MIT"
|
||
>
|
||
MIT license
|
||
</a>
|
||
. If you find this information useful, please{' '}
|
||
<a
|
||
className="text-blue-400 hover:underline"
|
||
href="mailto:data@fivethirtyeight.com"
|
||
>
|
||
let us know
|
||
</a>
|
||
.
|
||
</p>
|
||
</main>
|
||
</>
|
||
);
|
||
}
|