| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- import { useState, useEffect } from 'react';
- import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
- import { Loader2, Check, AlertTriangle, Printer, Eye, EyeOff, Info, ChevronDown, ExternalLink } from 'lucide-react';
- import { virtualPrinterApi } from '../api/client';
- import { Card, CardContent, CardHeader } from './Card';
- import { Button } from './Button';
- import { useToast } from '../contexts/ToastContext';
- export function VirtualPrinterSettings() {
- const queryClient = useQueryClient();
- const { showToast } = useToast();
- const [localEnabled, setLocalEnabled] = useState(false);
- const [localAccessCode, setLocalAccessCode] = useState('');
- const [localMode, setLocalMode] = useState<'immediate' | 'queue'>('immediate');
- const [localModel, setLocalModel] = useState('3DPrinter-X1-Carbon');
- const [showAccessCode, setShowAccessCode] = useState(false);
- const [pendingAction, setPendingAction] = useState<'toggle' | 'accessCode' | 'mode' | 'model' | null>(null);
- // Fetch current settings
- const { data: settings, isLoading } = useQuery({
- queryKey: ['virtual-printer-settings'],
- queryFn: virtualPrinterApi.getSettings,
- refetchInterval: 10000, // Refresh every 10 seconds for status updates
- });
- // Fetch available models
- const { data: modelsData } = useQuery({
- queryKey: ['virtual-printer-models'],
- queryFn: virtualPrinterApi.getModels,
- });
- // Initialize local state from settings
- useEffect(() => {
- if (settings) {
- setLocalEnabled(settings.enabled);
- setLocalMode(settings.mode);
- setLocalModel(settings.model);
- }
- }, [settings]);
- // Update mutation
- const updateMutation = useMutation({
- mutationFn: (data: { enabled?: boolean; access_code?: string; mode?: 'immediate' | 'queue'; model?: string }) =>
- virtualPrinterApi.updateSettings(data),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['virtual-printer-settings'] });
- showToast('Virtual printer settings updated');
- setPendingAction(null);
- },
- onError: (error: Error) => {
- showToast(error.message || 'Failed to update settings', 'error');
- // Revert local state on error
- if (settings) {
- setLocalEnabled(settings.enabled);
- setLocalMode(settings.mode);
- setLocalModel(settings.model);
- }
- setPendingAction(null);
- },
- });
- const handleToggleEnabled = () => {
- const newEnabled = !localEnabled;
- // If enabling, must have access code
- if (newEnabled && !localAccessCode && !settings?.access_code_set) {
- showToast('Please set an access code first', 'error');
- return;
- }
- setLocalEnabled(newEnabled);
- setPendingAction('toggle');
- updateMutation.mutate({
- enabled: newEnabled,
- access_code: localAccessCode || undefined,
- mode: localMode,
- });
- };
- const handleAccessCodeChange = () => {
- if (!localAccessCode) {
- showToast('Access code cannot be empty', 'error');
- return;
- }
- if (localAccessCode.length !== 8) {
- showToast('Access code must be exactly 8 characters', 'error');
- return;
- }
- setPendingAction('accessCode');
- updateMutation.mutate({
- access_code: localAccessCode,
- });
- setLocalAccessCode(''); // Clear after saving
- };
- const handleModeChange = (mode: 'immediate' | 'queue') => {
- setLocalMode(mode);
- setPendingAction('mode');
- updateMutation.mutate({ mode });
- };
- const handleModelChange = (model: string) => {
- setLocalModel(model);
- setPendingAction('model');
- updateMutation.mutate({ model });
- };
- if (isLoading) {
- return (
- <Card>
- <CardContent className="py-8 flex justify-center">
- <Loader2 className="w-6 h-6 animate-spin text-bambu-green" />
- </CardContent>
- </Card>
- );
- }
- const status = settings?.status;
- const isRunning = status?.running || false;
- return (
- <div className="flex flex-col lg:flex-row gap-6 lg:gap-8">
- {/* Left Column - Settings */}
- <div className="space-y-6 lg:w-[480px] lg:flex-shrink-0">
- <Card>
- <CardHeader>
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <Printer className="w-5 h-5 text-bambu-green" />
- <h2 className="text-lg font-semibold text-white">Virtual Printer</h2>
- </div>
- {status && (
- <div className={`flex items-center gap-2 text-sm ${isRunning ? 'text-green-400' : 'text-bambu-gray'}`}>
- <span className={`w-2 h-2 rounded-full ${isRunning ? 'bg-green-400 animate-pulse' : 'bg-gray-500'}`} />
- {isRunning ? 'Running' : 'Stopped'}
- </div>
- )}
- </div>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- Enable a virtual printer that appears in Bambu Studio and OrcaSlicer. Files sent to this printer
- will be archived directly without printing.
- </p>
- {/* Enable/Disable Toggle */}
- <div className="flex items-center justify-between py-3 border-t border-bambu-dark-tertiary">
- <div>
- <div className="text-white font-medium">Enable Virtual Printer</div>
- <div className="text-sm text-bambu-gray">
- {isRunning ? 'Visible as "Bambuddy" in slicer discovery' : 'Not visible to slicers'}
- </div>
- </div>
- <button
- onClick={handleToggleEnabled}
- disabled={pendingAction === 'toggle'}
- className={`relative w-12 h-6 rounded-full transition-colors ${
- localEnabled ? 'bg-bambu-green' : 'bg-bambu-dark-tertiary'
- } ${pendingAction === 'toggle' ? 'opacity-50' : ''}`}
- >
- <span
- className={`absolute top-1 left-1 w-4 h-4 bg-white rounded-full transition-transform ${
- localEnabled ? 'translate-x-6' : ''
- }`}
- />
- </button>
- </div>
- {/* Printer Model */}
- <div className="py-3 border-t border-bambu-dark-tertiary">
- <div className="text-white font-medium mb-2">Printer Model</div>
- <div className="text-sm text-bambu-gray mb-3">
- Select which printer model to emulate.
- </div>
- <div className="relative">
- <select
- value={localModel}
- onChange={(e) => handleModelChange(e.target.value)}
- disabled={pendingAction === 'model' || (localEnabled && isRunning)}
- className="w-full bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-md px-3 py-2 text-white appearance-none cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed pr-10"
- >
- {modelsData?.models && Object.entries(modelsData.models)
- .sort(([, a], [, b]) => (a as string).localeCompare(b as string))
- .map(([code, name]) => (
- <option key={code} value={code}>
- {name}
- </option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- {localEnabled && isRunning && (
- <p className="text-xs text-yellow-400 mt-2">
- <AlertTriangle className="w-3 h-3 inline mr-1" />
- Disable the virtual printer to change the model
- </p>
- )}
- </div>
- {/* Access Code */}
- <div className="py-3 border-t border-bambu-dark-tertiary">
- <div className="text-white font-medium mb-2">Access Code</div>
- <div className="text-sm text-bambu-gray mb-3">
- {settings?.access_code_set ? (
- <span className="flex items-center gap-1 text-green-400">
- <Check className="w-4 h-4" />
- Access code is set
- </span>
- ) : (
- <span className="flex items-center gap-1 text-yellow-400">
- <AlertTriangle className="w-4 h-4" />
- No access code set - required to enable
- </span>
- )}
- </div>
- <div className="flex gap-2">
- <div className="relative flex-1">
- <input
- type={showAccessCode ? 'text' : 'password'}
- value={localAccessCode}
- onChange={(e) => setLocalAccessCode(e.target.value)}
- placeholder={settings?.access_code_set ? 'Enter new code to change' : 'Enter 8-char code'}
- maxLength={8}
- className="w-full bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-md px-3 py-2 text-white placeholder-bambu-gray pr-10 font-mono"
- />
- <button
- onClick={() => setShowAccessCode(!showAccessCode)}
- className="absolute right-2 top-1/2 -translate-y-1/2 text-bambu-gray hover:text-white"
- >
- {showAccessCode ? <EyeOff className="w-4 h-4" /> : <Eye className="w-4 h-4" />}
- </button>
- </div>
- <Button
- onClick={handleAccessCodeChange}
- disabled={!localAccessCode || pendingAction === 'accessCode'}
- variant="primary"
- >
- {pendingAction === 'accessCode' ? <Loader2 className="w-4 h-4 animate-spin" /> : 'Save'}
- </Button>
- </div>
- <p className="text-xs text-bambu-gray mt-2">
- Must be exactly 8 characters. Used by slicers to authenticate.
- {localAccessCode && (
- <span className={localAccessCode.length === 8 ? 'text-green-400' : 'text-yellow-400'}>
- {' '}({localAccessCode.length}/8)
- </span>
- )}
- </p>
- </div>
- {/* Archive Mode */}
- <div className="py-3 border-t border-bambu-dark-tertiary">
- <div className="text-white font-medium mb-2">Archive Mode</div>
- <div className="grid grid-cols-2 gap-3">
- <button
- onClick={() => handleModeChange('immediate')}
- disabled={pendingAction === 'mode'}
- className={`p-3 rounded-lg border text-left transition-colors ${
- localMode === 'immediate'
- ? 'border-bambu-green bg-bambu-green/10'
- : 'border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- <div className="text-white font-medium">Immediate</div>
- <div className="text-xs text-bambu-gray">Archive files as soon as they are uploaded</div>
- </button>
- <button
- onClick={() => handleModeChange('queue')}
- disabled={pendingAction === 'mode'}
- className={`p-3 rounded-lg border text-left transition-colors ${
- localMode === 'queue'
- ? 'border-bambu-green bg-bambu-green/10'
- : 'border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- <div className="text-white font-medium">Queue for Review</div>
- <div className="text-xs text-bambu-gray">Review and tag files before archiving</div>
- </button>
- </div>
- </div>
- </CardContent>
- </Card>
- </div>
- {/* Right Column - Info & Status */}
- <div className="space-y-6 lg:w-[480px] lg:flex-shrink-0">
- {/* Setup Required Warning */}
- <Card className="border-l-4 border-l-yellow-500">
- <CardContent className="py-4">
- <div className="flex items-start gap-3">
- <AlertTriangle className="w-5 h-5 text-yellow-500 flex-shrink-0 mt-0.5" />
- <div className="text-sm">
- <p className="text-white font-medium mb-2">
- Setup Required
- </p>
- <p className="text-bambu-gray mb-3">
- The virtual printer feature requires additional system configuration before it will work.
- This includes port forwarding, firewall rules, and platform-specific settings.
- </p>
- <a
- href="https://wiki.bambuddy.cool/features/virtual-printer/"
- target="_blank"
- rel="noopener noreferrer"
- className="inline-flex items-center gap-2 px-4 py-2 bg-yellow-500/20 border border-yellow-500/50 rounded-md text-yellow-400 hover:bg-yellow-500/30 transition-colors"
- >
- <ExternalLink className="w-4 h-4" />
- Read the setup guide before enabling
- </a>
- </div>
- </div>
- </CardContent>
- </Card>
- {/* How it works */}
- <Card>
- <CardContent className="py-4">
- <div className="flex items-start gap-3">
- <Info className="w-5 h-5 text-blue-400 flex-shrink-0 mt-0.5" />
- <div className="text-sm text-bambu-gray">
- <p className="mb-2">
- <strong className="text-white">How it works:</strong>
- </p>
- <ol className="list-decimal list-inside space-y-1">
- <li>Complete the setup guide for your platform</li>
- <li>Enable the virtual printer and set an access code</li>
- <li>In Bambu Studio or OrcaSlicer, go to "Add Printer"</li>
- <li>The "Bambuddy" printer should appear in the discovery list</li>
- <li>Connect using the access code you set</li>
- <li>When you "print" to Bambuddy, the 3MF file is archived instead</li>
- </ol>
- </div>
- </div>
- </CardContent>
- </Card>
- {/* Status Details (when running) */}
- {status && isRunning && (
- <Card>
- <CardHeader>
- <h3 className="text-md font-semibold text-white">Status Details</h3>
- </CardHeader>
- <CardContent>
- <div className="grid grid-cols-2 gap-4 text-sm">
- <div>
- <div className="text-bambu-gray">Printer Name</div>
- <div className="text-white">{status.name}</div>
- </div>
- <div>
- <div className="text-bambu-gray">Model</div>
- <div className="text-white">{status.model_name || status.model}</div>
- </div>
- <div>
- <div className="text-bambu-gray">Serial Number</div>
- <div className="text-white font-mono">{status.serial}</div>
- </div>
- <div>
- <div className="text-bambu-gray">Mode</div>
- <div className="text-white capitalize">{status.mode}</div>
- </div>
- <div>
- <div className="text-bambu-gray">Pending Files</div>
- <div className="text-white">{status.pending_files}</div>
- </div>
- </div>
- </CardContent>
- </Card>
- )}
- </div>
- </div>
- );
- }
|