mirror of
https://github.com/shadcn-ui/ui.git
synced 2026-06-11 09:51:40 +00:00
* feat: init * fix * fix * fix * feat * feat * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: implement icons * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: update init command * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: dialog * feat * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: add registry:base item type * feat: rename frame to canva * fix * feat * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fi * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: add all colors * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: add outfit font * fix * fix * fix * fix * fix * chore: changeset * fix * fix * fix * fix * fix * fix * fix * fix
190 lines
5.3 KiB
TypeScript
190 lines
5.3 KiB
TypeScript
"use client"
|
|
|
|
import * as React from "react"
|
|
import { Label, Pie, PieChart, Sector } from "recharts"
|
|
import { type PieSectorDataItem } from "recharts/types/polar/Pie"
|
|
|
|
import {
|
|
Card,
|
|
CardAction,
|
|
CardContent,
|
|
CardDescription,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from "@/registry/new-york-v4/ui/card"
|
|
import {
|
|
ChartContainer,
|
|
ChartStyle,
|
|
ChartTooltip,
|
|
ChartTooltipContent,
|
|
type ChartConfig,
|
|
} from "@/registry/new-york-v4/ui/chart"
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "@/registry/new-york-v4/ui/select"
|
|
|
|
const desktopData = [
|
|
{ month: "january", desktop: 186, fill: "var(--color-january)" },
|
|
{ month: "february", desktop: 305, fill: "var(--color-february)" },
|
|
{ month: "march", desktop: 237, fill: "var(--color-march)" },
|
|
{ month: "april", desktop: 173, fill: "var(--color-april)" },
|
|
{ month: "may", desktop: 209, fill: "var(--color-may)" },
|
|
]
|
|
|
|
const chartConfig = {
|
|
visitors: {
|
|
label: "Visitors",
|
|
},
|
|
desktop: {
|
|
label: "Desktop",
|
|
},
|
|
mobile: {
|
|
label: "Mobile",
|
|
},
|
|
january: {
|
|
label: "January",
|
|
color: "var(--chart-1)",
|
|
},
|
|
february: {
|
|
label: "February",
|
|
color: "var(--chart-2)",
|
|
},
|
|
march: {
|
|
label: "March",
|
|
color: "var(--chart-3)",
|
|
},
|
|
april: {
|
|
label: "April",
|
|
color: "var(--chart-4)",
|
|
},
|
|
may: {
|
|
label: "May",
|
|
color: "var(--chart-5)",
|
|
},
|
|
} satisfies ChartConfig
|
|
|
|
export function ChartVisitors() {
|
|
const id = "pie-interactive"
|
|
const [activeMonth, setActiveMonth] = React.useState(desktopData[0].month)
|
|
|
|
const activeIndex = React.useMemo(
|
|
() => desktopData.findIndex((item) => item.month === activeMonth),
|
|
[activeMonth]
|
|
)
|
|
const months = React.useMemo(() => desktopData.map((item) => item.month), [])
|
|
|
|
return (
|
|
<Card data-chart={id}>
|
|
<ChartStyle id={id} config={chartConfig} />
|
|
<CardHeader>
|
|
<CardDescription>January - June 2024</CardDescription>
|
|
<CardTitle className="text-2xl font-bold">1,234 visitors</CardTitle>
|
|
<CardAction>
|
|
<Select value={activeMonth} onValueChange={setActiveMonth}>
|
|
<SelectTrigger
|
|
className="ml-auto h-8 w-[120px]"
|
|
aria-label="Select a value"
|
|
>
|
|
<SelectValue placeholder="Select month" />
|
|
</SelectTrigger>
|
|
<SelectContent align="end">
|
|
{months.map((key) => {
|
|
const config = chartConfig[key as keyof typeof chartConfig]
|
|
|
|
if (!config) {
|
|
return null
|
|
}
|
|
|
|
const color = "color" in config ? config.color : undefined
|
|
|
|
return (
|
|
<SelectItem key={key} value={key}>
|
|
<div className="flex items-center gap-2 text-xs">
|
|
<span
|
|
className="flex h-3 w-3 shrink-0 rounded-sm"
|
|
style={{
|
|
backgroundColor: color,
|
|
}}
|
|
/>
|
|
{config?.label}
|
|
</div>
|
|
</SelectItem>
|
|
)
|
|
})}
|
|
</SelectContent>
|
|
</Select>
|
|
</CardAction>
|
|
</CardHeader>
|
|
<CardContent className="flex flex-1 justify-center pb-0">
|
|
<ChartContainer
|
|
id={id}
|
|
config={chartConfig}
|
|
className="mx-auto aspect-square w-full max-w-[300px]"
|
|
>
|
|
<PieChart>
|
|
<ChartTooltip
|
|
cursor={false}
|
|
content={<ChartTooltipContent hideLabel />}
|
|
/>
|
|
<Pie
|
|
data={desktopData}
|
|
dataKey="desktop"
|
|
nameKey="month"
|
|
innerRadius={60}
|
|
strokeWidth={5}
|
|
activeIndex={activeIndex}
|
|
activeShape={({
|
|
outerRadius = 0,
|
|
...props
|
|
}: PieSectorDataItem) => (
|
|
<g>
|
|
<Sector {...props} outerRadius={outerRadius + 10} />
|
|
<Sector
|
|
{...props}
|
|
outerRadius={outerRadius + 25}
|
|
innerRadius={outerRadius + 12}
|
|
/>
|
|
</g>
|
|
)}
|
|
>
|
|
<Label
|
|
content={({ viewBox }) => {
|
|
if (viewBox && "cx" in viewBox && "cy" in viewBox) {
|
|
return (
|
|
<text
|
|
x={viewBox.cx}
|
|
y={viewBox.cy}
|
|
textAnchor="middle"
|
|
dominantBaseline="middle"
|
|
>
|
|
<tspan
|
|
x={viewBox.cx}
|
|
y={viewBox.cy}
|
|
className="fill-foreground text-3xl font-bold"
|
|
>
|
|
{desktopData[activeIndex].desktop.toLocaleString()}
|
|
</tspan>
|
|
<tspan
|
|
x={viewBox.cx}
|
|
y={(viewBox.cy || 0) + 24}
|
|
className="fill-muted-foreground"
|
|
>
|
|
Visitors
|
|
</tspan>
|
|
</text>
|
|
)
|
|
}
|
|
}}
|
|
/>
|
|
</Pie>
|
|
</PieChart>
|
|
</ChartContainer>
|
|
</CardContent>
|
|
</Card>
|
|
)
|
|
}
|