Components
Chart
Data visualization components built with Recharts for creating interactive charts and graphs.
Bar Chart
Trends in monthly revenue.
Installation
Install Required Dependicies
npm install recharts
Copy and paste the following code into your project
import React from "react"
import { VariantProps, cn, tv } from "@mijn-ui/react"
import { ResponsiveContainer, Tooltip } from "recharts"
/* -------------------------------------------------------------------------- */
/* ChartToolTip */
/* -------------------------------------------------------------------------- */
export const tooltipStyles = tv({
slots: {
base: "flex aspect-video flex-col justify-center rounded-medium border bg-card text-xs shadow-large",
label: "block font-medium",
separator: "my-1.5 h-divider bg-border",
content: "space-y-1",
entry: "flex items-center justify-between gap-4",
entryValue: "",
entryNameContainer: "flex items-center gap-3",
entryName: "text-muted-foreground",
entryIndicator: "",
},
variants: {
hideLabel: {
true: { base: "aspect-auto p-2" },
false: { base: "p-2.5" },
},
indicator: {
ring: {
entryIndicator:
"size-1.5 rounded-full ring ring-[var(--entry-indicator-color)]",
},
circle: {
entryIndicator:
"size-2.5 rounded-full bg-[var(--entry-indicator-color)]",
},
square: {
entryIndicator:
"size-2.5 rounded-small bg-[var(--entry-indicator-color)]",
},
line: {
entryIndicator: "h-3 w-1 bg-[var(--entry-indicator-color)]",
},
},
},
defaultVariants: {
indicator: "circle",
hideLabel: false,
},
})
export type TooltipSlots = keyof ReturnType<typeof tooltipStyles>
export type TooltipVariantProps = VariantProps<typeof tooltipStyles>
/* -------------------------------------------------------------------------- */
/* ChartContainer */
/* -------------------------------------------------------------------------- */
const ChartContainer = ({
className,
...props
}: React.ComponentProps<typeof ResponsiveContainer>) => {
return (
<ResponsiveContainer
width="100%"
height="100%"
className={cn(
"[&_.recharts-active-dot]:stroke-card [&_.recharts-cartesian-axis-tick_text]:fill-muted-foreground [&_.recharts-cartesian-grid_line]:stroke-border [&_.recharts-curve.recharts-tooltip-cursor]:stroke-border [&_.recharts-dot[stroke='#fff']]:stroke-transparent [&_.recharts-layer]:outline-none [&_.recharts-rectangle.recharts-tooltip-cursor]:fill-accent [&_.recharts-tooltip-cursor]:stroke-1",
className,
)}
{...props}
/>
)
}
/* -------------------------------------------------------------------------- */
/* ChartTooltip */
/* -------------------------------------------------------------------------- */
type TooltipProps = React.ComponentProps<typeof Tooltip> &
React.ComponentProps<"div"> & {
hideLabel?: boolean
hideIndicator?: boolean
indicator?: TooltipVariantProps["indicator"]
}
const ChartTooltip = ({
active,
className,
payload,
indicator = "circle",
hideIndicator = false,
hideLabel = false,
labelFormatter,
formatter,
color,
label,
labelClassName,
}: TooltipProps) => {
const singleEntry = payload?.length === 1
const {
base,
separator,
label: labelClasses,
content,
entryName,
entry: entryClasses,
entryIndicator,
entryNameContainer,
entryValue,
} = tooltipStyles({ hideLabel: hideLabel || singleEntry })
const LabelComponent = () => {
if (hideLabel || !payload?.length || singleEntry) {
return null
}
return (
<span className={labelClasses({ className: labelClassName })}>
{labelFormatter ? labelFormatter(label, payload) : label}
</span>
)
}
if (!active || !payload?.length) {
return null
}
return (
<div
role="tooltip"
className={base({
className: cn("relative z-50", className),
})}
>
<LabelComponent />
{!singleEntry && !hideLabel && (
<div role="separator" className={separator()} />
)}
<div className={content()}>
{payload.map((entry, index) =>
formatter && entry?.value !== undefined && entry.name ? (
formatter(entry.value, entry.name, entry, index, entry.payload)
) : (
<div
key={entry.name || entry.dataKey || "value"}
className={entryClasses()}
>
<div className={entryNameContainer()}>
{!hideIndicator && (
<div
aria-hidden
className={entryIndicator({ indicator })}
style={
{
"--entry-indicator-color": color || entry.color,
} as React.CSSProperties
}
/>
)}
<span className={entryName()}>{entry.name}</span>
</div>
{entry.value && (
<span className={entryValue()}>{entry.value}</span>
)}
</div>
),
)}
</div>
</div>
)
}
/* -------------------------------------------------------------------------- */
/* ChartLegend */
/* -------------------------------------------------------------------------- */
const ChartLegend = ({
value,
className,
}: React.ComponentPropsWithRef<"span"> & { value: string }) => {
return (
<span
className={cn(
"ml-1 mr-3 text-sm font-medium text-muted-foreground",
className,
)}
>
{value}
</span>
)
}
export { ChartTooltip, ChartLegend, ChartContainer }
Usage
Chart utilities work alongside Recharts components to provide consistent styling and simplify common patterns. Here's how to use them:
import {
Bar,
BarChart,
CartesianGrid,
Legend,
Tooltip,
XAxis,
YAxis
} from "recharts"
import { ChartContainer, ChartTooltip, ChartLegend } from "@/components/chart"
// Sample data
const data = [
{ month: "Jan", revenue: 12000, expenses: 8000 },
{ month: "Feb", revenue: 15000, expenses: 7500 },
{ month: "Mar", revenue: 18000, expenses: 9000 },
// ...more data
]
export function RevenueChart() {
return (
<div className="h-64 w-full">
<ChartContainer>
<BarChart data={data}>
{/* Grid and axes */}
<CartesianGrid vertical={false} strokeDasharray="3" />
<XAxis
dataKey="month"
tickLine={false}
axisLine={false}
/>
<YAxis
tickLine={false}
axisLine={false}
tickFormatter={(value) => `$${value / 1000}k`}
/>
{/* Enhanced tooltip */}
<Tooltip content={<ChartTooltip active />} />
{/* Chart bars */}
<Bar dataKey="revenue" fill="hsl(var(--chart-1))" radius={4} />
<Bar dataKey="expenses" fill="hsl(var(--chart-2))" radius={4} />
{/* Enhanced legend */}
<Legend
formatter={(value) => <ChartLegend value={value} />}
iconType="circle"
iconSize={8}
/>
</BarChart>
</ChartContainer>
</div>
)
}
Examples
Bar Chart - Multiple With Legend
Bar Chart - Multiple With Legend
Trends in monthly revenue and expenses.
Area Chart
Area Chart - Stacked With Legend
Trends in monthly revenue and expenses.
Line Chart
Line Chart - Multiple With Legend
Trends in monthly revenue and expenses.