feat: Improve Pulse analytics and Environment Report theming for light/dark mode
Some checks are pending
Test / backend-test (push) Waiting to run
Test / frontend-test (push) Waiting to run

This commit is contained in:
fullsizemalt 2026-01-06 12:07:41 -08:00
parent 64d7d56792
commit 71e6be0243
3 changed files with 117 additions and 118 deletions

View file

@ -78,9 +78,9 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
onClick={handleClick}
className={cn(
"group relative overflow-hidden rounded-2xl cursor-pointer transition-all",
"bg-gradient-to-br from-slate-800/80 to-slate-900/80",
"border border-slate-700/50 hover:border-emerald-500/50",
"shadow-xl hover:shadow-2xl hover:shadow-emerald-500/10"
"bg-white dark:bg-gradient-to-br dark:from-slate-800/80 dark:to-slate-900/80",
"border border-slate-200 dark:border-slate-700/50 hover:border-emerald-500/50",
"shadow-sm dark:shadow-xl hover:shadow-md dark:hover:shadow-2xl hover:shadow-emerald-500/10"
)}
>
{/* Header */}
@ -90,28 +90,28 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
<div className={cn(
"p-3 rounded-xl",
isOffline
? "bg-slate-700 text-slate-400"
? "bg-slate-100 dark:bg-slate-700 text-slate-500 dark:text-slate-400"
: isTempWarning || isVpdWarning
? "bg-amber-500/20 text-amber-400"
: "bg-emerald-500/20 text-emerald-400"
? "bg-amber-100 dark:bg-amber-500/20 text-amber-600 dark:text-amber-400"
: "bg-emerald-100 dark:bg-emerald-500/20 text-emerald-600 dark:text-emerald-400"
)}>
<Activity size={22} />
</div>
<div>
<h3 className="text-sm font-bold text-white tracking-wide">
<h3 className="text-sm font-bold text-slate-900 dark:text-white tracking-wide">
{reading.deviceName}
</h3>
<p className="text-xs text-slate-500">Pulse Grow Sensor</p>
<p className="text-xs text-slate-500 dark:text-slate-500">Pulse Grow Sensor</p>
</div>
</div>
<div className={cn(
"flex items-center gap-1.5 text-xs font-bold px-2.5 py-1 rounded-full border",
isOffline
? "text-red-400 bg-red-500/10 border-red-500/30"
: "text-emerald-400 bg-emerald-500/10 border-emerald-500/30"
? "text-red-600 dark:text-red-400 bg-red-50 dark:bg-red-500/10 border-red-200 dark:border-red-500/30"
: "text-emerald-600 dark:text-emerald-400 bg-emerald-50 dark:bg-emerald-500/10 border-emerald-200 dark:border-emerald-500/30"
)}>
<div className={cn("w-1.5 h-1.5 rounded-full", isOffline ? "bg-red-400" : "bg-emerald-400 animate-pulse")} />
<div className={cn("w-1.5 h-1.5 rounded-full", isOffline ? "bg-red-500 dark:bg-red-400" : "bg-emerald-500 dark:bg-emerald-400 animate-pulse")} />
{isOffline ? 'OFFLINE' : 'LIVE'}
</div>
</div>
@ -146,10 +146,10 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
{/* Temperature */}
<div className={cn(
"p-3 rounded-xl transition-colors",
isTempWarning ? "bg-amber-500/10" : "bg-slate-800/50"
isTempWarning ? "bg-amber-50 dark:bg-amber-500/10" : "bg-slate-50 dark:bg-slate-800/50"
)}>
<div className="flex items-center justify-between mb-1">
<div className="flex items-center gap-1.5 text-slate-400">
<div className="flex items-center gap-1.5 text-slate-500 dark:text-slate-400">
<Thermometer size={14} />
<span className="text-[10px] font-bold uppercase tracking-wider">Temp</span>
</div>
@ -157,21 +157,21 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
</div>
<p className={cn(
"text-2xl font-bold",
isOffline ? "text-slate-500" : isTempWarning ? "text-amber-400" : "text-white"
isOffline ? "text-slate-400 dark:text-slate-500" : isTempWarning ? "text-amber-600 dark:text-amber-400" : "text-slate-900 dark:text-white"
)}>
{reading.temperature.toFixed(1)}°
</p>
</div>
{/* Humidity */}
<div className="p-3 rounded-xl bg-slate-800/50">
<div className="flex items-center gap-1.5 mb-1 text-slate-400">
<div className="p-3 rounded-xl bg-slate-50 dark:bg-slate-800/50">
<div className="flex items-center gap-1.5 mb-1 text-slate-500 dark:text-slate-400">
<Droplets size={14} />
<span className="text-[10px] font-bold uppercase tracking-wider">RH</span>
</div>
<p className={cn(
"text-2xl font-bold",
isOffline ? "text-slate-500" : "text-blue-400"
isOffline ? "text-slate-400 dark:text-slate-500" : "text-blue-600 dark:text-blue-400"
)}>
{reading.humidity.toFixed(0)}%
</p>
@ -180,15 +180,15 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
{/* VPD */}
<div className={cn(
"p-3 rounded-xl transition-colors",
isVpdWarning ? "bg-purple-500/10" : "bg-slate-800/50"
isVpdWarning ? "bg-purple-50 dark:bg-purple-500/10" : "bg-slate-50 dark:bg-slate-800/50"
)}>
<div className="flex items-center gap-1.5 mb-1 text-slate-400">
<div className="flex items-center gap-1.5 mb-1 text-slate-500 dark:text-slate-400">
<CloudFog size={14} />
<span className="text-[10px] font-bold uppercase tracking-wider">VPD</span>
</div>
<p className={cn(
"text-2xl font-bold",
isOffline ? "text-slate-500" : isVpdWarning ? "text-purple-400" : "text-emerald-400"
isOffline ? "text-slate-400 dark:text-slate-500" : isVpdWarning ? "text-purple-600 dark:text-purple-400" : "text-emerald-600 dark:text-emerald-400"
)}>
{reading.vpd.toFixed(2)}
</p>
@ -197,11 +197,11 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
{/* Dewpoint Row */}
<div className="mt-3 flex items-center justify-between px-1">
<div className="flex items-center gap-2 text-slate-500">
<div className="flex items-center gap-2 text-slate-500 dark:text-slate-500">
<Wind size={12} />
<span className="text-xs">Dewpoint: <span className="text-slate-300">{reading.dewpoint.toFixed(1)}°F</span></span>
<span className="text-xs">Dewpoint: <span className="text-slate-700 dark:text-slate-300">{reading.dewpoint.toFixed(1)}°F</span></span>
</div>
<div className="flex items-center gap-1 text-slate-500 group-hover:text-emerald-400 transition-colors">
<div className="flex items-center gap-1 text-slate-400 dark:text-slate-500 group-hover:text-emerald-500 group-hover:dark:text-emerald-400 transition-colors">
<span className="text-xs font-medium">View Details</span>
<ChevronRight size={14} className="group-hover:translate-x-0.5 transition-transform" />
</div>
@ -210,9 +210,9 @@ export function PulseSensorCard({ reading, history, thresholds, onClick }: Pulse
{/* Timestamp Footer */}
<div className="px-5 pb-4">
<div className="flex items-center justify-between text-xs text-slate-500 pt-3 border-t border-slate-700/50">
<div className="flex items-center justify-between text-xs text-slate-400 dark:text-slate-500 pt-3 border-t border-slate-100 dark:border-slate-700/50">
<span>Last reading</span>
<span className={isOffline ? "text-red-400" : ""}>
<span className={isOffline ? "text-red-500 dark:text-red-400" : ""}>
{readingTime.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' })}
</span>
</div>

View file

@ -378,20 +378,21 @@ export default function EnvironmentReportPage() {
</div>
{/* Summary Stats */}
<div className="grid grid-cols-4 gap-4">
{/* Summary Stats */}
<div className="grid grid-cols-2 md:grid-cols-4 gap-4">
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
className="p-5 rounded-2xl bg-white dark:bg-gradient-to-br dark:from-red-500/20 dark:to-orange-500/10 border border-red-200 dark:border-red-500/20 shadow-sm dark:shadow-none"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2 text-slate-500 dark:text-slate-400">
<Thermometer className="w-5 h-5 text-red-500 dark:text-red-400" />
<span className="text-sm">Avg Temperature</span>
<span className="text-sm font-medium">Avg Temperature</span>
</div>
<p className="text-3xl font-bold text-slate-900 dark:text-white">
<p className="text-2xl md:text-3xl font-bold text-slate-900 dark:text-white">
{reportData?.devices[0]?.stats.temperature.avg.toFixed(1) || '--'}°F
</p>
<p className="text-xs text-slate-500 mt-1">
<p className="text-xs text-slate-500 dark:text-slate-400 mt-1">
Range: {reportData?.devices[0]?.stats.temperature.min.toFixed(1)}° {reportData?.devices[0]?.stats.temperature.max.toFixed(1)}°
</p>
</motion.div>
@ -400,16 +401,16 @@ export default function EnvironmentReportPage() {
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.1 }}
className="p-5 rounded-2xl bg-white dark:bg-gradient-to-br dark:from-blue-500/20 dark:to-cyan-500/10 border border-blue-200 dark:border-blue-500/20 shadow-sm dark:shadow-none"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2 text-slate-500 dark:text-slate-400">
<Droplets className="w-5 h-5 text-blue-500 dark:text-blue-400" />
<span className="text-sm">Avg Humidity</span>
<span className="text-sm font-medium">Avg Humidity</span>
</div>
<p className="text-3xl font-bold text-slate-900 dark:text-white">
<p className="text-2xl md:text-3xl font-bold text-slate-900 dark:text-white">
{reportData?.devices[0]?.stats.humidity.avg.toFixed(0) || '--'}%
</p>
<p className="text-xs text-slate-500 mt-1">
<p className="text-xs text-slate-500 dark:text-slate-400 mt-1">
Range: {reportData?.devices[0]?.stats.humidity.min.toFixed(0)}% {reportData?.devices[0]?.stats.humidity.max.toFixed(0)}%
</p>
</motion.div>
@ -418,16 +419,16 @@ export default function EnvironmentReportPage() {
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.2 }}
className="p-5 rounded-2xl bg-white dark:bg-gradient-to-br dark:from-purple-500/20 dark:to-pink-500/10 border border-purple-200 dark:border-purple-500/20 shadow-sm dark:shadow-none"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2 text-slate-500 dark:text-slate-400">
<Wind className="w-5 h-5 text-purple-500 dark:text-purple-400" />
<span className="text-sm">Avg VPD</span>
<span className="text-sm font-medium">Avg VPD</span>
</div>
<p className="text-3xl font-bold text-slate-900 dark:text-white">
<p className="text-2xl md:text-3xl font-bold text-slate-900 dark:text-white">
{reportData?.devices[0]?.stats.vpd.avg.toFixed(2) || '--'} kPa
</p>
<p className="text-xs text-slate-500 mt-1">
<p className="text-xs text-slate-500 dark:text-slate-400 mt-1">
Range: {reportData?.devices[0]?.stats.vpd.min.toFixed(2)} {reportData?.devices[0]?.stats.vpd.max.toFixed(2)}
</p>
</motion.div>
@ -436,16 +437,16 @@ export default function EnvironmentReportPage() {
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.3 }}
className="p-5 rounded-2xl bg-white dark:bg-gradient-to-br dark:from-amber-500/20 dark:to-yellow-500/10 border border-amber-200 dark:border-amber-500/20 shadow-sm dark:shadow-none"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2 text-slate-500 dark:text-slate-400">
<AlertTriangle className="w-5 h-5 text-amber-500 dark:text-amber-400" />
<span className="text-sm">Alerts</span>
<span className="text-sm font-medium">Alerts</span>
</div>
<p className="text-3xl font-bold text-slate-900 dark:text-white">
<p className="text-2xl md:text-3xl font-bold text-slate-900 dark:text-white">
{reportData?.alerts.total || 0}
</p>
<p className="text-xs text-slate-500 mt-1">
<p className="text-xs text-slate-500 dark:text-slate-400 mt-1">
<CheckCircle className="w-3 h-3 inline mr-1 text-green-500 dark:text-green-400" />
{reportData?.alerts.resolved || 0} resolved
</p>

View file

@ -129,30 +129,30 @@ export default function PulseTestPage() {
const currentReading = readings.find(r => r.deviceId === selectedDevice) || readings[0];
return (
<div className="min-h-screen bg-gradient-to-br from-slate-900 via-slate-800 to-slate-900 p-6">
<div className="min-h-screen bg-slate-50 dark:bg-slate-950 p-6 transition-colors duration-300">
<div className="max-w-7xl mx-auto space-y-6">
{/* Header */}
<div className="flex items-center justify-between">
<div className="flex flex-col md:flex-row md:items-center justify-between gap-4">
<div>
<h1 className="text-3xl font-bold text-white flex items-center gap-3">
<div className="p-2.5 rounded-xl bg-gradient-to-br from-emerald-500 to-teal-600 shadow-lg shadow-emerald-500/25">
<Activity className="w-6 h-6 text-white" />
<h1 className="text-3xl font-bold text-slate-900 dark:text-white flex items-center gap-3">
<div className="p-2.5 rounded-xl bg-gradient-to-br from-emerald-500 to-teal-600 shadow-lg shadow-emerald-500/25 text-white">
<Activity className="w-6 h-6" />
</div>
Pulse Sensor Analytics
</h1>
<p className="text-slate-400 ml-14 mt-1">
<p className="text-slate-500 dark:text-slate-400 ml-14 mt-1">
Real-time environmental monitoring with historical trends
</p>
</div>
<div className="flex items-center gap-4">
{/* WebSocket Status */}
<div className="flex items-center gap-2 px-3 py-1.5 rounded-full bg-slate-800/50 border border-slate-700">
<div className="flex items-center gap-2 px-3 py-1.5 rounded-full bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700 shadow-sm">
{wsConnected ? (
<Wifi className="w-4 h-4 text-green-500" />
<Wifi className="w-4 h-4 text-emerald-500" />
) : (
<WifiOff className="w-4 h-4 text-red-500" />
)}
<span className="text-xs font-medium text-slate-300">
<span className="text-xs font-medium text-slate-600 dark:text-slate-300">
{wsConnected ? 'Live' : 'Offline'}
</span>
{unreadCount > 0 && (
@ -167,16 +167,16 @@ export default function PulseTestPage() {
<button
onClick={fetchData}
disabled={loading}
className="p-2.5 rounded-xl bg-slate-800/50 border border-slate-700 hover:bg-slate-700 transition-colors disabled:opacity-50"
className="p-2.5 rounded-xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700 hover:bg-slate-50 dark:hover:bg-slate-700 transition-colors disabled:opacity-50 shadow-sm text-slate-500 dark:text-slate-400"
>
<RefreshCw className={`w-5 h-5 text-slate-300 ${loading ? 'animate-spin' : ''}`} />
<RefreshCw className={`w-5 h-5 ${loading ? 'animate-spin' : ''}`} />
</button>
</div>
</div>
{/* Error Message */}
{error && (
<div className="p-4 rounded-xl bg-red-500/10 border border-red-500/30 text-red-400">
<div className="p-4 rounded-xl bg-red-50 dark:bg-red-500/10 border border-red-200 dark:border-red-500/30 text-red-600 dark:text-red-400">
{error}
</div>
)}
@ -185,41 +185,41 @@ export default function PulseTestPage() {
<div className="grid grid-cols-1 lg:grid-cols-4 gap-6">
{/* Live Readings Panel */}
<div className="lg:col-span-1 space-y-4">
<h2 className="text-sm font-semibold text-slate-400 uppercase tracking-wider">Live Sensors</h2>
<h2 className="text-sm font-semibold text-slate-500 dark:text-slate-400 uppercase tracking-wider">Live Sensors</h2>
{readings.map((reading) => (
<motion.div
key={reading.deviceId}
whileHover={{ scale: 1.02 }}
onClick={() => setSelectedDevice(reading.deviceId)}
className={`p-4 rounded-xl cursor-pointer transition-all ${selectedDevice === reading.deviceId
? 'bg-emerald-500/20 border-2 border-emerald-500/50'
: 'bg-slate-800/50 border border-slate-700/50 hover:border-slate-600'
className={`p-4 rounded-xl cursor-pointer transition-all shadow-sm ${selectedDevice === reading.deviceId
? 'bg-emerald-50 dark:bg-emerald-500/20 border-2 border-emerald-500/50'
: 'bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 hover:border-slate-300 dark:hover:border-slate-600'
}`}
>
<div className="flex items-center justify-between mb-3">
<h3 className="font-semibold text-white text-sm">{reading.deviceName}</h3>
<h3 className="font-semibold text-slate-900 dark:text-white text-sm">{reading.deviceName}</h3>
<div className="flex items-center gap-1.5">
<div className="w-2 h-2 rounded-full bg-green-500 animate-pulse" />
<span className="text-xs text-green-400 font-medium">LIVE</span>
<div className="w-2 h-2 rounded-full bg-emerald-500 animate-pulse" />
<span className="text-xs text-emerald-600 dark:text-emerald-400 font-medium">LIVE</span>
</div>
</div>
<div className="grid grid-cols-3 gap-2">
<div>
<p className="text-xs text-slate-500 mb-0.5">Temp</p>
<p className="text-xs text-slate-500 dark:text-slate-400 mb-0.5">Temp</p>
<p className={`text-lg font-bold ${getTempColor(reading.temperature)}`}>
{reading.temperature.toFixed(1)}°
</p>
</div>
<div>
<p className="text-xs text-slate-500 mb-0.5">RH</p>
<p className="text-lg font-bold text-blue-400">
<p className="text-xs text-slate-500 dark:text-slate-400 mb-0.5">RH</p>
<p className="text-lg font-bold text-blue-500 dark:text-blue-400">
{reading.humidity.toFixed(0)}%
</p>
</div>
<div>
<p className="text-xs text-slate-500 mb-0.5">VPD</p>
<p className="text-xs text-slate-500 dark:text-slate-400 mb-0.5">VPD</p>
<p className={`text-lg font-bold ${getVpdColor(reading.vpd)}`}>
{reading.vpd.toFixed(2)}
</p>
@ -229,9 +229,9 @@ export default function PulseTestPage() {
))}
{readings.length === 0 && !loading && (
<div className="p-6 text-center rounded-xl bg-slate-800/50 border border-slate-700/50">
<Activity className="w-8 h-8 mx-auto mb-2 text-slate-500" />
<p className="text-slate-400 text-sm">No sensors connected</p>
<div className="p-6 text-center rounded-xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm">
<Activity className="w-8 h-8 mx-auto mb-2 text-slate-400 dark:text-slate-500" />
<p className="text-slate-500 dark:text-slate-400 text-sm">No sensors connected</p>
</div>
)}
</div>
@ -240,17 +240,17 @@ export default function PulseTestPage() {
<div className="lg:col-span-3 space-y-6">
{/* Current Stats */}
{currentReading && (
<div className="grid grid-cols-4 gap-4">
<div className="grid grid-cols-2 sm:grid-cols-4 gap-4">
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
className="p-5 rounded-2xl bg-gradient-to-br from-red-500/20 to-orange-500/10 border border-red-500/20"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/30 border border-red-100 dark:border-red-500/10 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2">
<Thermometer className="w-5 h-5 text-red-400" />
<span className="text-sm text-slate-400">Temperature</span>
<Thermometer className="w-5 h-5 text-red-500 dark:text-red-400" />
<span className="text-sm text-slate-500 dark:text-slate-400">Temperature</span>
</div>
<p className={`text-4xl font-bold ${getTempColor(currentReading.temperature)}`}>
<p className={`text-3xl lg:text-4xl font-bold ${getTempColor(currentReading.temperature)}`}>
{currentReading.temperature.toFixed(1)}°F
</p>
</motion.div>
@ -259,13 +259,13 @@ export default function PulseTestPage() {
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.1 }}
className="p-5 rounded-2xl bg-gradient-to-br from-blue-500/20 to-cyan-500/10 border border-blue-500/20"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/30 border border-blue-100 dark:border-blue-500/10 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2">
<Droplets className="w-5 h-5 text-blue-400" />
<span className="text-sm text-slate-400">Humidity</span>
<Droplets className="w-5 h-5 text-blue-500 dark:text-blue-400" />
<span className="text-sm text-slate-500 dark:text-slate-400">Humidity</span>
</div>
<p className="text-4xl font-bold text-blue-400">
<p className="text-3xl lg:text-4xl font-bold text-blue-500 dark:text-blue-400">
{currentReading.humidity.toFixed(0)}%
</p>
</motion.div>
@ -274,29 +274,29 @@ export default function PulseTestPage() {
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.2 }}
className="p-5 rounded-2xl bg-gradient-to-br from-purple-500/20 to-pink-500/10 border border-purple-500/20"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/30 border border-purple-100 dark:border-purple-500/10 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2">
<Wind className="w-5 h-5 text-purple-400" />
<span className="text-sm text-slate-400">VPD</span>
<Wind className="w-5 h-5 text-purple-500 dark:text-purple-400" />
<span className="text-sm text-slate-500 dark:text-slate-400">VPD</span>
</div>
<p className={`text-4xl font-bold ${getVpdColor(currentReading.vpd)}`}>
<p className={`text-3xl lg:text-4xl font-bold ${getVpdColor(currentReading.vpd)}`}>
{currentReading.vpd.toFixed(2)}
</p>
<p className="text-xs text-slate-500 mt-1">kPa</p>
<p className="text-xs text-slate-500 dark:text-slate-500 mt-1">kPa</p>
</motion.div>
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.3 }}
className="p-5 rounded-2xl bg-gradient-to-br from-cyan-500/20 to-teal-500/10 border border-cyan-500/20"
className="p-5 rounded-2xl bg-white dark:bg-slate-800/30 border border-cyan-100 dark:border-cyan-500/10 shadow-sm dark:shadow-none"
>
<div className="flex items-center gap-2 mb-2">
<Droplets className="w-5 h-5 text-cyan-400" />
<span className="text-sm text-slate-400">Dewpoint</span>
<Droplets className="w-5 h-5 text-cyan-500 dark:text-cyan-400" />
<span className="text-sm text-slate-500 dark:text-slate-400">Dewpoint</span>
</div>
<p className="text-4xl font-bold text-cyan-400">
<p className="text-3xl lg:text-4xl font-bold text-cyan-500 dark:text-cyan-400">
{currentReading.dewpoint.toFixed(1)}°F
</p>
</motion.div>
@ -304,9 +304,9 @@ export default function PulseTestPage() {
)}
{/* Time Range Selector */}
<div className="flex items-center justify-between">
<h2 className="text-lg font-semibold text-white flex items-center gap-2">
<TrendingUp className="w-5 h-5 text-emerald-400" />
<div className="flex items-center justify-between pt-4">
<h2 className="text-lg font-semibold text-slate-900 dark:text-white flex items-center gap-2">
<TrendingUp className="w-5 h-5 text-emerald-500 dark:text-emerald-400" />
Historical Trends
</h2>
<div className="flex items-center gap-2">
@ -314,7 +314,7 @@ export default function PulseTestPage() {
<select
value={historyHours}
onChange={(e) => setHistoryHours(Number(e.target.value))}
className="bg-slate-800 border border-slate-700 rounded-lg px-3 py-1.5 text-sm text-white focus:outline-none focus:ring-2 focus:ring-emerald-500"
className="bg-white dark:bg-slate-800 border border-slate-200 dark:border-slate-700 rounded-lg px-3 py-1.5 text-sm text-slate-900 dark:text-white focus:outline-none focus:ring-2 focus:ring-emerald-500 shadow-sm"
>
<option value={1}>Last hour</option>
<option value={6}>Last 6 hours</option>
@ -326,9 +326,9 @@ export default function PulseTestPage() {
</div>
{/* Temperature Chart */}
<div className="p-6 rounded-2xl bg-slate-800/50 border border-slate-700/50">
<h3 className="text-sm font-medium text-slate-400 mb-4 flex items-center gap-2">
<Thermometer className="w-4 h-4 text-red-400" />
<div className="p-6 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm">
<h3 className="text-sm font-medium text-slate-500 dark:text-slate-400 mb-4 flex items-center gap-2">
<Thermometer className="w-4 h-4 text-red-500 dark:text-red-400" />
Temperature History
</h3>
<div className="h-64">
@ -340,12 +340,12 @@ export default function PulseTestPage() {
<stop offset="95%" stopColor="#ef4444" stopOpacity={0} />
</linearGradient>
</defs>
<CartesianGrid strokeDasharray="3 3" stroke="#334155" />
<XAxis dataKey="timestamp" stroke="#64748b" fontSize={11} />
<YAxis stroke="#64748b" fontSize={11} domain={['auto', 'auto']} />
<CartesianGrid strokeDasharray="3 3" stroke="#94a3b8" strokeOpacity={0.2} />
<XAxis dataKey="timestamp" stroke="#94a3b8" fontSize={11} tickMargin={10} />
<YAxis stroke="#94a3b8" fontSize={11} domain={['auto', 'auto']} />
<Tooltip
contentStyle={{ backgroundColor: '#1e293b', border: '1px solid #334155', borderRadius: '8px' }}
labelStyle={{ color: '#94a3b8' }}
contentStyle={{ backgroundColor: 'var(--color-bg-secondary)', borderColor: 'var(--color-border-subtle)', borderRadius: '8px', color: 'var(--color-text-primary)' }}
itemStyle={{ color: 'var(--color-text-primary)' }}
/>
<ReferenceLine y={82} stroke="#f59e0b" strokeDasharray="5 5" />
<ReferenceLine y={65} stroke="#3b82f6" strokeDasharray="5 5" />
@ -356,10 +356,10 @@ export default function PulseTestPage() {
</div>
{/* Humidity & VPD Charts */}
<div className="grid grid-cols-2 gap-6">
<div className="p-6 rounded-2xl bg-slate-800/50 border border-slate-700/50">
<h3 className="text-sm font-medium text-slate-400 mb-4 flex items-center gap-2">
<Droplets className="w-4 h-4 text-blue-400" />
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="p-6 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm">
<h3 className="text-sm font-medium text-slate-500 dark:text-slate-400 mb-4 flex items-center gap-2">
<Droplets className="w-4 h-4 text-blue-500 dark:text-blue-400" />
Humidity History
</h3>
<div className="h-48">
@ -371,11 +371,11 @@ export default function PulseTestPage() {
<stop offset="95%" stopColor="#3b82f6" stopOpacity={0} />
</linearGradient>
</defs>
<CartesianGrid strokeDasharray="3 3" stroke="#334155" />
<XAxis dataKey="timestamp" stroke="#64748b" fontSize={10} />
<YAxis stroke="#64748b" fontSize={10} domain={[0, 100]} />
<CartesianGrid strokeDasharray="3 3" stroke="#94a3b8" strokeOpacity={0.2} />
<XAxis dataKey="timestamp" stroke="#94a3b8" fontSize={10} tickMargin={10} />
<YAxis stroke="#94a3b8" fontSize={10} domain={[0, 100]} />
<Tooltip
contentStyle={{ backgroundColor: '#1e293b', border: '1px solid #334155', borderRadius: '8px' }}
contentStyle={{ backgroundColor: 'var(--color-bg-secondary)', borderColor: 'var(--color-border-subtle)', borderRadius: '8px' }}
/>
<Area type="monotone" dataKey="humidity" stroke="#3b82f6" fill="url(#humidityGradient)" strokeWidth={2} />
</AreaChart>
@ -383,9 +383,9 @@ export default function PulseTestPage() {
</div>
</div>
<div className="p-6 rounded-2xl bg-slate-800/50 border border-slate-700/50">
<h3 className="text-sm font-medium text-slate-400 mb-4 flex items-center gap-2">
<Wind className="w-4 h-4 text-purple-400" />
<div className="p-6 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm">
<h3 className="text-sm font-medium text-slate-500 dark:text-slate-400 mb-4 flex items-center gap-2">
<Wind className="w-4 h-4 text-purple-500 dark:text-purple-400" />
VPD History
</h3>
<div className="h-48">
@ -397,14 +397,12 @@ export default function PulseTestPage() {
<stop offset="95%" stopColor="#a855f7" stopOpacity={0} />
</linearGradient>
</defs>
<CartesianGrid strokeDasharray="3 3" stroke="#334155" />
<XAxis dataKey="timestamp" stroke="#64748b" fontSize={10} />
<YAxis stroke="#64748b" fontSize={10} domain={[0, 2]} />
<CartesianGrid strokeDasharray="3 3" stroke="#94a3b8" strokeOpacity={0.2} />
<XAxis dataKey="timestamp" stroke="#94a3b8" fontSize={10} tickMargin={10} />
<YAxis stroke="#94a3b8" fontSize={10} domain={[0, 4]} />
<Tooltip
contentStyle={{ backgroundColor: '#1e293b', border: '1px solid #334155', borderRadius: '8px' }}
contentStyle={{ backgroundColor: 'var(--color-bg-secondary)', borderColor: 'var(--color-border-subtle)', borderRadius: '8px' }}
/>
<ReferenceLine y={1.2} stroke="#f59e0b" strokeDasharray="3 3" />
<ReferenceLine y={0.8} stroke="#3b82f6" strokeDasharray="3 3" />
<Area type="monotone" dataKey="vpd" stroke="#a855f7" fill="url(#vpdGradient)" strokeWidth={2} />
</AreaChart>
</ResponsiveContainer>
@ -416,9 +414,9 @@ export default function PulseTestPage() {
{/* Recent Alerts */}
{alerts.length > 0 && (
<div className="p-6 rounded-2xl bg-slate-800/50 border border-slate-700/50">
<h3 className="font-semibold text-white mb-4 flex items-center gap-2">
<Bell className="w-5 h-5 text-amber-400" />
<div className="p-6 rounded-2xl bg-white dark:bg-slate-800/50 border border-slate-200 dark:border-slate-700/50 shadow-sm">
<h3 className="font-semibold text-slate-900 dark:text-white mb-4 flex items-center gap-2">
<Bell className="w-5 h-5 text-amber-500 dark:text-amber-400" />
Recent Alerts ({alerts.length})
</h3>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-3">