1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
| import { IconType } from "react-icons"; import { VariantProps, cva } from "class-variance-authority";
import { Skeleton } from "./ui/skeleton"; import { cn, formatCurrency, formatPercentage } from "@/lib/utils";
import { Card, CardContent, CardDescription, CardHeader, CardTitle, } from "@/components/ui/card";
import { CountUp } from "@/components/count-up";
const boxVariant = cva( "shrink-0 rounded-md p-3", { variants:{ variant: { default: "fill-blue-500", success: "fill-emerald-500", danger: "fill-rose-500", warning: "fill-yellow-500", } }, defaultVariants: { variant: "default", }, }, );
const iconVariant = cva( "size-6", { variants:{ variant: { default: "fill-blue-500", success: "fill-emerald-500", danger: "fill-rose-500", warning: "fill-yellow-500", } }, defaultVariants: { variant: "default", }, }, );
type BoxVariants = VariantProps<typeof boxVariant>; type IconVariants = VariantProps<typeof iconVariant>;
interface DataCardProps extends BoxVariants, IconVariants { icon: IconType; title: string; value?: number; dateRange: string; percentageChange?: number; }
export const DataCard = ({ icon: Icon, title, value = 0, variant, dateRange, percentageChange = 0, }:DataCardProps) => { return ( <Card className="border-none drop-shadow-sm"> <CardHeader className="flex flex-row items-center justify-between gap-x-4"> <div className="space-y-2"> <CardTitle className="text-2xl line-clamp-1"> {title} </CardTitle> <CardDescription className="line-clamp-1"> {dateRange} </CardDescription> </div> <div className={cn(boxVariant({variant}))}> <Icon className={cn(iconVariant({ variant }))} /> </div> </CardHeader> <CardContent> <h1 className="font-bold text-2xl mb-2 line-clamp-1 break-all"> <CountUp preserveValue start={0} end={value} decimals={2} decimalPlaces={2} formattingFn={formatCurrency} /> </h1> <p className={cn( "text-muted-foreground text-sm line-clamp-1", percentageChange >= 0 && "text-emerald-500", percentageChange < 0 && "text-rose-500", )}> {formatPercentage(percentageChange)} from last period </p> </CardContent> </Card> ) }
export const DateCardLoading =() => { return ( <Card className="border-none drop-shadow-sm h-[192px]"> <CardHeader className="flex flex-row items-center justify-between gap-x-4"> <div className="space-y-2"> <Skeleton className="h-6 w-24" /> <Skeleton className="h-4 w-40" /> </div> <Skeleton className="size-12" /> </CardHeader> <CardContent> <Skeleton className="shrink-0 h-10 w-24 mb-2" /> <Skeleton className="shrink-0 h-4 w-40" /> </CardContent> </Card> ); };
|