- Adicionar tokens semânticos de estado (success, warning, info) no globals.css - Migrar ~60+ componentes de cores hardcoded do Tailwind para tokens semânticos - Unificar 3 arrays duplicados de cores de categorias em importação única - Corrigir widgets de boleto/fatura que não atualizavam após pagamento (actions de fatura e antecipação não invalidavam cache do dashboard) - Corrigir scroll em listas Popover+Command (modal prop) - Adicionar link "detalhes" no card de orçamento para página da categoria - Adicionar indicadores de tendência coloridos nos cards de métricas - Estender cores de chart de 6 para 10 - Normalizar dark mode e remover tokens não utilizados Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
123 lines
3.2 KiB
TypeScript
123 lines
3.2 KiB
TypeScript
import {
|
|
RiArrowDownLine,
|
|
RiArrowDownSFill,
|
|
RiArrowUpLine,
|
|
RiArrowUpSFill,
|
|
RiCashLine,
|
|
RiIncreaseDecreaseLine,
|
|
RiSubtractLine,
|
|
} from "@remixicon/react";
|
|
import {
|
|
Card,
|
|
CardAction,
|
|
CardFooter,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from "@/components/ui/card";
|
|
import type { DashboardCardMetrics } from "@/lib/dashboard/metrics";
|
|
import MoneyValues from "../money-values";
|
|
|
|
type SectionCardsProps = {
|
|
metrics: DashboardCardMetrics;
|
|
};
|
|
|
|
type Trend = "up" | "down" | "flat";
|
|
|
|
const TREND_THRESHOLD = 0.005;
|
|
|
|
const CARDS = [
|
|
{
|
|
label: "Receitas",
|
|
key: "receitas",
|
|
icon: RiArrowUpLine,
|
|
invertTrend: false,
|
|
},
|
|
{
|
|
label: "Despesas",
|
|
key: "despesas",
|
|
icon: RiArrowDownLine,
|
|
invertTrend: true,
|
|
},
|
|
{
|
|
label: "Balanço",
|
|
key: "balanco",
|
|
icon: RiIncreaseDecreaseLine,
|
|
invertTrend: false,
|
|
},
|
|
{ label: "Previsto", key: "previsto", icon: RiCashLine, invertTrend: false },
|
|
] as const;
|
|
|
|
const TREND_ICONS = {
|
|
up: RiArrowUpSFill,
|
|
down: RiArrowDownSFill,
|
|
flat: RiSubtractLine,
|
|
} as const;
|
|
|
|
const getTrend = (current: number, previous: number): Trend => {
|
|
const diff = current - previous;
|
|
if (diff > TREND_THRESHOLD) return "up";
|
|
if (diff < -TREND_THRESHOLD) return "down";
|
|
return "flat";
|
|
};
|
|
|
|
const getPercentChange = (current: number, previous: number): string => {
|
|
const EPSILON = 0.01; // Considera valores menores que 1 centavo como zero
|
|
|
|
if (Math.abs(previous) < EPSILON) {
|
|
if (Math.abs(current) < EPSILON) return "0%";
|
|
return "—";
|
|
}
|
|
|
|
const change = ((current - previous) / Math.abs(previous)) * 100;
|
|
return Number.isFinite(change) && Math.abs(change) < 1000000
|
|
? `${change > 0 ? "+" : ""}${change.toFixed(1)}%`
|
|
: "—";
|
|
};
|
|
|
|
const getTrendColor = (trend: Trend, invertTrend: boolean): string => {
|
|
if (trend === "flat") return "";
|
|
const isPositive = invertTrend ? trend === "down" : trend === "up";
|
|
return isPositive
|
|
? "text-success border-success"
|
|
: "text-destructive border-destructive";
|
|
};
|
|
|
|
export function SectionCards({ metrics }: SectionCardsProps) {
|
|
return (
|
|
<div className="*:data-[slot=card]:from-primary/5 *:data-[slot=card]:to-card dark:*:data-[slot=card]:bg-card grid grid-cols-1 gap-3 @xl/main:grid-cols-2 @5xl/main:grid-cols-4">
|
|
{CARDS.map(({ label, key, icon: Icon, invertTrend }) => {
|
|
const metric = metrics[key];
|
|
const trend = getTrend(metric.current, metric.previous);
|
|
const TrendIcon = TREND_ICONS[trend];
|
|
const trendColor = getTrendColor(trend, invertTrend);
|
|
|
|
return (
|
|
<Card key={label} className="@container/card gap-2">
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-1">
|
|
<Icon className="size-4 text-primary" />
|
|
{label}
|
|
</CardTitle>
|
|
<MoneyValues className="text-2xl" amount={metric.current} />
|
|
<CardAction>
|
|
<div className={`flex items-center text-xs ${trendColor}`}>
|
|
<TrendIcon size={16} />
|
|
{getPercentChange(metric.current, metric.previous)}
|
|
</div>
|
|
</CardAction>
|
|
</CardHeader>
|
|
<CardFooter className="flex-col items-start gap-2 text-sm">
|
|
<div className="line-clamp-1 flex gap-2 text-xs">
|
|
Mês anterior
|
|
</div>
|
|
<div className="text-muted-foreground">
|
|
<MoneyValues amount={metric.previous} />
|
|
</div>
|
|
</CardFooter>
|
|
</Card>
|
|
);
|
|
})}
|
|
</div>
|
|
);
|
|
}
|