wu981526092 commited on
Commit
27615c9
·
1 Parent(s): be62bc1

✨ UPDATE CHAT AND DEVICE COMPONENTS: Refactor for improved layout and functionality

Browse files

✅ Changes made:
- Adjusted layout in the Chat component for better responsiveness and maximum height constraints.
- Refactored the MyDevice page to enhance code consistency and readability, including updated state management and improved button interactions.
- Updated the Playground page to ensure full height for the component, enhancing user experience.

🛠️ These updates contribute to a more robust and user-friendly interface across the application!

frontend/src/components/ui/chat.tsx CHANGED
@@ -66,14 +66,14 @@ const Chat = React.forwardRef<HTMLDivElement, ChatProps>(
66
 
67
  return (
68
  <div
69
- className={cn("flex h-full flex-col", className)}
70
  ref={ref}
71
  {...props}
72
  >
73
  {/* Messages */}
74
  <div
75
  ref={messagesContainerRef}
76
- className="flex-1 overflow-y-auto p-4 space-y-4"
77
  >
78
  {messages.length === 0 ? (
79
  <div className="flex items-center justify-center h-full text-muted-foreground">
 
66
 
67
  return (
68
  <div
69
+ className={cn("flex flex-col h-full max-h-screen", className)}
70
  ref={ref}
71
  {...props}
72
  >
73
  {/* Messages */}
74
  <div
75
  ref={messagesContainerRef}
76
+ className="flex-1 overflow-y-auto p-4 space-y-4 max-h-96 min-h-0"
77
  >
78
  {messages.length === 0 ? (
79
  <div className="flex items-center justify-center h-full text-muted-foreground">
frontend/src/pages/MyDevice.tsx CHANGED
@@ -1,85 +1,92 @@
1
- import { useState, useEffect } from 'react'
2
- import { useNavigate } from 'react-router-dom'
3
- import { Button } from '@/components/ui/button'
4
- import { Card } from '@/components/ui/card'
5
- import { Input } from '@/components/ui/input'
6
- import { Label } from '@/components/ui/label'
7
- import { Textarea } from '@/components/ui/textarea'
8
- import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select'
9
- import { Slider } from '@/components/ui/slider'
10
- import { Badge } from '@/components/ui/badge'
11
- import { ASSISTANT_CONFIGS } from '@/config/assistants'
12
- import { useChat } from '@/hooks/useChat'
13
- import { DocumentsTab } from '@/components/playground/DocumentsTab'
 
 
 
 
 
 
14
 
15
  interface DeviceConfig {
16
- assistantId: string
17
- model: string
18
- temperature: number
19
- maxTokens: number
20
- role: string
21
- goal: string
22
- description: string
23
- document: string
24
  }
25
 
26
  interface SelectedAssistant {
27
- id: string
28
- name: string
29
- type: 'user'|'template'|'new'
30
- originalTemplate?: string
31
  }
32
 
33
  export function MyDevice() {
34
- const navigate = useNavigate()
35
- const [currentStep, setCurrentStep] = useState(1)
36
  const [deviceConfig, setDeviceConfig] = useState<DeviceConfig>({
37
- assistantId: '',
38
- model: 'Qwen/Qwen3-30B-A3B',
39
  temperature: 0.7,
40
  maxTokens: 1024,
41
- role: '',
42
- goal: '',
43
- description: '',
44
- document: ''
45
- })
46
-
47
  // RAG document state
48
- const [ragEnabled, setRagEnabled] = useState(false)
49
- const [retrievalCount, setRetrievalCount] = useState(3)
50
- const [currentAssistant, setCurrentAssistant] = useState<SelectedAssistant | null>(null)
 
51
 
52
  // Load assistant configuration when selected
53
  const handleAssistantSelect = (assistantId: string) => {
54
- if (assistantId === 'create-new') {
55
  // Reset to blank configuration for new assistant
56
- setDeviceConfig(prev => ({
57
  ...prev,
58
- assistantId: 'create-new',
59
- model: 'Qwen/Qwen3-30B-A3B',
60
  temperature: 0.7,
61
  maxTokens: 1024,
62
- role: '',
63
- goal: '',
64
- description: ''
65
- }))
66
-
67
  // Reset chat settings
68
- setSelectedModel('Qwen/Qwen3-30B-A3B')
69
- setSystemPrompt('')
70
- setTemperature(0.7)
71
- setMaxTokens(1024)
72
-
73
  // Set current assistant for RAG
74
  setCurrentAssistant({
75
- id: 'create-new',
76
- name: 'New Custom Assistant',
77
- type: 'new'
78
- })
79
  } else {
80
- const config = assistantConfigs.find(c => c.id === assistantId)
81
  if (config) {
82
- setDeviceConfig(prev => ({
83
  ...prev,
84
  assistantId,
85
  model: config.model,
@@ -87,33 +94,33 @@ export function MyDevice() {
87
  maxTokens: config.maxTokens,
88
  role: config.description,
89
  goal: config.category,
90
- description: config.systemPrompt
91
- }))
92
-
93
  // Also update chat settings for Step 2
94
- setSelectedModel(config.model)
95
- setSystemPrompt(config.systemPrompt)
96
- setTemperature(config.temperature)
97
- setMaxTokens(config.maxTokens)
98
-
99
  // Set current assistant for RAG
100
  setCurrentAssistant({
101
  id: assistantId,
102
  name: config.name,
103
- type: 'template'
104
- })
105
  }
106
  }
107
- }
108
 
109
  // Check for pre-selected use case from Use Cases page
110
  useEffect(() => {
111
- const selectedUseCase = localStorage.getItem('selectedUseCase')
112
  if (selectedUseCase) {
113
- handleAssistantSelect(selectedUseCase)
114
- localStorage.removeItem('selectedUseCase') // Clear after use
115
  }
116
- }, [])
117
 
118
  // Chat functionality for Step 2
119
  const {
@@ -126,56 +133,94 @@ export function MyDevice() {
126
  setMaxTokens,
127
  setSelectedModel,
128
  input,
129
- setInput
130
- } = useChat()
131
 
132
- const assistantConfigs = ASSISTANT_CONFIGS
133
- const messages = currentSession?.messages || []
134
 
135
  // Sync deviceConfig changes to chat settings
136
  useEffect(() => {
137
- setSelectedModel(deviceConfig.model)
138
- setSystemPrompt(deviceConfig.description)
139
- setTemperature(deviceConfig.temperature)
140
- setMaxTokens(deviceConfig.maxTokens)
141
- }, [deviceConfig, setSelectedModel, setSystemPrompt, setTemperature, setMaxTokens])
 
 
 
 
 
 
142
 
143
  const renderStepIndicator = () => (
144
  <div className="flex items-center justify-between mb-8 max-w-4xl mx-auto">
145
  <div className="flex items-center">
146
- <div className={`flex items-center ${currentStep >= 1 ? 'text-purple-600' : 'text-gray-400'}`}>
147
- <div className={`w-8 h-8 rounded-full border-2 flex items-center justify-center text-sm font-medium ${
148
- currentStep >= 1 ? 'bg-purple-600 text-white border-purple-600' : 'border-gray-300'
149
- }`}>
 
 
 
 
 
 
 
 
150
  1
151
  </div>
152
  <span className="ml-2 font-medium">Configuration</span>
153
  </div>
154
- <div className={`w-32 h-0.5 mx-4 ${currentStep >= 2 ? 'bg-purple-600' : 'bg-gray-300'}`} />
 
 
 
 
155
  </div>
156
-
157
  <div className="flex items-center">
158
- <div className={`flex items-center ${currentStep >= 2 ? 'text-purple-600' : 'text-gray-400'}`}>
159
- <div className={`w-8 h-8 rounded-full border-2 flex items-center justify-center text-sm font-medium ${
160
- currentStep >= 2 ? 'bg-purple-600 text-white border-purple-600' : 'border-gray-300'
161
- }`}>
 
 
 
 
 
 
 
 
162
  2
163
  </div>
164
  <span className="ml-2 font-medium">Test performance</span>
165
  </div>
166
- <div className={`w-32 h-0.5 mx-4 ${currentStep >= 3 ? 'bg-purple-600' : 'bg-gray-300'}`} />
 
 
 
 
167
  </div>
168
 
169
  <div className="flex items-center">
170
- <div className={`flex items-center ${currentStep >= 3 ? 'text-purple-600' : 'text-gray-400'}`}>
171
- <div className={`w-8 h-8 rounded-full border-2 flex items-center justify-center text-sm font-medium ${
172
- currentStep >= 3 ? 'bg-purple-600 text-white border-purple-600' : 'border-gray-300'
173
- }`}>
 
 
 
 
 
 
 
 
174
  3
175
  </div>
176
  <span className="ml-2 font-medium">Deploy to device</span>
177
  </div>
178
- <Button
179
  className="ml-8 bg-purple-600 hover:bg-purple-700 text-white px-8"
180
  onClick={() => setCurrentStep(3)}
181
  disabled={currentStep < 2}
@@ -184,7 +229,7 @@ export function MyDevice() {
184
  </Button>
185
  </div>
186
  </div>
187
- )
188
 
189
  const renderStep1 = () => (
190
  <div className="max-w-2xl mx-auto">
@@ -192,8 +237,13 @@ export function MyDevice() {
192
  <div className="space-y-6">
193
  {/* Load Assistant */}
194
  <div>
195
- <Label className="text-sm font-medium text-gray-700 mb-2 block">Load Assistant</Label>
196
- <Select onValueChange={handleAssistantSelect} value={deviceConfig.assistantId}>
 
 
 
 
 
197
  <SelectTrigger>
198
  <SelectValue placeholder="Select an assistant or create new" />
199
  </SelectTrigger>
@@ -209,7 +259,7 @@ export function MyDevice() {
209
  <SelectItem key={config.id} value={config.id}>
210
  <div className="flex items-center gap-2">
211
  <span>{config.icon}</span>
212
- <span>{config.name.replace(/🏔️|🏥|🧸/g, '').trim()}</span>
213
  </div>
214
  </SelectItem>
215
  ))}
@@ -220,21 +270,31 @@ export function MyDevice() {
220
  {/* Parameters */}
221
  <div>
222
  <h3 className="text-lg font-semibold mb-4">Parameters</h3>
223
-
224
  {/* Model Selection */}
225
  <div className="mb-4">
226
- <Label className="text-sm font-medium text-gray-700 mb-2 block">Select a model</Label>
227
- <Select
228
- value={deviceConfig.model}
229
- onValueChange={(value) => setDeviceConfig(prev => ({ ...prev, model: value }))}
 
 
 
 
230
  >
231
  <SelectTrigger>
232
  <SelectValue />
233
  </SelectTrigger>
234
  <SelectContent>
235
- <SelectItem value="Qwen/Qwen3-30B-A3B">Qwen3-30B-A3B</SelectItem>
236
- <SelectItem value="Qwen/Qwen2.5-32B-Instruct">Qwen2.5-32B-Instruct</SelectItem>
237
- <SelectItem value="deepseek-ai/DeepSeek-V2.5">DeepSeek-V2.5</SelectItem>
 
 
 
 
 
 
238
  </SelectContent>
239
  </Select>
240
  </div>
@@ -246,7 +306,12 @@ export function MyDevice() {
246
  </Label>
247
  <Slider
248
  value={[deviceConfig.temperature]}
249
- onValueChange={(value) => setDeviceConfig(prev => ({ ...prev, temperature: value[0] }))}
 
 
 
 
 
250
  max={1}
251
  min={0}
252
  step={0.1}
@@ -256,11 +321,18 @@ export function MyDevice() {
256
 
257
  {/* Max Tokens */}
258
  <div className="mb-6">
259
- <Label className="text-sm font-medium text-gray-700 mb-2 block">Max Tokens</Label>
 
 
260
  <Input
261
  type="number"
262
  value={deviceConfig.maxTokens}
263
- onChange={(e) => setDeviceConfig(prev => ({ ...prev, maxTokens: parseInt(e.target.value) || 1024 }))}
 
 
 
 
 
264
  min={1}
265
  max={4096}
266
  />
@@ -270,14 +342,17 @@ export function MyDevice() {
270
  {/* Description */}
271
  <div>
272
  <h3 className="text-lg font-semibold mb-4">Description</h3>
273
-
274
  <div className="mb-4">
275
  <Label className="text-sm font-medium text-gray-700 mb-2 block">
276
- Which role do you want your assistant to be? (Senior data researcher, journalist, lawyer, etc.)
 
277
  </Label>
278
  <Input
279
  value={deviceConfig.role}
280
- onChange={(e) => setDeviceConfig(prev => ({ ...prev, role: e.target.value }))}
 
 
281
  placeholder="e.g., Senior data researcher, journalist, lawyer"
282
  />
283
  </div>
@@ -288,7 +363,9 @@ export function MyDevice() {
288
  </Label>
289
  <Input
290
  value={deviceConfig.goal}
291
- onChange={(e) => setDeviceConfig(prev => ({ ...prev, goal: e.target.value }))}
 
 
292
  placeholder="Describe the main goal or objective"
293
  />
294
  </div>
@@ -299,7 +376,12 @@ export function MyDevice() {
299
  </Label>
300
  <Textarea
301
  value={deviceConfig.description}
302
- onChange={(e) => setDeviceConfig(prev => ({ ...prev, description: e.target.value }))}
 
 
 
 
 
303
  placeholder="Describe the tasks and expected outputs..."
304
  rows={4}
305
  />
@@ -308,7 +390,9 @@ export function MyDevice() {
308
 
309
  {/* Document - RAG Integration */}
310
  <div>
311
- <Label className="text-sm font-medium text-gray-700 mb-2 block">Documents & Knowledge Base</Label>
 
 
312
  <DocumentsTab
313
  isLoading={isLoading}
314
  ragEnabled={ragEnabled}
@@ -319,21 +403,25 @@ export function MyDevice() {
319
  />
320
  </div>
321
 
322
- <Button
323
- onClick={() => setCurrentStep(2)}
324
  className="w-full bg-purple-600 hover:bg-purple-700 text-white"
325
- disabled={!deviceConfig.assistantId || (deviceConfig.assistantId === 'create-new' && !deviceConfig.description.trim())}
 
 
 
 
326
  >
327
  Continue to Test Performance
328
  </Button>
329
  </div>
330
  </Card>
331
  </div>
332
- )
333
 
334
  const renderStep2 = () => (
335
  <div className="max-w-6xl mx-auto">
336
- <div className="grid grid-cols-1 lg:grid-cols-4 gap-6 h-[600px]">
337
  {/* Chat Sessions Sidebar */}
338
  <div className="lg:col-span-1">
339
  <Card className="h-full p-4">
@@ -362,31 +450,43 @@ export function MyDevice() {
362
  <div className="flex items-center justify-between">
363
  <h3 className="font-semibold">Test Performance</h3>
364
  <Badge variant="secondary">
365
- {deviceConfig.assistantId === 'create-new' ? 'Custom Assistant' :
366
- deviceConfig.assistantId ? assistantConfigs.find(c => c.id === deviceConfig.assistantId)?.name.replace(/🏔️|🏥|🧸/g, '').trim() : 'No Assistant'}
 
 
 
 
 
 
367
  </Badge>
368
  </div>
369
  </div>
370
-
371
  <div className="flex-1 overflow-auto p-4 space-y-4">
372
  {messages.length === 0 ? (
373
  <div className="flex items-center justify-center h-full text-gray-500">
374
  <div className="text-center">
375
- <p className="mb-2">Start testing your assistant configuration</p>
376
- <p className="text-sm">Send a message to see how your assistant responds</p>
 
 
 
 
377
  </div>
378
  </div>
379
  ) : (
380
  messages.map((message, index) => (
381
  <div
382
  key={index}
383
- className={`flex ${message.role === 'user' ? 'justify-end' : 'justify-start'}`}
 
 
384
  >
385
  <div
386
  className={`max-w-[80%] p-3 rounded-lg ${
387
- message.role === 'user'
388
- ? 'bg-purple-600 text-white'
389
- : 'bg-gray-100 text-gray-900'
390
  }`}
391
  >
392
  <p className="whitespace-pre-wrap">{message.content}</p>
@@ -399,8 +499,14 @@ export function MyDevice() {
399
  <div className="bg-gray-100 p-3 rounded-lg">
400
  <div className="flex items-center space-x-2">
401
  <div className="w-2 h-2 bg-gray-400 rounded-full animate-bounce"></div>
402
- <div className="w-2 h-2 bg-gray-400 rounded-full animate-bounce" style={{ animationDelay: '0.1s' }}></div>
403
- <div className="w-2 h-2 bg-gray-400 rounded-full animate-bounce" style={{ animationDelay: '0.2s' }}></div>
 
 
 
 
 
 
404
  </div>
405
  </div>
406
  </div>
@@ -414,18 +520,24 @@ export function MyDevice() {
414
  value={input}
415
  onChange={(e) => setInput(e.target.value)}
416
  onKeyPress={(e) => {
417
- if (e.key === 'Enter' && !e.shiftKey) {
418
- e.preventDefault()
419
  if (input.trim()) {
420
- sendMessage(undefined, { useRag: ragEnabled, retrievalCount })
 
 
 
421
  }
422
  }
423
  }}
424
  />
425
- <Button
426
  onClick={() => {
427
  if (input.trim()) {
428
- sendMessage(undefined, { useRag: ragEnabled, retrievalCount })
 
 
 
429
  }
430
  }}
431
  disabled={isLoading}
@@ -440,34 +552,44 @@ export function MyDevice() {
440
  </div>
441
 
442
  <div className="mt-6 flex justify-between max-w-6xl mx-auto">
443
- <Button
444
- variant="outline"
445
- onClick={() => setCurrentStep(1)}
446
- >
447
  Back to Configuration
448
  </Button>
449
- <Button
450
- onClick={() => setCurrentStep(3)}
451
  className="bg-purple-600 hover:bg-purple-700 text-white"
452
  >
453
  Continue to Deploy Device
454
  </Button>
455
  </div>
456
  </div>
457
- )
458
 
459
  const renderStep3 = () => (
460
  <div className="max-w-2xl mx-auto">
461
  <Card className="p-8 text-center">
462
  <div className="mb-6">
463
  <div className="w-16 h-16 bg-purple-100 rounded-full flex items-center justify-center mx-auto mb-4">
464
- <svg className="w-8 h-8 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
465
- <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M8.111 16.404a5.5 5.5 0 017.778 0M12 20h.01m-7.08-7.071c3.904-3.905 10.236-3.905 14.141 0M1.394 9.393c5.857-5.857 15.355-5.857 21.212 0" />
 
 
 
 
 
 
 
 
 
 
466
  </svg>
467
  </div>
468
- <h2 className="text-2xl font-bold text-gray-900 mb-2">Deploy to Device</h2>
 
 
469
  <p className="text-gray-600">
470
- Your assistant configuration is ready to be deployed to your edge device.
 
471
  </p>
472
  </div>
473
 
@@ -475,26 +597,62 @@ export function MyDevice() {
475
  <div className="bg-gray-50 p-4 rounded-lg">
476
  <h3 className="font-semibold mb-2">Configuration Summary</h3>
477
  <div className="space-y-1 text-sm">
478
- <p><span className="font-medium">Assistant:</span> {
479
- deviceConfig.assistantId === 'create-new' ? 'Custom Assistant' :
480
- assistantConfigs.find(c => c.id === deviceConfig.assistantId)?.name.replace(/🏔️|🏥|🧸/g, '').trim() || 'None'
481
- }</p>
482
- <p><span className="font-medium">Model:</span> {deviceConfig.model}</p>
483
- <p><span className="font-medium">Temperature:</span> {deviceConfig.temperature}</p>
484
- <p><span className="font-medium">Max Tokens:</span> {deviceConfig.maxTokens}</p>
485
- {deviceConfig.role && <p><span className="font-medium">Role:</span> {deviceConfig.role}</p>}
486
- <p><span className="font-medium">RAG Enabled:</span> {ragEnabled ? 'Yes' : 'No'}</p>
487
- {ragEnabled && <p><span className="font-medium">Retrieval Count:</span> {retrievalCount}</p>}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
488
  </div>
489
  </div>
490
-
491
  <div className="bg-blue-50 p-4 rounded-lg border border-blue-200">
492
  <div className="flex items-center">
493
- <svg className="w-5 h-5 text-blue-600 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
494
- <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
 
 
 
 
 
 
 
 
 
 
495
  </svg>
496
  <p className="text-blue-800 text-sm">
497
- <span className="font-medium">Ready to deploy:</span> Your assistant configuration has been optimized for edge deployment.
 
498
  </p>
499
  </div>
500
  </div>
@@ -504,13 +662,17 @@ export function MyDevice() {
504
  <Button className="w-full bg-purple-600 hover:bg-purple-700 text-white py-3">
505
  Deploy to Device
506
  </Button>
507
- <Button variant="outline" className="w-full" onClick={() => setCurrentStep(2)}>
 
 
 
 
508
  Back to Testing
509
  </Button>
510
  </div>
511
  </Card>
512
  </div>
513
- )
514
 
515
  return (
516
  <div className="min-h-screen bg-gray-50">
@@ -519,38 +681,36 @@ export function MyDevice() {
519
  <div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
520
  <div className="flex items-center justify-between h-20">
521
  <div className="flex items-center space-x-2">
522
- <img
523
- src="/assets/logo.png"
524
- alt="EdgeLLM Logo"
525
  className="h-16 w-16"
526
  onError={(e) => {
527
- console.error('Logo failed to load')
528
- e.currentTarget.style.display = 'none'
529
  }}
530
  />
531
  </div>
532
  <nav className="flex space-x-8">
533
- <button
534
- onClick={() => navigate('/')}
535
  className="text-gray-600 hover:text-gray-900 px-3 py-2 text-sm font-medium"
536
  >
537
  Home
538
  </button>
539
- <button
540
- onClick={() => navigate('/technology')}
541
  className="text-gray-600 hover:text-gray-900 px-3 py-2 text-sm font-medium"
542
  >
543
  Technology
544
  </button>
545
- <button
546
- onClick={() => navigate('/usecases')}
547
  className="text-gray-600 hover:text-gray-900 px-3 py-2 text-sm font-medium"
548
  >
549
  Use Case
550
  </button>
551
- <button
552
- className="bg-purple-600 text-white px-4 py-2 text-sm font-medium rounded"
553
- >
554
  My Device
555
  </button>
556
  </nav>
@@ -561,11 +721,11 @@ export function MyDevice() {
561
  {/* Main Content */}
562
  <main className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
563
  {renderStepIndicator()}
564
-
565
  {currentStep === 1 && renderStep1()}
566
  {currentStep === 2 && renderStep2()}
567
  {currentStep === 3 && renderStep3()}
568
  </main>
569
  </div>
570
- )
571
  }
 
1
+ import { useState, useEffect } from "react";
2
+ import { useNavigate } from "react-router-dom";
3
+ import { Button } from "@/components/ui/button";
4
+ import { Card } from "@/components/ui/card";
5
+ import { Input } from "@/components/ui/input";
6
+ import { Label } from "@/components/ui/label";
7
+ import { Textarea } from "@/components/ui/textarea";
8
+ import {
9
+ Select,
10
+ SelectContent,
11
+ SelectItem,
12
+ SelectTrigger,
13
+ SelectValue,
14
+ } from "@/components/ui/select";
15
+ import { Slider } from "@/components/ui/slider";
16
+ import { Badge } from "@/components/ui/badge";
17
+ import { ASSISTANT_CONFIGS } from "@/config/assistants";
18
+ import { useChat } from "@/hooks/useChat";
19
+ import { DocumentsTab } from "@/components/playground/DocumentsTab";
20
 
21
  interface DeviceConfig {
22
+ assistantId: string;
23
+ model: string;
24
+ temperature: number;
25
+ maxTokens: number;
26
+ role: string;
27
+ goal: string;
28
+ description: string;
29
+ document: string;
30
  }
31
 
32
  interface SelectedAssistant {
33
+ id: string;
34
+ name: string;
35
+ type: "user" | "template" | "new";
36
+ originalTemplate?: string;
37
  }
38
 
39
  export function MyDevice() {
40
+ const navigate = useNavigate();
41
+ const [currentStep, setCurrentStep] = useState(1);
42
  const [deviceConfig, setDeviceConfig] = useState<DeviceConfig>({
43
+ assistantId: "",
44
+ model: "Qwen/Qwen3-30B-A3B",
45
  temperature: 0.7,
46
  maxTokens: 1024,
47
+ role: "",
48
+ goal: "",
49
+ description: "",
50
+ document: "",
51
+ });
52
+
53
  // RAG document state
54
+ const [ragEnabled, setRagEnabled] = useState(false);
55
+ const [retrievalCount, setRetrievalCount] = useState(3);
56
+ const [currentAssistant, setCurrentAssistant] =
57
+ useState<SelectedAssistant | null>(null);
58
 
59
  // Load assistant configuration when selected
60
  const handleAssistantSelect = (assistantId: string) => {
61
+ if (assistantId === "create-new") {
62
  // Reset to blank configuration for new assistant
63
+ setDeviceConfig((prev) => ({
64
  ...prev,
65
+ assistantId: "create-new",
66
+ model: "Qwen/Qwen3-30B-A3B",
67
  temperature: 0.7,
68
  maxTokens: 1024,
69
+ role: "",
70
+ goal: "",
71
+ description: "",
72
+ }));
73
+
74
  // Reset chat settings
75
+ setSelectedModel("Qwen/Qwen3-30B-A3B");
76
+ setSystemPrompt("");
77
+ setTemperature(0.7);
78
+ setMaxTokens(1024);
79
+
80
  // Set current assistant for RAG
81
  setCurrentAssistant({
82
+ id: "create-new",
83
+ name: "New Custom Assistant",
84
+ type: "new",
85
+ });
86
  } else {
87
+ const config = assistantConfigs.find((c) => c.id === assistantId);
88
  if (config) {
89
+ setDeviceConfig((prev) => ({
90
  ...prev,
91
  assistantId,
92
  model: config.model,
 
94
  maxTokens: config.maxTokens,
95
  role: config.description,
96
  goal: config.category,
97
+ description: config.systemPrompt,
98
+ }));
99
+
100
  // Also update chat settings for Step 2
101
+ setSelectedModel(config.model);
102
+ setSystemPrompt(config.systemPrompt);
103
+ setTemperature(config.temperature);
104
+ setMaxTokens(config.maxTokens);
105
+
106
  // Set current assistant for RAG
107
  setCurrentAssistant({
108
  id: assistantId,
109
  name: config.name,
110
+ type: "template",
111
+ });
112
  }
113
  }
114
+ };
115
 
116
  // Check for pre-selected use case from Use Cases page
117
  useEffect(() => {
118
+ const selectedUseCase = localStorage.getItem("selectedUseCase");
119
  if (selectedUseCase) {
120
+ handleAssistantSelect(selectedUseCase);
121
+ localStorage.removeItem("selectedUseCase"); // Clear after use
122
  }
123
+ }, []);
124
 
125
  // Chat functionality for Step 2
126
  const {
 
133
  setMaxTokens,
134
  setSelectedModel,
135
  input,
136
+ setInput,
137
+ } = useChat();
138
 
139
+ const assistantConfigs = ASSISTANT_CONFIGS;
140
+ const messages = currentSession?.messages || [];
141
 
142
  // Sync deviceConfig changes to chat settings
143
  useEffect(() => {
144
+ setSelectedModel(deviceConfig.model);
145
+ setSystemPrompt(deviceConfig.description);
146
+ setTemperature(deviceConfig.temperature);
147
+ setMaxTokens(deviceConfig.maxTokens);
148
+ }, [
149
+ deviceConfig,
150
+ setSelectedModel,
151
+ setSystemPrompt,
152
+ setTemperature,
153
+ setMaxTokens,
154
+ ]);
155
 
156
  const renderStepIndicator = () => (
157
  <div className="flex items-center justify-between mb-8 max-w-4xl mx-auto">
158
  <div className="flex items-center">
159
+ <div
160
+ className={`flex items-center ${
161
+ currentStep >= 1 ? "text-purple-600" : "text-gray-400"
162
+ }`}
163
+ >
164
+ <div
165
+ className={`w-8 h-8 rounded-full border-2 flex items-center justify-center text-sm font-medium ${
166
+ currentStep >= 1
167
+ ? "bg-purple-600 text-white border-purple-600"
168
+ : "border-gray-300"
169
+ }`}
170
+ >
171
  1
172
  </div>
173
  <span className="ml-2 font-medium">Configuration</span>
174
  </div>
175
+ <div
176
+ className={`w-32 h-0.5 mx-4 ${
177
+ currentStep >= 2 ? "bg-purple-600" : "bg-gray-300"
178
+ }`}
179
+ />
180
  </div>
181
+
182
  <div className="flex items-center">
183
+ <div
184
+ className={`flex items-center ${
185
+ currentStep >= 2 ? "text-purple-600" : "text-gray-400"
186
+ }`}
187
+ >
188
+ <div
189
+ className={`w-8 h-8 rounded-full border-2 flex items-center justify-center text-sm font-medium ${
190
+ currentStep >= 2
191
+ ? "bg-purple-600 text-white border-purple-600"
192
+ : "border-gray-300"
193
+ }`}
194
+ >
195
  2
196
  </div>
197
  <span className="ml-2 font-medium">Test performance</span>
198
  </div>
199
+ <div
200
+ className={`w-32 h-0.5 mx-4 ${
201
+ currentStep >= 3 ? "bg-purple-600" : "bg-gray-300"
202
+ }`}
203
+ />
204
  </div>
205
 
206
  <div className="flex items-center">
207
+ <div
208
+ className={`flex items-center ${
209
+ currentStep >= 3 ? "text-purple-600" : "text-gray-400"
210
+ }`}
211
+ >
212
+ <div
213
+ className={`w-8 h-8 rounded-full border-2 flex items-center justify-center text-sm font-medium ${
214
+ currentStep >= 3
215
+ ? "bg-purple-600 text-white border-purple-600"
216
+ : "border-gray-300"
217
+ }`}
218
+ >
219
  3
220
  </div>
221
  <span className="ml-2 font-medium">Deploy to device</span>
222
  </div>
223
+ <Button
224
  className="ml-8 bg-purple-600 hover:bg-purple-700 text-white px-8"
225
  onClick={() => setCurrentStep(3)}
226
  disabled={currentStep < 2}
 
229
  </Button>
230
  </div>
231
  </div>
232
+ );
233
 
234
  const renderStep1 = () => (
235
  <div className="max-w-2xl mx-auto">
 
237
  <div className="space-y-6">
238
  {/* Load Assistant */}
239
  <div>
240
+ <Label className="text-sm font-medium text-gray-700 mb-2 block">
241
+ Load Assistant
242
+ </Label>
243
+ <Select
244
+ onValueChange={handleAssistantSelect}
245
+ value={deviceConfig.assistantId}
246
+ >
247
  <SelectTrigger>
248
  <SelectValue placeholder="Select an assistant or create new" />
249
  </SelectTrigger>
 
259
  <SelectItem key={config.id} value={config.id}>
260
  <div className="flex items-center gap-2">
261
  <span>{config.icon}</span>
262
+ <span>{config.name.replace(/🏔️|🏥|🧸/g, "").trim()}</span>
263
  </div>
264
  </SelectItem>
265
  ))}
 
270
  {/* Parameters */}
271
  <div>
272
  <h3 className="text-lg font-semibold mb-4">Parameters</h3>
273
+
274
  {/* Model Selection */}
275
  <div className="mb-4">
276
+ <Label className="text-sm font-medium text-gray-700 mb-2 block">
277
+ Select a model
278
+ </Label>
279
+ <Select
280
+ value={deviceConfig.model}
281
+ onValueChange={(value) =>
282
+ setDeviceConfig((prev) => ({ ...prev, model: value }))
283
+ }
284
  >
285
  <SelectTrigger>
286
  <SelectValue />
287
  </SelectTrigger>
288
  <SelectContent>
289
+ <SelectItem value="Qwen/Qwen3-30B-A3B">
290
+ Qwen3-30B-A3B
291
+ </SelectItem>
292
+ <SelectItem value="Qwen/Qwen2.5-32B-Instruct">
293
+ Qwen2.5-32B-Instruct
294
+ </SelectItem>
295
+ <SelectItem value="deepseek-ai/DeepSeek-V2.5">
296
+ DeepSeek-V2.5
297
+ </SelectItem>
298
  </SelectContent>
299
  </Select>
300
  </div>
 
306
  </Label>
307
  <Slider
308
  value={[deviceConfig.temperature]}
309
+ onValueChange={(value) =>
310
+ setDeviceConfig((prev) => ({
311
+ ...prev,
312
+ temperature: value[0],
313
+ }))
314
+ }
315
  max={1}
316
  min={0}
317
  step={0.1}
 
321
 
322
  {/* Max Tokens */}
323
  <div className="mb-6">
324
+ <Label className="text-sm font-medium text-gray-700 mb-2 block">
325
+ Max Tokens
326
+ </Label>
327
  <Input
328
  type="number"
329
  value={deviceConfig.maxTokens}
330
+ onChange={(e) =>
331
+ setDeviceConfig((prev) => ({
332
+ ...prev,
333
+ maxTokens: parseInt(e.target.value) || 1024,
334
+ }))
335
+ }
336
  min={1}
337
  max={4096}
338
  />
 
342
  {/* Description */}
343
  <div>
344
  <h3 className="text-lg font-semibold mb-4">Description</h3>
345
+
346
  <div className="mb-4">
347
  <Label className="text-sm font-medium text-gray-700 mb-2 block">
348
+ Which role do you want your assistant to be? (Senior data
349
+ researcher, journalist, lawyer, etc.)
350
  </Label>
351
  <Input
352
  value={deviceConfig.role}
353
+ onChange={(e) =>
354
+ setDeviceConfig((prev) => ({ ...prev, role: e.target.value }))
355
+ }
356
  placeholder="e.g., Senior data researcher, journalist, lawyer"
357
  />
358
  </div>
 
363
  </Label>
364
  <Input
365
  value={deviceConfig.goal}
366
+ onChange={(e) =>
367
+ setDeviceConfig((prev) => ({ ...prev, goal: e.target.value }))
368
+ }
369
  placeholder="Describe the main goal or objective"
370
  />
371
  </div>
 
376
  </Label>
377
  <Textarea
378
  value={deviceConfig.description}
379
+ onChange={(e) =>
380
+ setDeviceConfig((prev) => ({
381
+ ...prev,
382
+ description: e.target.value,
383
+ }))
384
+ }
385
  placeholder="Describe the tasks and expected outputs..."
386
  rows={4}
387
  />
 
390
 
391
  {/* Document - RAG Integration */}
392
  <div>
393
+ <Label className="text-sm font-medium text-gray-700 mb-2 block">
394
+ Documents & Knowledge Base
395
+ </Label>
396
  <DocumentsTab
397
  isLoading={isLoading}
398
  ragEnabled={ragEnabled}
 
403
  />
404
  </div>
405
 
406
+ <Button
407
+ onClick={() => setCurrentStep(2)}
408
  className="w-full bg-purple-600 hover:bg-purple-700 text-white"
409
+ disabled={
410
+ !deviceConfig.assistantId ||
411
+ (deviceConfig.assistantId === "create-new" &&
412
+ !deviceConfig.description.trim())
413
+ }
414
  >
415
  Continue to Test Performance
416
  </Button>
417
  </div>
418
  </Card>
419
  </div>
420
+ );
421
 
422
  const renderStep2 = () => (
423
  <div className="max-w-6xl mx-auto">
424
+ <div className="grid grid-cols-1 lg:grid-cols-4 gap-6 min-h-[600px]">
425
  {/* Chat Sessions Sidebar */}
426
  <div className="lg:col-span-1">
427
  <Card className="h-full p-4">
 
450
  <div className="flex items-center justify-between">
451
  <h3 className="font-semibold">Test Performance</h3>
452
  <Badge variant="secondary">
453
+ {deviceConfig.assistantId === "create-new"
454
+ ? "Custom Assistant"
455
+ : deviceConfig.assistantId
456
+ ? assistantConfigs
457
+ .find((c) => c.id === deviceConfig.assistantId)
458
+ ?.name.replace(/🏔️|🏥|🧸/g, "")
459
+ .trim()
460
+ : "No Assistant"}
461
  </Badge>
462
  </div>
463
  </div>
464
+
465
  <div className="flex-1 overflow-auto p-4 space-y-4">
466
  {messages.length === 0 ? (
467
  <div className="flex items-center justify-center h-full text-gray-500">
468
  <div className="text-center">
469
+ <p className="mb-2">
470
+ Start testing your assistant configuration
471
+ </p>
472
+ <p className="text-sm">
473
+ Send a message to see how your assistant responds
474
+ </p>
475
  </div>
476
  </div>
477
  ) : (
478
  messages.map((message, index) => (
479
  <div
480
  key={index}
481
+ className={`flex ${
482
+ message.role === "user" ? "justify-end" : "justify-start"
483
+ }`}
484
  >
485
  <div
486
  className={`max-w-[80%] p-3 rounded-lg ${
487
+ message.role === "user"
488
+ ? "bg-purple-600 text-white"
489
+ : "bg-gray-100 text-gray-900"
490
  }`}
491
  >
492
  <p className="whitespace-pre-wrap">{message.content}</p>
 
499
  <div className="bg-gray-100 p-3 rounded-lg">
500
  <div className="flex items-center space-x-2">
501
  <div className="w-2 h-2 bg-gray-400 rounded-full animate-bounce"></div>
502
+ <div
503
+ className="w-2 h-2 bg-gray-400 rounded-full animate-bounce"
504
+ style={{ animationDelay: "0.1s" }}
505
+ ></div>
506
+ <div
507
+ className="w-2 h-2 bg-gray-400 rounded-full animate-bounce"
508
+ style={{ animationDelay: "0.2s" }}
509
+ ></div>
510
  </div>
511
  </div>
512
  </div>
 
520
  value={input}
521
  onChange={(e) => setInput(e.target.value)}
522
  onKeyPress={(e) => {
523
+ if (e.key === "Enter" && !e.shiftKey) {
524
+ e.preventDefault();
525
  if (input.trim()) {
526
+ sendMessage(undefined, {
527
+ useRag: ragEnabled,
528
+ retrievalCount,
529
+ });
530
  }
531
  }
532
  }}
533
  />
534
+ <Button
535
  onClick={() => {
536
  if (input.trim()) {
537
+ sendMessage(undefined, {
538
+ useRag: ragEnabled,
539
+ retrievalCount,
540
+ });
541
  }
542
  }}
543
  disabled={isLoading}
 
552
  </div>
553
 
554
  <div className="mt-6 flex justify-between max-w-6xl mx-auto">
555
+ <Button variant="outline" onClick={() => setCurrentStep(1)}>
 
 
 
556
  Back to Configuration
557
  </Button>
558
+ <Button
559
+ onClick={() => setCurrentStep(3)}
560
  className="bg-purple-600 hover:bg-purple-700 text-white"
561
  >
562
  Continue to Deploy Device
563
  </Button>
564
  </div>
565
  </div>
566
+ );
567
 
568
  const renderStep3 = () => (
569
  <div className="max-w-2xl mx-auto">
570
  <Card className="p-8 text-center">
571
  <div className="mb-6">
572
  <div className="w-16 h-16 bg-purple-100 rounded-full flex items-center justify-center mx-auto mb-4">
573
+ <svg
574
+ className="w-8 h-8 text-purple-600"
575
+ fill="none"
576
+ stroke="currentColor"
577
+ viewBox="0 0 24 24"
578
+ >
579
+ <path
580
+ strokeLinecap="round"
581
+ strokeLinejoin="round"
582
+ strokeWidth={2}
583
+ d="M8.111 16.404a5.5 5.5 0 017.778 0M12 20h.01m-7.08-7.071c3.904-3.905 10.236-3.905 14.141 0M1.394 9.393c5.857-5.857 15.355-5.857 21.212 0"
584
+ />
585
  </svg>
586
  </div>
587
+ <h2 className="text-2xl font-bold text-gray-900 mb-2">
588
+ Deploy to Device
589
+ </h2>
590
  <p className="text-gray-600">
591
+ Your assistant configuration is ready to be deployed to your edge
592
+ device.
593
  </p>
594
  </div>
595
 
 
597
  <div className="bg-gray-50 p-4 rounded-lg">
598
  <h3 className="font-semibold mb-2">Configuration Summary</h3>
599
  <div className="space-y-1 text-sm">
600
+ <p>
601
+ <span className="font-medium">Assistant:</span>{" "}
602
+ {deviceConfig.assistantId === "create-new"
603
+ ? "Custom Assistant"
604
+ : assistantConfigs
605
+ .find((c) => c.id === deviceConfig.assistantId)
606
+ ?.name.replace(/🏔️|🏥|🧸/g, "")
607
+ .trim() || "None"}
608
+ </p>
609
+ <p>
610
+ <span className="font-medium">Model:</span> {deviceConfig.model}
611
+ </p>
612
+ <p>
613
+ <span className="font-medium">Temperature:</span>{" "}
614
+ {deviceConfig.temperature}
615
+ </p>
616
+ <p>
617
+ <span className="font-medium">Max Tokens:</span>{" "}
618
+ {deviceConfig.maxTokens}
619
+ </p>
620
+ {deviceConfig.role && (
621
+ <p>
622
+ <span className="font-medium">Role:</span> {deviceConfig.role}
623
+ </p>
624
+ )}
625
+ <p>
626
+ <span className="font-medium">RAG Enabled:</span>{" "}
627
+ {ragEnabled ? "Yes" : "No"}
628
+ </p>
629
+ {ragEnabled && (
630
+ <p>
631
+ <span className="font-medium">Retrieval Count:</span>{" "}
632
+ {retrievalCount}
633
+ </p>
634
+ )}
635
  </div>
636
  </div>
637
+
638
  <div className="bg-blue-50 p-4 rounded-lg border border-blue-200">
639
  <div className="flex items-center">
640
+ <svg
641
+ className="w-5 h-5 text-blue-600 mr-2"
642
+ fill="none"
643
+ stroke="currentColor"
644
+ viewBox="0 0 24 24"
645
+ >
646
+ <path
647
+ strokeLinecap="round"
648
+ strokeLinejoin="round"
649
+ strokeWidth={2}
650
+ d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
651
+ />
652
  </svg>
653
  <p className="text-blue-800 text-sm">
654
+ <span className="font-medium">Ready to deploy:</span> Your
655
+ assistant configuration has been optimized for edge deployment.
656
  </p>
657
  </div>
658
  </div>
 
662
  <Button className="w-full bg-purple-600 hover:bg-purple-700 text-white py-3">
663
  Deploy to Device
664
  </Button>
665
+ <Button
666
+ variant="outline"
667
+ className="w-full"
668
+ onClick={() => setCurrentStep(2)}
669
+ >
670
  Back to Testing
671
  </Button>
672
  </div>
673
  </Card>
674
  </div>
675
+ );
676
 
677
  return (
678
  <div className="min-h-screen bg-gray-50">
 
681
  <div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
682
  <div className="flex items-center justify-between h-20">
683
  <div className="flex items-center space-x-2">
684
+ <img
685
+ src="/assets/logo.png"
686
+ alt="EdgeLLM Logo"
687
  className="h-16 w-16"
688
  onError={(e) => {
689
+ console.error("Logo failed to load");
690
+ e.currentTarget.style.display = "none";
691
  }}
692
  />
693
  </div>
694
  <nav className="flex space-x-8">
695
+ <button
696
+ onClick={() => navigate("/")}
697
  className="text-gray-600 hover:text-gray-900 px-3 py-2 text-sm font-medium"
698
  >
699
  Home
700
  </button>
701
+ <button
702
+ onClick={() => navigate("/technology")}
703
  className="text-gray-600 hover:text-gray-900 px-3 py-2 text-sm font-medium"
704
  >
705
  Technology
706
  </button>
707
+ <button
708
+ onClick={() => navigate("/usecases")}
709
  className="text-gray-600 hover:text-gray-900 px-3 py-2 text-sm font-medium"
710
  >
711
  Use Case
712
  </button>
713
+ <button className="bg-purple-600 text-white px-4 py-2 text-sm font-medium rounded">
 
 
714
  My Device
715
  </button>
716
  </nav>
 
721
  {/* Main Content */}
722
  <main className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
723
  {renderStepIndicator()}
724
+
725
  {currentStep === 1 && renderStep1()}
726
  {currentStep === 2 && renderStep2()}
727
  {currentStep === 3 && renderStep3()}
728
  </main>
729
  </div>
730
+ );
731
  }
frontend/src/pages/Playground.tsx CHANGED
@@ -707,7 +707,7 @@ export function Playground() {
707
  }}
708
  isGenerating={isLoading}
709
  stop={stopGeneration}
710
- className="flex-1"
711
  />
712
  </div>
713
 
 
707
  }}
708
  isGenerating={isLoading}
709
  stop={stopGeneration}
710
+ className="h-full w-full"
711
  />
712
  </div>
713
 
static/assets/index-170e6404.css ADDED
@@ -0,0 +1 @@
 
 
1
+ @import"https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;500;600;700;800;900&display=swap";*,:before,:after{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }*,:before,:after{box-sizing:border-box;border-width:0;border-style:solid;border-color:#e5e7eb}:before,:after{--tw-content: ""}html,:host{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:Poppins,sans-serif;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dl,dd,h1,h2,h3,h4,h5,h6,hr,figure,p,pre{margin:0}fieldset{margin:0;padding:0}legend{padding:0}ol,ul,menu{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}button,[role=button]{cursor:pointer}:disabled{cursor:default}img,svg,video,canvas,audio,iframe,embed,object{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}:root{--background: 0 0% 100%;--foreground: 222.2 84% 4.9%;--card: 0 0% 100%;--card-foreground: 222.2 84% 4.9%;--popover: 0 0% 100%;--popover-foreground: 222.2 84% 4.9%;--primary: 248 46% 40%;--primary-foreground: 210 40% 98%;--secondary: 210 40% 96%;--secondary-foreground: 222.2 84% 4.9%;--muted: 210 40% 96%;--muted-foreground: 215.4 16.3% 46.9%;--accent: 210 40% 96%;--accent-foreground: 222.2 84% 4.9%;--destructive: 0 84.2% 60.2%;--destructive-foreground: 210 40% 98%;--border: 214.3 31.8% 91.4%;--input: 214.3 31.8% 91.4%;--ring: 221.2 83.2% 53.3%;--radius: .5rem}*{border-color:hsl(var(--border))}body{background-color:hsl(var(--background));color:hsl(var(--foreground));font-family:Poppins,sans-serif}*{font-family:Poppins,sans-serif}.pointer-events-none{pointer-events:none}.invisible{visibility:hidden}.collapse{visibility:collapse}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.sticky{position:sticky}.inset-0{top:0;right:0;bottom:0;left:0}.-right-1{right:-.25rem}.-right-2{right:-.5rem}.-top-1{top:-.25rem}.-top-2{top:-.5rem}.left-2{left:.5rem}.left-4{left:1rem}.left-\[50\%\]{left:50%}.top-0{top:0}.top-1\/2,.top-\[50\%\]{top:50%}.z-40{z-index:40}.z-50{z-index:50}.m-0{margin:0}.m-4{margin:1rem}.-mx-1{margin-left:-.25rem;margin-right:-.25rem}.-mx-4{margin-left:-1rem;margin-right:-1rem}.mx-4{margin-left:1rem;margin-right:1rem}.mx-auto{margin-left:auto;margin-right:auto}.my-1{margin-top:.25rem;margin-bottom:.25rem}.mb-0{margin-bottom:0}.mb-1{margin-bottom:.25rem}.mb-12{margin-bottom:3rem}.mb-16{margin-bottom:4rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-6{margin-bottom:1.5rem}.mb-8{margin-bottom:2rem}.ml-1{margin-left:.25rem}.ml-2{margin-left:.5rem}.ml-8{margin-left:2rem}.ml-auto{margin-left:auto}.mr-1{margin-right:.25rem}.mr-2{margin-right:.5rem}.mt-0\.5{margin-top:.125rem}.mt-1{margin-top:.25rem}.mt-1\.5{margin-top:.375rem}.mt-2{margin-top:.5rem}.mt-6{margin-top:1.5rem}.mt-8{margin-top:2rem}.line-clamp-1{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:1}.line-clamp-2{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:2}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.grid{display:grid}.hidden{display:none}.h-0\.5{height:.125rem}.h-1\.5{height:.375rem}.h-10{height:2.5rem}.h-11{height:2.75rem}.h-12{height:3rem}.h-16{height:4rem}.h-2{height:.5rem}.h-20{height:5rem}.h-3{height:.75rem}.h-3\.5{height:.875rem}.h-4{height:1rem}.h-5{height:1.25rem}.h-6{height:1.5rem}.h-60{height:15rem}.h-8{height:2rem}.h-9{height:2.25rem}.h-\[250px\]{height:250px}.h-\[var\(--radix-select-trigger-height\)\]{height:var(--radix-select-trigger-height)}.h-auto{height:auto}.h-full{height:100%}.h-px{height:1px}.h-screen{height:100vh}.max-h-96{max-height:24rem}.max-h-screen{max-height:100vh}.min-h-0{min-height:0px}.min-h-\[600px\]{min-height:600px}.min-h-\[60px\]{min-height:60px}.min-h-\[80px\]{min-height:80px}.min-h-screen{min-height:100vh}.w-1\.5{width:.375rem}.w-10{width:2.5rem}.w-11{width:2.75rem}.w-12{width:3rem}.w-16{width:4rem}.w-2{width:.5rem}.w-20{width:5rem}.w-3{width:.75rem}.w-3\.5{width:.875rem}.w-32{width:8rem}.w-4{width:1rem}.w-5{width:1.25rem}.w-6{width:1.5rem}.w-64{width:16rem}.w-8{width:2rem}.w-80{width:20rem}.w-\[480px\]{width:480px}.w-auto{width:auto}.w-full{width:100%}.min-w-0{min-width:0px}.min-w-\[8rem\]{min-width:8rem}.min-w-\[var\(--radix-select-trigger-width\)\]{min-width:var(--radix-select-trigger-width)}.max-w-2xl{max-width:42rem}.max-w-3xl{max-width:48rem}.max-w-4xl{max-width:56rem}.max-w-6xl{max-width:72rem}.max-w-7xl{max-width:80rem}.max-w-\[75\%\]{max-width:75%}.max-w-\[80\%\]{max-width:80%}.max-w-lg{max-width:32rem}.max-w-none{max-width:none}.flex-1{flex:1 1 0%}.flex-shrink-0,.shrink-0{flex-shrink:0}.flex-grow,.grow{flex-grow:1}.-translate-y-1\/2{--tw-translate-y: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-\[-50\%\]{--tw-translate-x: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-y-\[-50\%\]{--tw-translate-y: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes bounce{0%,to{transform:translateY(-25%);animation-timing-function:cubic-bezier(.8,0,1,1)}50%{transform:none;animation-timing-function:cubic-bezier(0,0,.2,1)}}.animate-bounce{animation:bounce 1s infinite}@keyframes pulse{50%{opacity:.5}}.animate-pulse{animation:pulse 2s cubic-bezier(.4,0,.6,1) infinite}@keyframes spin{to{transform:rotate(360deg)}}.animate-spin{animation:spin 1s linear infinite}.cursor-default{cursor:default}.cursor-pointer{cursor:pointer}.touch-none{touch-action:none}.select-none{-webkit-user-select:none;-moz-user-select:none;user-select:none}.resize-none{resize:none}.list-decimal{list-style-type:decimal}.list-disc{list-style-type:disc}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.flex-wrap{flex-wrap:wrap}.items-start{align-items:flex-start}.items-center{align-items:center}.justify-start{justify-content:flex-start}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.gap-1{gap:.25rem}.gap-12{gap:3rem}.gap-2{gap:.5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-6{gap:1.5rem}.gap-8{gap:2rem}.space-x-1>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(.25rem * var(--tw-space-x-reverse));margin-left:calc(.25rem * calc(1 - var(--tw-space-x-reverse)))}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(.5rem * var(--tw-space-x-reverse));margin-left:calc(.5rem * calc(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(1rem * var(--tw-space-x-reverse));margin-left:calc(1rem * calc(1 - var(--tw-space-x-reverse)))}.space-x-6>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(1.5rem * var(--tw-space-x-reverse));margin-left:calc(1.5rem * calc(1 - var(--tw-space-x-reverse)))}.space-x-8>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(2rem * var(--tw-space-x-reverse));margin-left:calc(2rem * calc(1 - var(--tw-space-x-reverse)))}.space-y-1>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.25rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.25rem * var(--tw-space-y-reverse))}.space-y-1\.5>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.375rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.375rem * var(--tw-space-y-reverse))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.5rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem * var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.75rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem * var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(1rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem * var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(1.5rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem * var(--tw-space-y-reverse))}.space-y-8>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(2rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(2rem * var(--tw-space-y-reverse))}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.whitespace-nowrap{white-space:nowrap}.whitespace-pre-wrap{white-space:pre-wrap}.break-words{overflow-wrap:break-word}.break-all{word-break:break-all}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:1rem}.rounded-3xl{border-radius:1.5rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:var(--radius)}.rounded-md{border-radius:calc(var(--radius) - 2px)}.rounded-sm{border-radius:calc(var(--radius) - 4px)}.rounded-xl{border-radius:.75rem}.border{border-width:1px}.border-0{border-width:0px}.border-2{border-width:2px}.border-b{border-bottom-width:1px}.border-l{border-left-width:1px}.border-l-2{border-left-width:2px}.border-l-4{border-left-width:4px}.border-r{border-right-width:1px}.border-t{border-top-width:1px}.border-dashed{border-style:dashed}.border-blue-200{--tw-border-opacity: 1;border-color:rgb(191 219 254 / var(--tw-border-opacity, 1))}.border-blue-500{--tw-border-opacity: 1;border-color:rgb(59 130 246 / var(--tw-border-opacity, 1))}.border-gray-200{--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1))}.border-gray-300{--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1))}.border-green-200{--tw-border-opacity: 1;border-color:rgb(187 247 208 / var(--tw-border-opacity, 1))}.border-input{border-color:hsl(var(--input))}.border-muted{border-color:hsl(var(--muted))}.border-primary,.border-purple-600{--tw-border-opacity: 1;border-color:rgb(103 80 164 / var(--tw-border-opacity, 1))}.border-transparent{border-color:transparent}.bg-accent{background-color:hsl(var(--accent))}.bg-background{background-color:hsl(var(--background))}.bg-background\/50{background-color:hsl(var(--background) / .5)}.bg-background\/80{background-color:hsl(var(--background) / .8)}.bg-background\/95{background-color:hsl(var(--background) / .95)}.bg-blue-100{--tw-bg-opacity: 1;background-color:rgb(219 234 254 / var(--tw-bg-opacity, 1))}.bg-blue-400{--tw-bg-opacity: 1;background-color:rgb(96 165 250 / var(--tw-bg-opacity, 1))}.bg-blue-50{--tw-bg-opacity: 1;background-color:rgb(239 246 255 / var(--tw-bg-opacity, 1))}.bg-blue-500{--tw-bg-opacity: 1;background-color:rgb(59 130 246 / var(--tw-bg-opacity, 1))}.bg-blue-600{--tw-bg-opacity: 1;background-color:rgb(37 99 235 / var(--tw-bg-opacity, 1))}.bg-card{background-color:hsl(var(--card))}.bg-destructive{background-color:hsl(var(--destructive))}.bg-gray-100{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.bg-gray-300{--tw-bg-opacity: 1;background-color:rgb(209 213 219 / var(--tw-bg-opacity, 1))}.bg-gray-400{--tw-bg-opacity: 1;background-color:rgb(156 163 175 / var(--tw-bg-opacity, 1))}.bg-gray-50{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.bg-gray-50\/50{background-color:#f9fafb80}.bg-gray-600{--tw-bg-opacity: 1;background-color:rgb(75 85 99 / var(--tw-bg-opacity, 1))}.bg-green-100{--tw-bg-opacity: 1;background-color:rgb(220 252 231 / var(--tw-bg-opacity, 1))}.bg-green-50{--tw-bg-opacity: 1;background-color:rgb(240 253 244 / var(--tw-bg-opacity, 1))}.bg-green-500{--tw-bg-opacity: 1;background-color:rgb(34 197 94 / var(--tw-bg-opacity, 1))}.bg-muted{background-color:hsl(var(--muted))}.bg-popover{background-color:hsl(var(--popover))}.bg-primary{--tw-bg-opacity: 1;background-color:rgb(103 80 164 / var(--tw-bg-opacity, 1))}.bg-purple-100{--tw-bg-opacity: 1;background-color:rgb(233 227 247 / var(--tw-bg-opacity, 1))}.bg-purple-400\/20{background-color:#9e78d733}.bg-purple-600{--tw-bg-opacity: 1;background-color:rgb(103 80 164 / var(--tw-bg-opacity, 1))}.bg-secondary{background-color:hsl(var(--secondary))}.bg-white{--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.bg-yellow-50{--tw-bg-opacity: 1;background-color:rgb(254 252 232 / var(--tw-bg-opacity, 1))}.bg-yellow-500{--tw-bg-opacity: 1;background-color:rgb(234 179 8 / var(--tw-bg-opacity, 1))}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.from-blue-100{--tw-gradient-from: #dbeafe var(--tw-gradient-from-position);--tw-gradient-to: rgb(219 234 254 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-blue-500{--tw-gradient-from: #3b82f6 var(--tw-gradient-from-position);--tw-gradient-to: rgb(59 130 246 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-gray-200{--tw-gradient-from: #e5e7eb var(--tw-gradient-from-position);--tw-gradient-to: rgb(229 231 235 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-gray-50{--tw-gradient-from: #f9fafb var(--tw-gradient-from-position);--tw-gradient-to: rgb(249 250 251 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-green-100{--tw-gradient-from: #dcfce7 var(--tw-gradient-from-position);--tw-gradient-to: rgb(220 252 231 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-purple-100{--tw-gradient-from: #e9e3f7 var(--tw-gradient-from-position);--tw-gradient-to: rgb(233 227 247 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-purple-400\/30{--tw-gradient-from: rgb(158 120 215 / .3) var(--tw-gradient-from-position);--tw-gradient-to: rgb(158 120 215 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-purple-50{--tw-gradient-from: #f4f1fb var(--tw-gradient-from-position);--tw-gradient-to: rgb(244 241 251 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.via-blue-500\/30{--tw-gradient-to: rgb(59 130 246 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), rgb(59 130 246 / .3) var(--tw-gradient-via-position), var(--tw-gradient-to)}.to-blue-100{--tw-gradient-to: #dbeafe var(--tw-gradient-to-position)}.to-blue-50{--tw-gradient-to: #eff6ff var(--tw-gradient-to-position)}.to-gray-100{--tw-gradient-to: #f3f4f6 var(--tw-gradient-to-position)}.to-gray-300{--tw-gradient-to: #d1d5db var(--tw-gradient-to-position)}.to-pink-100{--tw-gradient-to: #fce7f3 var(--tw-gradient-to-position)}.to-pink-500\/30{--tw-gradient-to: rgb(236 72 153 / .3) var(--tw-gradient-to-position)}.to-purple-100{--tw-gradient-to: #e9e3f7 var(--tw-gradient-to-position)}.to-purple-600{--tw-gradient-to: #6750A4 var(--tw-gradient-to-position)}.fill-current{fill:currentColor}.object-cover{-o-object-fit:cover;object-fit:cover}.p-0{padding:0}.p-1{padding:.25rem}.p-12{padding:3rem}.p-2{padding:.5rem}.p-3{padding:.75rem}.p-4{padding:1rem}.p-6{padding:1.5rem}.p-8{padding:2rem}.px-1{padding-left:.25rem;padding-right:.25rem}.px-1\.5{padding-left:.375rem;padding-right:.375rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-2\.5{padding-left:.625rem;padding-right:.625rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.px-8{padding-left:2rem;padding-right:2rem}.py-0{padding-top:0;padding-bottom:0}.py-0\.5{padding-top:.125rem;padding-bottom:.125rem}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-1\.5{padding-top:.375rem;padding-bottom:.375rem}.py-12{padding-top:3rem;padding-bottom:3rem}.py-16{padding-top:4rem;padding-bottom:4rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pb-3{padding-bottom:.75rem}.pb-4{padding-bottom:1rem}.pb-6{padding-bottom:1.5rem}.pl-12{padding-left:3rem}.pl-3{padding-left:.75rem}.pl-4{padding-left:1rem}.pl-8{padding-left:2rem}.pr-2{padding-right:.5rem}.pr-4{padding-right:1rem}.pt-0{padding-top:0}.pt-2{padding-top:.5rem}.pt-4{padding-top:1rem}.pt-6{padding-top:1.5rem}.text-left{text-align:left}.text-center{text-align:center}.text-right{text-align:right}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-\[10px\]{font-size:10px}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.capitalize{text-transform:capitalize}.italic{font-style:italic}.leading-none{line-height:1}.leading-relaxed{line-height:1.625}.leading-tight{line-height:1.25}.tracking-tight{letter-spacing:-.025em}.tracking-wide{letter-spacing:.025em}.text-accent-foreground{color:hsl(var(--accent-foreground))}.text-black{--tw-text-opacity: 1;color:rgb(0 0 0 / var(--tw-text-opacity, 1))}.text-blue-500{--tw-text-opacity: 1;color:rgb(59 130 246 / var(--tw-text-opacity, 1))}.text-blue-600{--tw-text-opacity: 1;color:rgb(37 99 235 / var(--tw-text-opacity, 1))}.text-blue-700{--tw-text-opacity: 1;color:rgb(29 78 216 / var(--tw-text-opacity, 1))}.text-blue-800{--tw-text-opacity: 1;color:rgb(30 64 175 / var(--tw-text-opacity, 1))}.text-blue-900{--tw-text-opacity: 1;color:rgb(30 58 138 / var(--tw-text-opacity, 1))}.text-card-foreground{color:hsl(var(--card-foreground))}.text-destructive{color:hsl(var(--destructive))}.text-destructive-foreground{color:hsl(var(--destructive-foreground))}.text-foreground{color:hsl(var(--foreground))}.text-gray-400{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.text-gray-500{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.text-gray-600{--tw-text-opacity: 1;color:rgb(75 85 99 / var(--tw-text-opacity, 1))}.text-gray-700{--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.text-gray-800{--tw-text-opacity: 1;color:rgb(31 41 55 / var(--tw-text-opacity, 1))}.text-gray-900{--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.text-green-500{--tw-text-opacity: 1;color:rgb(34 197 94 / var(--tw-text-opacity, 1))}.text-green-600{--tw-text-opacity: 1;color:rgb(22 163 74 / var(--tw-text-opacity, 1))}.text-green-700{--tw-text-opacity: 1;color:rgb(21 128 61 / var(--tw-text-opacity, 1))}.text-green-800{--tw-text-opacity: 1;color:rgb(22 101 52 / var(--tw-text-opacity, 1))}.text-muted-foreground{color:hsl(var(--muted-foreground))}.text-pink-500{--tw-text-opacity: 1;color:rgb(236 72 153 / var(--tw-text-opacity, 1))}.text-popover-foreground{color:hsl(var(--popover-foreground))}.text-primary{--tw-text-opacity: 1;color:rgb(103 80 164 / var(--tw-text-opacity, 1))}.text-primary-foreground{color:hsl(var(--primary-foreground))}.text-purple-500{--tw-text-opacity: 1;color:rgb(132 84 199 / var(--tw-text-opacity, 1))}.text-purple-600{--tw-text-opacity: 1;color:rgb(103 80 164 / var(--tw-text-opacity, 1))}.text-purple-700{--tw-text-opacity: 1;color:rgb(90 61 143 / var(--tw-text-opacity, 1))}.text-red-500{--tw-text-opacity: 1;color:rgb(239 68 68 / var(--tw-text-opacity, 1))}.text-secondary-foreground{color:hsl(var(--secondary-foreground))}.text-white{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.text-yellow-500{--tw-text-opacity: 1;color:rgb(234 179 8 / var(--tw-text-opacity, 1))}.text-yellow-600{--tw-text-opacity: 1;color:rgb(202 138 4 / var(--tw-text-opacity, 1))}.underline-offset-4{text-underline-offset:4px}.opacity-50{opacity:.5}.opacity-70{opacity:.7}.shadow-lg{--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / .1), 0 4px 6px -4px rgb(0 0 0 / .1);--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-md{--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / .1), 0 2px 4px -2px rgb(0 0 0 / .1);--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-sm{--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-xl{--tw-shadow: 0 20px 25px -5px rgb(0 0 0 / .1), 0 8px 10px -6px rgb(0 0 0 / .1);--tw-shadow-colored: 0 20px 25px -5px var(--tw-shadow-color), 0 8px 10px -6px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.outline-none{outline:2px solid transparent;outline-offset:2px}.outline{outline-style:solid}.ring-0{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.ring-offset-background{--tw-ring-offset-color: hsl(var(--background))}.blur{--tw-blur: blur(8px);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur{--tw-backdrop-blur: blur(8px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-lg{--tw-backdrop-blur: blur(16px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-sm{--tw-backdrop-blur: blur(4px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-filter{-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-shadow{transition-property:box-shadow;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-transform{transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.ease-in-out{transition-timing-function:cubic-bezier(.4,0,.2,1)}@keyframes enter{0%{opacity:var(--tw-enter-opacity, 1);transform:translate3d(var(--tw-enter-translate-x, 0),var(--tw-enter-translate-y, 0),0) scale3d(var(--tw-enter-scale, 1),var(--tw-enter-scale, 1),var(--tw-enter-scale, 1)) rotate(var(--tw-enter-rotate, 0))}}@keyframes exit{to{opacity:var(--tw-exit-opacity, 1);transform:translate3d(var(--tw-exit-translate-x, 0),var(--tw-exit-translate-y, 0),0) scale3d(var(--tw-exit-scale, 1),var(--tw-exit-scale, 1),var(--tw-exit-scale, 1)) rotate(var(--tw-exit-rotate, 0))}}.duration-200{animation-duration:.2s}.duration-300{animation-duration:.3s}.ease-in-out{animation-timing-function:cubic-bezier(.4,0,.2,1)}.file\:border-0::file-selector-button{border-width:0px}.file\:bg-transparent::file-selector-button{background-color:transparent}.file\:text-sm::file-selector-button{font-size:.875rem;line-height:1.25rem}.file\:font-medium::file-selector-button{font-weight:500}.placeholder\:text-muted-foreground::-moz-placeholder{color:hsl(var(--muted-foreground))}.placeholder\:text-muted-foreground::placeholder{color:hsl(var(--muted-foreground))}.last\:mb-0:last-child{margin-bottom:0}.hover\:-translate-y-1:hover{--tw-translate-y: -.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.hover\:border-blue-200:hover{--tw-border-opacity: 1;border-color:rgb(191 219 254 / var(--tw-border-opacity, 1))}.hover\:border-gray-300:hover{--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1))}.hover\:bg-accent:hover{background-color:hsl(var(--accent))}.hover\:bg-destructive\/80:hover{background-color:hsl(var(--destructive) / .8)}.hover\:bg-destructive\/90:hover{background-color:hsl(var(--destructive) / .9)}.hover\:bg-gray-100:hover{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-200:hover{--tw-bg-opacity: 1;background-color:rgb(229 231 235 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-50:hover{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.hover\:bg-primary\/80:hover{background-color:#6750a4cc}.hover\:bg-primary\/90:hover{background-color:#6750a4e6}.hover\:bg-purple-50:hover{--tw-bg-opacity: 1;background-color:rgb(244 241 251 / var(--tw-bg-opacity, 1))}.hover\:bg-purple-700:hover{--tw-bg-opacity: 1;background-color:rgb(90 61 143 / var(--tw-bg-opacity, 1))}.hover\:bg-red-50:hover{--tw-bg-opacity: 1;background-color:rgb(254 242 242 / var(--tw-bg-opacity, 1))}.hover\:bg-secondary\/80:hover{background-color:hsl(var(--secondary) / .8)}.hover\:text-accent-foreground:hover{color:hsl(var(--accent-foreground))}.hover\:text-destructive:hover{color:hsl(var(--destructive))}.hover\:text-foreground:hover{color:hsl(var(--foreground))}.hover\:text-gray-700:hover{--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.hover\:text-gray-900:hover{--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.hover\:text-purple-600:hover{--tw-text-opacity: 1;color:rgb(103 80 164 / var(--tw-text-opacity, 1))}.hover\:text-red-500:hover{--tw-text-opacity: 1;color:rgb(239 68 68 / var(--tw-text-opacity, 1))}.hover\:text-red-600:hover{--tw-text-opacity: 1;color:rgb(220 38 38 / var(--tw-text-opacity, 1))}.hover\:underline:hover{text-decoration-line:underline}.hover\:shadow-md:hover{--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / .1), 0 2px 4px -2px rgb(0 0 0 / .1);--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.hover\:shadow-xl:hover{--tw-shadow: 0 20px 25px -5px rgb(0 0 0 / .1), 0 8px 10px -6px rgb(0 0 0 / .1);--tw-shadow-colored: 0 20px 25px -5px var(--tw-shadow-color), 0 8px 10px -6px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.focus\:border-blue-400:focus{--tw-border-opacity: 1;border-color:rgb(96 165 250 / var(--tw-border-opacity, 1))}.focus\:border-blue-500:focus{--tw-border-opacity: 1;border-color:rgb(59 130 246 / var(--tw-border-opacity, 1))}.focus\:bg-accent:focus{background-color:hsl(var(--accent))}.focus\:text-accent-foreground:focus{color:hsl(var(--accent-foreground))}.focus\:outline-none:focus{outline:2px solid transparent;outline-offset:2px}.focus\:ring-2:focus{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.focus\:ring-blue-100:focus{--tw-ring-opacity: 1;--tw-ring-color: rgb(219 234 254 / var(--tw-ring-opacity, 1))}.focus\:ring-ring:focus{--tw-ring-color: hsl(var(--ring))}.focus\:ring-offset-2:focus{--tw-ring-offset-width: 2px}.focus-visible\:outline-none:focus-visible{outline:2px solid transparent;outline-offset:2px}.focus-visible\:ring-2:focus-visible{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.focus-visible\:ring-ring:focus-visible{--tw-ring-color: hsl(var(--ring))}.focus-visible\:ring-offset-2:focus-visible{--tw-ring-offset-width: 2px}.focus-visible\:ring-offset-background:focus-visible{--tw-ring-offset-color: hsl(var(--background))}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-50:disabled{opacity:.5}.group:hover .group-hover\:bg-blue-600{--tw-bg-opacity: 1;background-color:rgb(37 99 235 / var(--tw-bg-opacity, 1))}.group:hover .group-hover\:text-blue-600{--tw-text-opacity: 1;color:rgb(37 99 235 / var(--tw-text-opacity, 1))}.group:hover .group-hover\:text-white{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.peer:disabled~.peer-disabled\:cursor-not-allowed{cursor:not-allowed}.peer:disabled~.peer-disabled\:opacity-70{opacity:.7}.data-\[disabled\]\:pointer-events-none[data-disabled]{pointer-events:none}.data-\[side\=bottom\]\:translate-y-1[data-side=bottom]{--tw-translate-y: .25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.data-\[side\=left\]\:-translate-x-1[data-side=left]{--tw-translate-x: -.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.data-\[side\=right\]\:translate-x-1[data-side=right]{--tw-translate-x: .25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.data-\[side\=top\]\:-translate-y-1[data-side=top]{--tw-translate-y: -.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.data-\[state\=checked\]\:translate-x-5[data-state=checked]{--tw-translate-x: 1.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.data-\[state\=unchecked\]\:translate-x-0[data-state=unchecked]{--tw-translate-x: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.data-\[state\=active\]\:bg-background[data-state=active]{background-color:hsl(var(--background))}.data-\[state\=checked\]\:bg-primary[data-state=checked]{--tw-bg-opacity: 1;background-color:rgb(103 80 164 / var(--tw-bg-opacity, 1))}.data-\[state\=unchecked\]\:bg-input[data-state=unchecked]{background-color:hsl(var(--input))}.data-\[state\=active\]\:text-foreground[data-state=active]{color:hsl(var(--foreground))}.data-\[disabled\]\:opacity-50[data-disabled]{opacity:.5}.data-\[state\=active\]\:shadow-sm[data-state=active]{--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.data-\[state\=open\]\:animate-in[data-state=open]{animation-name:enter;animation-duration:.15s;--tw-enter-opacity: initial;--tw-enter-scale: initial;--tw-enter-rotate: initial;--tw-enter-translate-x: initial;--tw-enter-translate-y: initial}.data-\[state\=closed\]\:animate-out[data-state=closed]{animation-name:exit;animation-duration:.15s;--tw-exit-opacity: initial;--tw-exit-scale: initial;--tw-exit-rotate: initial;--tw-exit-translate-x: initial;--tw-exit-translate-y: initial}.data-\[state\=closed\]\:fade-out-0[data-state=closed]{--tw-exit-opacity: 0}.data-\[state\=open\]\:fade-in-0[data-state=open]{--tw-enter-opacity: 0}.data-\[state\=closed\]\:zoom-out-95[data-state=closed]{--tw-exit-scale: .95}.data-\[state\=open\]\:zoom-in-95[data-state=open]{--tw-enter-scale: .95}.data-\[side\=bottom\]\:slide-in-from-top-2[data-side=bottom]{--tw-enter-translate-y: -.5rem}.data-\[side\=left\]\:slide-in-from-right-2[data-side=left]{--tw-enter-translate-x: .5rem}.data-\[side\=right\]\:slide-in-from-left-2[data-side=right]{--tw-enter-translate-x: -.5rem}.data-\[side\=top\]\:slide-in-from-bottom-2[data-side=top]{--tw-enter-translate-y: .5rem}.data-\[state\=closed\]\:slide-out-to-left-1\/2[data-state=closed]{--tw-exit-translate-x: -50%}.data-\[state\=closed\]\:slide-out-to-top-\[48\%\][data-state=closed]{--tw-exit-translate-y: -48%}.data-\[state\=open\]\:slide-in-from-left-1\/2[data-state=open]{--tw-enter-translate-x: -50%}.data-\[state\=open\]\:slide-in-from-top-\[48\%\][data-state=open]{--tw-enter-translate-y: -48%}@supports (backdrop-filter: var(--tw)){.supports-\[backdrop-filter\]\:bg-background\/60{background-color:hsl(var(--background) / .6)}}@media (min-width: 640px){.sm\:mt-0{margin-top:0}.sm\:inline{display:inline}.sm\:flex-row{flex-direction:row}.sm\:justify-end{justify-content:flex-end}.sm\:space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(.5rem * var(--tw-space-x-reverse));margin-left:calc(.5rem * calc(1 - var(--tw-space-x-reverse)))}.sm\:rounded-lg{border-radius:var(--radius)}.sm\:px-6{padding-left:1.5rem;padding-right:1.5rem}.sm\:text-left{text-align:left}}@media (min-width: 768px){.md\:flex{display:flex}.md\:hidden{display:none}.md\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:text-5xl{font-size:3rem;line-height:1}.md\:text-6xl{font-size:3.75rem;line-height:1}.md\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width: 1024px){.lg\:col-span-1{grid-column:span 1 / span 1}.lg\:col-span-3{grid-column:span 3 / span 3}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.lg\:px-8{padding-left:2rem;padding-right:2rem}.lg\:text-7xl{font-size:4.5rem;line-height:1}}@media (min-width: 1280px){.xl\:w-\[520px\]{width:520px}}.\[\&\>span\]\:line-clamp-1>span{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:1}
static/assets/index-9355198e.js ADDED
The diff for this file is too large to render. See raw diff
 
static/assets/index-9355198e.js.map ADDED
The diff for this file is too large to render. See raw diff
 
static/index.html CHANGED
@@ -5,8 +5,8 @@
5
  <link rel="icon" type="image/svg+xml" href="/vite.svg" />
6
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
  <title>Edge LLM</title>
8
- <script type="module" crossorigin src="/assets/index-99602292.js"></script>
9
- <link rel="stylesheet" href="/assets/index-851c8378.css">
10
  </head>
11
  <body>
12
  <div id="root"></div>
 
5
  <link rel="icon" type="image/svg+xml" href="/vite.svg" />
6
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
  <title>Edge LLM</title>
8
+ <script type="module" crossorigin src="/assets/index-9355198e.js"></script>
9
+ <link rel="stylesheet" href="/assets/index-170e6404.css">
10
  </head>
11
  <body>
12
  <div id="root"></div>