| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195 |
- import { useState } from 'react';
- import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
- import { Plug, Power, PowerOff, Loader2, Wifi, WifiOff, Zap, Radio, Eye } from 'lucide-react';
- import { api } from '../api/client';
- import type { SmartPlug } from '../api/client';
- import { ConfirmModal } from './ConfirmModal';
- interface SwitchbarPopoverProps {
- onClose: () => void;
- }
- function SwitchItem({ plug }: { plug: SmartPlug }) {
- const queryClient = useQueryClient();
- const [confirmAction, setConfirmAction] = useState<'on' | 'off' | null>(null);
- // Fetch current status
- const { data: status, isLoading: statusLoading } = useQuery({
- queryKey: ['smart-plug-status', plug.id],
- queryFn: () => api.getSmartPlugStatus(plug.id),
- refetchInterval: 10000, // Refresh every 10 seconds when popover is open
- });
- // Control mutation
- const controlMutation = useMutation({
- mutationFn: (action: 'on' | 'off') => api.controlSmartPlug(plug.id, action),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['smart-plug-status', plug.id] });
- },
- });
- const isOn = status?.state === 'ON';
- // For MQTT plugs, consider reachable if we have power data
- const hasMqttData = plug.plug_type === 'mqtt' && (status?.energy?.power !== null && status?.energy?.power !== undefined);
- const isReachable = (status?.reachable ?? false) || hasMqttData;
- const isPending = controlMutation.isPending;
- const isMqtt = plug.plug_type === 'mqtt';
- const handleConfirm = () => {
- if (confirmAction) {
- controlMutation.mutate(confirmAction);
- setConfirmAction(null);
- }
- };
- return (
- <>
- <div className="flex items-center justify-between py-2 px-3 hover:bg-bambu-dark-tertiary rounded-lg transition-colors">
- <div className="flex items-center gap-2">
- <div className={`p-1.5 rounded ${
- isMqtt
- ? (isReachable ? 'bg-teal-500/20' : 'bg-red-500/20')
- : (isReachable ? (isOn ? 'bg-bambu-green/20' : 'bg-bambu-dark') : 'bg-red-500/20')
- }`}>
- {isMqtt ? (
- <Radio className={`w-4 h-4 ${isReachable ? 'text-teal-400' : 'text-red-400'}`} />
- ) : (
- <Plug className={`w-4 h-4 ${isReachable ? (isOn ? 'text-bambu-green' : 'text-bambu-gray') : 'text-red-400'}`} />
- )}
- </div>
- <div>
- <p className="text-sm text-white font-medium">{plug.name}</p>
- <div className="flex items-center gap-1 text-xs">
- {statusLoading ? (
- <Loader2 className="w-3 h-3 text-bambu-gray animate-spin" />
- ) : isMqtt ? (
- /* MQTT plugs show power and monitor-only indicator */
- isReachable ? (
- <>
- <Zap className="w-3 h-3 text-teal-400" />
- <span className="text-teal-400">{Math.round(status?.energy?.power ?? 0)}W</span>
- <span className="text-bambu-gray mx-1">|</span>
- <Eye className="w-3 h-3 text-bambu-gray" />
- <span className="text-bambu-gray">Monitor</span>
- </>
- ) : (
- <>
- <WifiOff className="w-3 h-3 text-status-error" />
- <span className="text-status-error">Waiting</span>
- </>
- )
- ) : isReachable ? (
- <>
- <Wifi className="w-3 h-3 text-status-ok" />
- <span className={isOn ? 'text-status-ok' : 'text-bambu-gray'}>{status?.state || 'Unknown'}</span>
- {status?.energy?.power !== null && status?.energy?.power !== undefined && (
- <>
- <span className="text-bambu-gray mx-1">|</span>
- <Zap className="w-3 h-3 text-yellow-400" />
- <span className="text-yellow-400">{Math.round(status.energy.power)}W</span>
- </>
- )}
- </>
- ) : (
- <>
- <WifiOff className="w-3 h-3 text-status-error" />
- <span className="text-status-error">Offline</span>
- </>
- )}
- </div>
- </div>
- </div>
- {/* Hide on/off buttons for MQTT plugs (monitor-only) */}
- {!isMqtt && (
- <div className="flex gap-1">
- <button
- onClick={() => setConfirmAction('on')}
- disabled={!isReachable || isPending}
- className={`p-1.5 rounded transition-colors ${
- isOn
- ? 'bg-bambu-green text-white'
- : 'bg-bambu-dark text-bambu-gray hover:text-white'
- } disabled:opacity-50 disabled:cursor-not-allowed`}
- title="Turn On"
- >
- {isPending ? <Loader2 className="w-4 h-4 animate-spin" /> : <Power className="w-4 h-4" />}
- </button>
- <button
- onClick={() => setConfirmAction('off')}
- disabled={!isReachable || isPending}
- className={`p-1.5 rounded transition-colors ${
- !isOn && isReachable
- ? 'bg-bambu-dark-tertiary text-white'
- : 'bg-bambu-dark text-bambu-gray hover:text-white'
- } disabled:opacity-50 disabled:cursor-not-allowed`}
- title="Turn Off"
- >
- {isPending ? <Loader2 className="w-4 h-4 animate-spin" /> : <PowerOff className="w-4 h-4" />}
- </button>
- </div>
- )}
- </div>
- {confirmAction && (
- <ConfirmModal
- title={`Turn ${confirmAction === 'on' ? 'On' : 'Off'} Smart Plug`}
- message={`Are you sure you want to turn ${confirmAction === 'on' ? 'on' : 'off'} "${plug.name}"?`}
- confirmText={confirmAction === 'on' ? 'Turn On' : 'Turn Off'}
- variant={confirmAction === 'off' ? 'warning' : 'default'}
- onConfirm={handleConfirm}
- onCancel={() => setConfirmAction(null)}
- />
- )}
- </>
- );
- }
- export function SwitchbarPopover({ onClose }: SwitchbarPopoverProps) {
- // Fetch all smart plugs
- const { data: plugs, isLoading } = useQuery({
- queryKey: ['smart-plugs'],
- queryFn: api.getSmartPlugs,
- });
- // Filter to only show plugs with show_in_switchbar enabled
- const switchbarPlugs = plugs?.filter(p => p.show_in_switchbar) || [];
- return (
- <div
- className="absolute bottom-full left-0 mb-2 w-72 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-xl shadow-xl z-50"
- onMouseLeave={onClose}
- >
- {/* Header */}
- <div className="px-4 py-3 border-b border-bambu-dark-tertiary">
- <h3 className="text-sm font-semibold text-white flex items-center gap-2">
- <Plug className="w-4 h-4 text-bambu-green" />
- Smart Switches
- </h3>
- </div>
- {/* Content */}
- <div className="p-2 max-h-80 overflow-y-auto">
- {isLoading ? (
- <div className="flex items-center justify-center py-8">
- <Loader2 className="w-6 h-6 text-bambu-gray animate-spin" />
- </div>
- ) : switchbarPlugs.length === 0 ? (
- <div className="text-center py-6 px-4">
- <Plug className="w-8 h-8 text-bambu-gray mx-auto mb-2" />
- <p className="text-sm text-bambu-gray">No switches in switchbar</p>
- <p className="text-xs text-bambu-gray mt-1">
- Enable "Show in Switchbar" in Settings > Smart Plugs
- </p>
- </div>
- ) : (
- <div className="space-y-1">
- {switchbarPlugs.map(plug => (
- <SwitchItem key={plug.id} plug={plug} />
- ))}
- </div>
- )}
- </div>
- </div>
- );
- }
|