150 lines
4.9 KiB
Svelte
150 lines
4.9 KiB
Svelte
<script lang="ts">
|
|
import { cn } from "$lib/utils";
|
|
import { Slider } from "$lib/components/ui/slider";
|
|
import { Label } from "$lib/components/ui/label";
|
|
import { Card, CardContent, CardHeader } from "$lib/components/ui/card";
|
|
import type { BluetoothDevice } from "$lib/types";
|
|
import { _ } from "svelte-i18n";
|
|
|
|
interface Props {
|
|
device: BluetoothDevice;
|
|
onChange: (scalarIndex: number, val: number) => void;
|
|
onStop: () => void;
|
|
}
|
|
|
|
let { device, onChange, onStop }: Props = $props();
|
|
|
|
function getBatteryColor(level: number) {
|
|
if (!device.hasBattery) {
|
|
return "text-gray-400";
|
|
}
|
|
if (level > 60) return "text-green-400";
|
|
if (level > 30) return "text-yellow-400";
|
|
return "text-red-400";
|
|
}
|
|
|
|
function getBatteryBgColor(level: number) {
|
|
if (!device.hasBattery) {
|
|
return "bg-gray-400/20";
|
|
}
|
|
if (level > 60) return "bg-green-400/20";
|
|
if (level > 30) return "bg-yellow-400/20";
|
|
return "bg-red-400/20";
|
|
}
|
|
|
|
function getScalarAnimations() {
|
|
return device.actuators
|
|
.filter((a) => a.value > 0)
|
|
.map((a) => `animate-${a.outputType.toLowerCase()}`);
|
|
}
|
|
|
|
function isActive() {
|
|
return device.actuators.some((a) => a.value > 0);
|
|
}
|
|
</script>
|
|
|
|
<Card
|
|
class="group hover:shadow-lg hover:shadow-primary/10 transition-all duration-300 border-border/50 hover:border-primary/30 bg-card/50 backdrop-blur-sm"
|
|
>
|
|
<CardHeader class="pb-3">
|
|
<div class="flex items-start justify-between">
|
|
<div class="flex items-center gap-3">
|
|
<div
|
|
class="p-2 rounded-lg bg-gradient-to-br from-primary/20 to-accent/20 border border-primary/30 flex shrink-0 grow-0"
|
|
>
|
|
<span
|
|
class={cn([...getScalarAnimations(), "icon-[ri--rocket-line] w-5 h-5 text-primary"])}
|
|
></span>
|
|
</div>
|
|
<div>
|
|
<h3 class="font-semibold text-card-foreground group-hover:text-primary transition-colors">
|
|
{device.name}
|
|
</h3>
|
|
<!-- <p class="text-sm text-muted-foreground">
|
|
{device.deviceType}
|
|
</p> -->
|
|
</div>
|
|
</div>
|
|
<button
|
|
class={`${isActive() ? "cursor-pointer" : ""} flex items-center gap-2`}
|
|
onclick={() => isActive() && onStop()}
|
|
>
|
|
<div class="relative">
|
|
<div class="w-2 h-2 rounded-full {isActive() ? 'bg-green-400' : 'bg-red-400'}"></div>
|
|
{#if isActive()}
|
|
<div
|
|
class="absolute inset-0 w-2 h-2 rounded-full bg-green-400 animate-ping opacity-75"
|
|
></div>
|
|
{/if}
|
|
</div>
|
|
<span class="text-xs font-medium {isActive() ? 'text-green-400' : 'text-red-400'}">
|
|
{isActive() ? $_("device_card.active") : $_("device_card.paused")}
|
|
</span>
|
|
</button>
|
|
</div>
|
|
</CardHeader>
|
|
|
|
<CardContent class="space-y-4">
|
|
<!-- Current Value -->
|
|
<!-- <div
|
|
class="flex items-center justify-between p-3 rounded-lg bg-muted/30 border border-border/30"
|
|
>
|
|
<span class="text-sm text-muted-foreground"
|
|
>{$_("device_card.current_value")}</span
|
|
>
|
|
<span class="font-medium text-card-foreground">{device.currentValue}</span
|
|
>
|
|
</div> -->
|
|
|
|
<!-- Battery Level -->
|
|
<div class="space-y-2">
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center gap-2">
|
|
<span
|
|
class="icon-[ri--battery-2-charge-line] w-4 h-4 {getBatteryColor(device.batteryLevel)}"
|
|
></span>
|
|
<span class="text-sm text-muted-foreground">{$_("device_card.battery")}</span>
|
|
</div>
|
|
{#if device.hasBattery}
|
|
<span class="text-sm font-medium {getBatteryColor(device.batteryLevel)}">
|
|
{device.batteryLevel}%
|
|
</span>
|
|
{/if}
|
|
</div>
|
|
<div class="w-full bg-muted/50 rounded-full h-2 overflow-hidden">
|
|
<div
|
|
class="h-full rounded-full transition-all duration-500 {getBatteryBgColor(
|
|
device.batteryLevel,
|
|
)} bg-gradient-to-r from-current to-current/80"
|
|
style="width: {device.batteryLevel}%"
|
|
></div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Last Seen -->
|
|
<!-- <div
|
|
class="flex items-center justify-between text-xs text-muted-foreground"
|
|
>
|
|
<span>{$_("device_card.last_seen")}</span>
|
|
<span>{device.lastSeen.toLocaleTimeString()}</span>
|
|
</div> -->
|
|
|
|
<!-- Action Button -->
|
|
{#each device.actuators as actuator, idx (idx)}
|
|
<div class="space-y-2">
|
|
<Label for={`device-${device.info.index}-${actuator.featureIndex}-${actuator.outputType}`}
|
|
>{$_(`device_card.actuator_types.${actuator.outputType.toLowerCase()}`)}</Label
|
|
>
|
|
<Slider
|
|
id={`device-${device.info.index}-${actuator.featureIndex}-${actuator.outputType}`}
|
|
type="single"
|
|
value={actuator.value}
|
|
onValueChange={(val) => onChange(idx, val)}
|
|
max={actuator.maxSteps}
|
|
step={1}
|
|
/>
|
|
</div>
|
|
{/each}
|
|
</CardContent>
|
|
</Card>
|