Initial commit

This commit is contained in:
DK
2026-02-09 15:11:07 +00:00
commit 9ea79a56e4
656 changed files with 77428 additions and 0 deletions

View File

@@ -0,0 +1,145 @@
"use client";
import { memo } from "react";
import {
Area,
AreaChart,
CartesianGrid,
YAxis,
Tooltip,
ResponsiveContainer,
} from "recharts";
import { cls, shouldUseInvertedText } from "@/lib/utils";
import { useTheme } from "@/providers/themeProvider/ThemeProvider";
import { formatNumber, calculateYAxisWidth, type ChartDataItem } from "./utils";
import CustomTooltip from "./CustomTooltip";
import type { InvertedBackground } from "@/providers/themeProvider/config/constants";
interface BentoLineChartProps {
data?: ChartDataItem[];
dataKey?: string;
metricLabel?: string;
isPercentage?: boolean;
useInvertedBackground: InvertedBackground;
className?: string;
}
const defaultData: ChartDataItem[] = [
{ value: 120 },
{ value: 180 },
{ value: 150 },
{ value: 280 },
{ value: 220 },
{ value: 350 },
{ value: 300 },
{ value: 250 },
];
const BentoLineChart = memo<BentoLineChartProps>(
({
data = defaultData,
dataKey = "value",
metricLabel = "Value",
isPercentage = false,
useInvertedBackground,
className = "",
}) => {
const theme = useTheme();
const shouldUseLightText = shouldUseInvertedText(useInvertedBackground, theme.cardStyle);
const yAxisWidth = calculateYAxisWidth(data, isPercentage);
const strokeColor = "var(--primary-cta)";
const gridColor = "color-mix(in srgb, var(--background-accent) 30%, transparent)";
const tickColor = shouldUseLightText ? "var(--background)" : "var(--foreground)";
return (
<div
className={cls("w-full h-full **:outline-none **:focus:outline-none", className)}
style={{
maskImage: "linear-gradient(to bottom, black 40%, transparent 100%)",
WebkitMaskImage: "linear-gradient(to bottom, black 40%, transparent 100%)",
}}
>
<ResponsiveContainer width="100%" height="100%">
<AreaChart
data={data}
margin={{
top: 10,
right: 5,
left: 0,
bottom: 14,
}}
>
<defs>
<linearGradient id="bentoLineChartFill" x1="0" y1="0" x2="0" y2="1">
<stop offset="5%" stopColor={strokeColor} stopOpacity={0.4} />
<stop offset="95%" stopColor={strokeColor} stopOpacity={0} />
</linearGradient>
<linearGradient id="bentoFadeGradient" x1="0" y1="0" x2="1" y2="0">
<stop offset="0%" stopColor="black" stopOpacity={0} />
<stop offset="5%" stopColor="black" stopOpacity={0} />
<stop offset="15%" stopColor="white" stopOpacity={1} />
<stop offset="95%" stopColor="white" stopOpacity={1} />
<stop offset="100%" stopColor="black" stopOpacity={0} />
</linearGradient>
<mask id="bentoFadeMask">
<rect
x="0"
y="0"
width="100%"
height="100%"
fill="url(#bentoFadeGradient)"
/>
</mask>
</defs>
<CartesianGrid
vertical={false}
stroke={gridColor}
strokeWidth={1}
/>
<YAxis
tickLine={false}
axisLine={false}
tick={{
fill: tickColor,
fontSize: 10,
}}
width={yAxisWidth}
tickFormatter={(value) =>
isPercentage ? `${value}%` : formatNumber(value)
}
/>
<Tooltip
content={
<CustomTooltip
metricLabel={metricLabel}
isPercentage={isPercentage}
totalItems={data.length}
/>
}
cursor={{
stroke: gridColor,
}}
/>
<Area
dataKey={dataKey}
type="monotone"
fill="url(#bentoLineChartFill)"
stroke={strokeColor}
strokeWidth={2}
mask="url(#bentoFadeMask)"
activeDot={{
fill: strokeColor,
r: 5,
}}
/>
</AreaChart>
</ResponsiveContainer>
</div>
);
}
);
BentoLineChart.displayName = "BentoLineChart";
export default BentoLineChart;

View File

@@ -0,0 +1,64 @@
"use client";
import { memo } from "react";
import { formatNumber } from "./utils";
interface CustomTooltipProps {
active?: boolean;
payload?: Array<{
value: number;
color: string;
}>;
label?: number;
metricLabel?: string;
isPercentage?: boolean;
totalItems: number;
}
const CustomTooltip = memo<CustomTooltipProps>(
({
active,
payload,
label = 0,
metricLabel = "Value",
isPercentage = false,
totalItems,
}: CustomTooltipProps) => {
if (active && payload && payload.length) {
const value = isPercentage
? `${payload[0].value}%`
: formatNumber(payload[0].value);
const today = new Date();
const daysAgo = totalItems - 1 - label;
const date = new Date(today);
date.setDate(today.getDate() - daysAgo);
return (
<div className="card rounded-theme-capped p-3">
<p className="text-xs text-foreground mb-2">
{date.toLocaleDateString("en-US", {
month: "short",
day: "numeric",
year: "numeric",
})}
</p>
<div className="flex items-center gap-2">
<div
className="h-1.5 aspect-square rounded-full"
style={{
backgroundColor: payload[0].color,
}}
/>
<span className="text-xs text-foreground">
{metricLabel}: {value}
</span>
</div>
</div>
);
}
return null;
}
);
CustomTooltip.displayName = "CustomTooltip";
export default CustomTooltip;

View File

@@ -0,0 +1,33 @@
export const formatNumber = (value: number): string => {
if (value >= 100000) {
const millions = value / 1000000;
return `${millions.toFixed(1)}M`;
}
if (value >= 1000) {
const thousands = value / 1000;
const rounded = Math.round(thousands * 10) / 10;
return `${rounded}K`;
}
return value.toString();
};
export interface ChartDataItem {
value: number;
}
export const calculateYAxisWidth = (
data: ChartDataItem[],
isPercentage: boolean
): number => {
const maxValue = Math.max(...data.map((item) => item.value));
const formattedMax = isPercentage ? `${maxValue}%` : formatNumber(maxValue);
let multiplier = 9;
if (formattedMax.length === 2) {
multiplier = 11;
} else if (formattedMax.length === 3) {
multiplier = 13;
}
return formattedMax.length * multiplier;
};