105 lines
3.6 KiB
TypeScript
105 lines
3.6 KiB
TypeScript
import { Gauge, Thermometer, Waves } from "lucide-react";
|
|
import type { DashboardOverview } from "../../dashboard/types/DashboardOverview";
|
|
import { StatusPill } from "../../dashboard/components/StatusPill";
|
|
|
|
type Props = {
|
|
zones: DashboardOverview["climate"]["zones"];
|
|
};
|
|
|
|
function formatValue(value: number | null, unit: string, decimals = 1) {
|
|
if (value === null) return "--";
|
|
return `${value.toFixed(decimals)} ${unit}`;
|
|
}
|
|
|
|
export function DashboardClimateSection({ zones }: Props) {
|
|
return (
|
|
<section className="space-y-4">
|
|
<div>
|
|
<h2 className="text-lg font-semibold text-slate-900 dark:text-slate-50">
|
|
Clima por Zona
|
|
</h2>
|
|
<p className="text-sm text-slate-500 dark:text-slate-400">
|
|
Temperatura, humidade, CO₂ e estados principais dos equipamentos.
|
|
</p>
|
|
</div>
|
|
|
|
<div className="grid grid-cols-1 gap-4 xl:grid-cols-2">
|
|
{zones.map((zone) => (
|
|
<div
|
|
key={zone.zoneNumber}
|
|
className="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900"
|
|
>
|
|
<div className="mb-4 flex items-center justify-between">
|
|
<h3 className="text-base font-semibold text-slate-900 dark:text-slate-50">
|
|
Zona {zone.zoneNumber}
|
|
</h3>
|
|
|
|
<div className="flex gap-2">
|
|
<StatusPill active={zone.fansOn} activeLabel="Vent." inactiveLabel="Vent." />
|
|
<StatusPill active={zone.extractorsOn} activeLabel="Extr." inactiveLabel="Extr." />
|
|
</div>
|
|
</div>
|
|
|
|
<div className="grid grid-cols-2 gap-3 md:grid-cols-3">
|
|
<MiniValue icon={Thermometer} label="Temperatura" value={formatValue(zone.temperature, "°C")} />
|
|
<MiniValue icon={Waves} label="Humidade" value={formatValue(zone.humidity, "%", 0)} />
|
|
<MiniValue icon={Gauge} label="CO₂" value={formatValue(zone.co2, "ppm", 0)} />
|
|
</div>
|
|
|
|
<div className="mt-5 grid grid-cols-2 gap-3 md:grid-cols-4">
|
|
<OpeningBar label="Zenital E" value={zone.zenitalLeftPercent} />
|
|
<OpeningBar label="Zenital D" value={zone.zenitalRightPercent} />
|
|
<OpeningBar label="Lateral E" value={zone.lateralLeftPercent} />
|
|
<OpeningBar label="Lateral D" value={zone.lateralRightPercent} />
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</section>
|
|
);
|
|
}
|
|
|
|
type MiniValueProps = {
|
|
icon: React.ElementType;
|
|
label: string;
|
|
value: string;
|
|
};
|
|
|
|
function MiniValue({ icon: Icon, label, value }: MiniValueProps) {
|
|
return (
|
|
<div className="rounded-xl bg-slate-50 p-3 dark:bg-slate-950">
|
|
<div className="mb-2 flex items-center gap-2 text-slate-500 dark:text-slate-400">
|
|
<Icon className="h-4 w-4" />
|
|
<span className="text-xs">{label}</span>
|
|
</div>
|
|
<p className="text-lg font-semibold text-slate-900 dark:text-slate-50">
|
|
{value}
|
|
</p>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
type OpeningBarProps = {
|
|
label: string;
|
|
value: number | null;
|
|
};
|
|
|
|
function OpeningBar({ label, value }: OpeningBarProps) {
|
|
const safeValue = value ?? 0;
|
|
|
|
return (
|
|
<div>
|
|
<div className="mb-1 flex justify-between text-xs text-slate-500 dark:text-slate-400">
|
|
<span>{label}</span>
|
|
<span>{value === null ? "--" : `${value.toFixed(0)}%`}</span>
|
|
</div>
|
|
|
|
<div className="h-2 rounded-full bg-slate-200 dark:bg-slate-800">
|
|
<div
|
|
className="h-2 rounded-full bg-blue-500"
|
|
style={{ width: `${safeValue}%` }}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
} |