github-actions[bot]
GitHub deploy: 4d024c91d61f15a8b39171610ab1406915ef598d
d6703a1
<script lang="ts">
import dayjs from 'dayjs';
interface Props {
data: { date: string; models: Record<string, number> }[];
models: string[];
colors: string[];
height?: number;
period?: 'hour' | 'week' | 'month' | 'year' | 'all';
}
let { data, models, colors, height = 300, period = 'week' }: Props = $props();
let hoveredIdx: number | null = $state(null);
let mouseX = $state(0);
let colorMap = $derived(new Map(models.map((n, i) => [n, colors[i % colors.length]])));
let maxCount = $derived(Math.max(...data.flatMap((d) => Object.values(d.models || {})), 1));
const pad = { t: 8, r: 0, b: 20, l: 0 };
const w = 1000;
let cw = $derived(w - pad.l - pad.r);
let ch = $derived(height - pad.t - pad.b);
const getX = (i: number) =>
data.length <= 1 ? pad.l + cw / 2 : pad.l + (i / (data.length - 1)) * cw;
const getY = (v: number) => pad.t + ch - (v / maxCount) * ch;
const path = (m: string) => {
const pts = data.map((d, i) => `${getX(i)},${getY(d.models?.[m] || 0)}`);
return pts.length > 1 ? `M${pts.join('L')}` : '';
};
const onMove = (e: MouseEvent) => {
const svg = e.currentTarget as SVGSVGElement;
const r = svg.getBoundingClientRect();
mouseX = (e.clientX - r.left) * (w / r.width);
hoveredIdx = Math.max(
0,
Math.min(data.length - 1, Math.round(((mouseX - pad.l) / cw) * (data.length - 1)))
);
};
let hovered = $derived(hoveredIdx !== null ? data[hoveredIdx] : null);
</script>
<div class="relative w-full" style="height:{height}px">
<svg
viewBox="0 0 {w} {height - 20}"
class="h-[calc(100%-20px)] w-full"
preserveAspectRatio="none"
onmousemove={onMove}
onmouseleave={() => (hoveredIdx = null)}
>
{#each models as m}
<path
d={path(m)}
fill="none"
stroke={colorMap.get(m)}
stroke-width="1.5"
class={hovered && !hovered.models?.[m] ? 'opacity-20' : ''}
/>
{/each}
{#if hoveredIdx !== null}
<line
x1={getX(hoveredIdx)}
y1={pad.t}
x2={getX(hoveredIdx)}
y2={ch + pad.t}
stroke="#ddd"
stroke-width="1"
/>
{#each models as m}
{@const v = hovered?.models?.[m] || 0}
{#if v > 0}
<circle cx={getX(hoveredIdx)} cy={getY(v)} r="3" fill={colorMap.get(m)} />
{/if}
{/each}
{/if}
</svg>
<!-- X-axis labels as HTML -->
{#if data.length > 1}
{@const labelCount = Math.min(7, data.length)}
{@const step = labelCount > 1 ? Math.floor((data.length - 1) / (labelCount - 1)) || 1 : 1}
{@const isHourly = data[0]?.date?.includes(':')}
{@const dateFormat = isHourly
? 'h A'
: period === 'year' || period === 'all'
? 'M/D/YY'
: 'M/D'}
<div class="flex justify-between px-0.5 text-[10px] text-gray-400">
{#each Array(labelCount) as _, i}
{@const idx = i === labelCount - 1 ? data.length - 1 : Math.min(i * step, data.length - 1)}
{#if data[idx]}
<span class={i === 0 ? 'text-left' : i === labelCount - 1 ? 'text-right' : 'text-center'}
>{dayjs(data[idx].date).format(dateFormat)}</span
>
{/if}
{/each}
</div>
{/if}
{#if hovered}
{@const total = Object.values(hovered.models || {}).reduce((a, b) => a + b, 0)}
<div
class="pointer-events-none absolute top-1 text-[11px]"
style="left:{Math.min(Math.max((mouseX / w) * 100, 8), 92)}%"
>
<div
class="min-w-[140px] -translate-x-1/2 rounded border border-gray-100 bg-white px-2.5 py-1.5 shadow-sm dark:border-gray-800 dark:bg-gray-900"
>
<div class="mb-1.5 text-[10px] text-gray-400">
{#if hovered.date?.includes(':')}
{dayjs(hovered.date).format('MMM D, h A')}
{:else}
{dayjs(hovered.date).format('MMM D, YYYY')}
{/if}
</div>
{#each Object.entries(hovered.models || {})
.sort(([, a], [, b]) => b - a)
.slice(0, 5) as [n, c]}
<div class="flex items-center justify-between gap-2 py-0.5">
<span class="min-w-0 truncate text-gray-600 dark:text-gray-300">{n}</span>
<span class="shrink-0 text-gray-900 tabular-nums dark:text-white"
>{c.toLocaleString()}
<span class="text-gray-400">({total > 0 ? ((c / total) * 100).toFixed(0) : 0}%)</span
></span
>
</div>
{/each}
</div>
</div>
{/if}
</div>