@djangocfg/ui-tools 2.1.384 → 2.1.387
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +25 -11
- package/dist/ChatRoot-4KM2JMGA.mjs +6 -0
- package/dist/{ChatRoot-JVR3M3H2.mjs.map → ChatRoot-4KM2JMGA.mjs.map} +1 -1
- package/dist/ChatRoot-OILWMMZ6.cjs +15 -0
- package/dist/{ChatRoot-LXIUBOXF.cjs.map → ChatRoot-OILWMMZ6.cjs.map} +1 -1
- package/dist/DictationField-AS2F33WI.cjs +13 -0
- package/dist/{DictationField-U25MEYAL.mjs.map → DictationField-AS2F33WI.cjs.map} +1 -1
- package/dist/DictationField-WPONUCYE.mjs +4 -0
- package/dist/{DictationField-XWR5VOID.cjs.map → DictationField-WPONUCYE.mjs.map} +1 -1
- package/dist/MapContainer-AKIPABJK.mjs +4 -0
- package/dist/MapContainer-AKIPABJK.mjs.map +1 -0
- package/dist/MapContainer-STVDMC36.cjs +17 -0
- package/dist/MapContainer-STVDMC36.cjs.map +1 -0
- package/dist/{MapContainer-76YL2JXL.cjs → chunk-5D2OCOPQ.cjs} +3 -2
- package/dist/chunk-5D2OCOPQ.cjs.map +1 -0
- package/dist/{MapContainer-7HXBI3OH.mjs → chunk-7CWGZPO3.mjs} +3 -3
- package/dist/chunk-7CWGZPO3.mjs.map +1 -0
- package/dist/{chunk-4PFW7MIJ.cjs → chunk-ADEN3UA4.cjs} +60 -5
- package/dist/chunk-ADEN3UA4.cjs.map +1 -0
- package/dist/chunk-BVESQTBM.mjs +1439 -0
- package/dist/chunk-BVESQTBM.mjs.map +1 -0
- package/dist/{chunk-PEKBT75W.mjs → chunk-DMX7W4XZ.mjs} +53 -1387
- package/dist/chunk-DMX7W4XZ.mjs.map +1 -0
- package/dist/chunk-HNIMIIFR.mjs +1361 -0
- package/dist/chunk-HNIMIIFR.mjs.map +1 -0
- package/dist/chunk-L25HA3TM.cjs +1478 -0
- package/dist/chunk-L25HA3TM.cjs.map +1 -0
- package/dist/{chunk-HPK3EWBF.cjs → chunk-TBSHZO5R.cjs} +50 -1409
- package/dist/chunk-TBSHZO5R.cjs.map +1 -0
- package/dist/chunk-TSNRU3UO.cjs +1387 -0
- package/dist/chunk-TSNRU3UO.cjs.map +1 -0
- package/dist/{chunk-C2YN6WEO.mjs → chunk-UNCS5V5F.mjs} +61 -7
- package/dist/chunk-UNCS5V5F.mjs.map +1 -0
- package/dist/index.cjs +1236 -1768
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +780 -780
- package/dist/index.d.ts +780 -780
- package/dist/index.mjs +853 -1423
- package/dist/index.mjs.map +1 -1
- package/dist/launcher-5WYPDPEP.mjs +7 -0
- package/dist/launcher-5WYPDPEP.mjs.map +1 -0
- package/dist/launcher-FCI3LTDY.css +7 -0
- package/dist/launcher-FCI3LTDY.css.map +1 -0
- package/dist/launcher-QAOG2NUI.cjs +60 -0
- package/dist/launcher-QAOG2NUI.cjs.map +1 -0
- package/package.json +23 -18
- package/src/tools/AudioPlayer/lazy.tsx +100 -0
- package/src/tools/Chat/README.md +85 -1
- package/src/tools/Chat/context/ChatProvider.tsx +42 -0
- package/src/tools/Chat/lazy.tsx +213 -1
- package/src/tools/CodeEditor/lazy.tsx +70 -0
- package/src/tools/Map/lazy.tsx +38 -1
- package/src/tools/MarkdownEditor/lazy.tsx +42 -0
- package/src/tools/SpeechRecognition/README.md +48 -0
- package/src/tools/SpeechRecognition/core/index.ts +6 -1
- package/src/tools/SpeechRecognition/core/logger.ts +107 -1
- package/src/tools/SpeechRecognition/hooks/useSpeechRecognition.ts +15 -4
- package/src/tools/SpeechRecognition/index.ts +9 -0
- package/src/tools/SpeechRecognition/widgets/VoiceComposerSlot.tsx +37 -2
- package/dist/ChatRoot-JVR3M3H2.mjs +0 -5
- package/dist/ChatRoot-LXIUBOXF.cjs +0 -14
- package/dist/DictationField-U25MEYAL.mjs +0 -4
- package/dist/DictationField-XWR5VOID.cjs +0 -13
- package/dist/MapContainer-76YL2JXL.cjs.map +0 -1
- package/dist/MapContainer-7HXBI3OH.mjs.map +0 -1
- package/dist/chunk-4PFW7MIJ.cjs.map +0 -1
- package/dist/chunk-C2YN6WEO.mjs.map +0 -1
- package/dist/chunk-HPK3EWBF.cjs.map +0 -1
- package/dist/chunk-PEKBT75W.mjs.map +0 -1
package/README.md
CHANGED
|
@@ -33,7 +33,7 @@ Sixteen tools, each one lazy-loaded so it doesn't ship until used. Bundle size i
|
|
|
33
33
|
| `Chat` | ~150KB | Streaming chat (SSE + tool calls + attachments). [README](src/tools/Chat/README.md) |
|
|
34
34
|
| `SpeechRecognition` | ~40KB | Mic capture + STT with pluggable engines (Web Speech / HTTP / WS). [README](src/tools/SpeechRecognition/README.md) |
|
|
35
35
|
| `VideoPlayer` | ~150KB | Vidstack-based pro player |
|
|
36
|
-
| `MarkdownMessage` | ~120KB | Read-only chat-tuned markdown. [README](src/components/markdown/MarkdownMessage/README.md) |
|
|
36
|
+
| `MarkdownMessage` | ~120KB | Read-only chat-tuned markdown. **SSR-safe** — use as a Client Component, the result is server-rendered. [README](src/components/markdown/MarkdownMessage/README.md) |
|
|
37
37
|
| `JsonTree` | ~100KB | JSON visualization (full/compact/inline modes) |
|
|
38
38
|
| `AudioPlayer` | ~80KB | WebView-safe waveform player |
|
|
39
39
|
| `Gallery` | ~50KB | Image/video gallery + lightbox |
|
|
@@ -57,19 +57,33 @@ For bundle splitting, import from the tool's subpath — only that tool ships:
|
|
|
57
57
|
```tsx
|
|
58
58
|
import { LazyJsonTree } from '@djangocfg/ui-tools/json-tree';
|
|
59
59
|
import { Gallery } from '@djangocfg/ui-tools/gallery';
|
|
60
|
-
import {
|
|
61
|
-
import {
|
|
62
|
-
import {
|
|
63
|
-
import {
|
|
60
|
+
import { LazyMapContainer, MapMarker } from '@djangocfg/ui-tools/map';
|
|
61
|
+
import { MarkdownMessage } from '@djangocfg/ui-tools/markdown-message';
|
|
62
|
+
import { LazyEditor, useMonaco } from '@djangocfg/ui-tools/code-editor';
|
|
63
|
+
import { LazyChat, createPydanticAIChatTransport } from '@djangocfg/ui-tools/chat';
|
|
64
|
+
import { LazyMarkdownEditor, mentionPresets } from '@djangocfg/ui-tools/markdown-editor';
|
|
65
|
+
import { LazyPlayer, PlayerProvider } from '@djangocfg/ui-tools/audio-player';
|
|
64
66
|
// …same pattern for every tool
|
|
65
67
|
```
|
|
66
68
|
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
69
|
+
Subpaths come in three flavors:
|
|
70
|
+
|
|
71
|
+
- **Lazy + light surface.** The heavy component is exported only as a `Lazy*` wrapper; types, hooks, transports, slot components, and pure helpers are exported synchronously. This is the standard pattern.
|
|
72
|
+
- **Lazy primitives co-exist with light primitives.** `./map` exports `LazyMapContainer` (heavy MapLibre GL) plus `MapMarker` / `MapPopup` etc. — the markers are thin `react-map-gl` wrappers that don't import the heavy lib at module scope.
|
|
73
|
+
- **Synchronous, SSR-safe.** `./markdown-message` is intentionally not lazy — the component is `'use client'` but produces plain HTML, so Next renders it on the server.
|
|
74
|
+
|
|
75
|
+
| Subpath | Ships | Notes |
|
|
76
|
+
|---------|-------|-------|
|
|
77
|
+
| `@djangocfg/ui-tools` | All tools (lazy) | App-level barrel. Treeshakes per-tool when used via subpaths. |
|
|
78
|
+
| `@djangocfg/ui-tools/chat` | `LazyChat`, `LazyChatLauncher`, transports (`createPydanticAIChatTransport`, `createHttpTransport`, …), hooks (`useChat`, `useChatAudio`, …), reducer/core/types/style tokens | Heavy `ChatRoot` + `ChatLauncher` only load via the `Lazy*` wrappers. Custom layouts that need raw `MessageBubble` / `Composer` should import from the root barrel. |
|
|
79
|
+
| `@djangocfg/ui-tools/code-editor` | `LazyEditor`, `LazyDiffEditor`, `useMonaco`, `useEditor`, `useLanguage`, `useEditorTheme`, `EditorProvider`, types | Monaco (~550 KB) loads only when an editor mounts. `useMonaco` does its own dynamic import. |
|
|
80
|
+
| `@djangocfg/ui-tools/audio-player` | `LazyPlayer`, `PlayerProvider`, selector hooks, slot components (`Cover`, `Title`, `PlayButton`, `Waveform`, …), peaks-cache helpers, store, types | Slot components are presentational — safe synchronous re-exports. They only do anything inside a `<LazyPlayer>` provider. |
|
|
81
|
+
| `@djangocfg/ui-tools/markdown-editor` | `LazyMarkdownEditor`, `mentionPresets`, types | TipTap + ProseMirror (~200 KB) only loads via the lazy wrapper. |
|
|
82
|
+
| `@djangocfg/ui-tools/map` | `LazyMapContainer`, `LazyMapView`, plus light primitives (`MapMarker`, `MapPopup`, `MapCluster`, `MapSource`, `MapLayer`, `MapControls`, `MapProvider`, types) | The heavy MapLibre GL chunk (~800 KB) only loads when `LazyMapContainer` actually mounts. Markers and popups are thin `react-map-gl` wrappers — exported synchronously. |
|
|
83
|
+
| `@djangocfg/ui-tools/markdown-message` | `MarkdownMessage`, `ChatMessageRow`, `ActionRow`, `extractTextFromChildren`, types | **SSR-safe.** The component itself is `'use client'`, but rendering produces plain HTML — Next.js will pre-render it on the server when imported from a Client Component. Use this when you want the markdown renderer without dragging in the full chat. |
|
|
84
|
+
| `@djangocfg/ui-tools/<tool-name>` | One tool | `mermaid`, `speech-recognition`, `json-tree`, `pretty-code`, `openapi-viewer`, `json-form`, `lottie-player`, `video-player`, `image-viewer`, `cron-scheduler`, `gallery`, `tour`, `tree`, `file-icon`, `upload` |
|
|
85
|
+
| `@djangocfg/ui-tools/styles` | Tailwind source CSS | |
|
|
86
|
+
| `@djangocfg/ui-tools/dist.css` | Pre-compiled CSS | |
|
|
73
87
|
|
|
74
88
|
---
|
|
75
89
|
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":[],"names":[],"mappings":"","file":"ChatRoot-
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":"","file":"ChatRoot-4KM2JMGA.mjs"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var chunkTSNRU3UO_cjs = require('./chunk-TSNRU3UO.cjs');
|
|
4
|
+
require('./chunk-TBSHZO5R.cjs');
|
|
5
|
+
require('./chunk-FIRK5CEH.cjs');
|
|
6
|
+
require('./chunk-OLISEQHS.cjs');
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
Object.defineProperty(exports, "ChatRoot", {
|
|
11
|
+
enumerable: true,
|
|
12
|
+
get: function () { return chunkTSNRU3UO_cjs.ChatRoot; }
|
|
13
|
+
});
|
|
14
|
+
//# sourceMappingURL=ChatRoot-OILWMMZ6.cjs.map
|
|
15
|
+
//# sourceMappingURL=ChatRoot-OILWMMZ6.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":[],"names":[],"mappings":"","file":"ChatRoot-
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":"","file":"ChatRoot-OILWMMZ6.cjs"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var chunkADEN3UA4_cjs = require('./chunk-ADEN3UA4.cjs');
|
|
4
|
+
require('./chunk-OLISEQHS.cjs');
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
Object.defineProperty(exports, "DictationField", {
|
|
9
|
+
enumerable: true,
|
|
10
|
+
get: function () { return chunkADEN3UA4_cjs.DictationField; }
|
|
11
|
+
});
|
|
12
|
+
//# sourceMappingURL=DictationField-AS2F33WI.cjs.map
|
|
13
|
+
//# sourceMappingURL=DictationField-AS2F33WI.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":[],"names":[],"mappings":"","file":"DictationField-
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":"","file":"DictationField-AS2F33WI.cjs"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":[],"names":[],"mappings":"","file":"DictationField-
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":"","file":"DictationField-WPONUCYE.mjs"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":"","file":"MapContainer-AKIPABJK.mjs"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var chunk5D2OCOPQ_cjs = require('./chunk-5D2OCOPQ.cjs');
|
|
4
|
+
require('./chunk-OLISEQHS.cjs');
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
Object.defineProperty(exports, "MapContainer", {
|
|
9
|
+
enumerable: true,
|
|
10
|
+
get: function () { return chunk5D2OCOPQ_cjs.MapContainer; }
|
|
11
|
+
});
|
|
12
|
+
Object.defineProperty(exports, "MapView", {
|
|
13
|
+
enumerable: true,
|
|
14
|
+
get: function () { return chunk5D2OCOPQ_cjs.MapView; }
|
|
15
|
+
});
|
|
16
|
+
//# sourceMappingURL=MapContainer-STVDMC36.cjs.map
|
|
17
|
+
//# sourceMappingURL=MapContainer-STVDMC36.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":"","file":"MapContainer-STVDMC36.cjs"}
|
|
@@ -217,5 +217,6 @@ chunkOLISEQHS_cjs.__name(MapView, "MapView");
|
|
|
217
217
|
|
|
218
218
|
exports.MapContainer = MapContainer;
|
|
219
219
|
exports.MapView = MapView;
|
|
220
|
-
|
|
221
|
-
//# sourceMappingURL=
|
|
220
|
+
exports.useMapContext = useMapContext;
|
|
221
|
+
//# sourceMappingURL=chunk-5D2OCOPQ.cjs.map
|
|
222
|
+
//# sourceMappingURL=chunk-5D2OCOPQ.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/tools/Map/context/MapContext.tsx","../src/tools/Map/styles/index.ts","../src/tools/Map/components/MapContainer.tsx"],"names":["createContext","useRef","useState","useCallback","useMemo","jsx","__name","useContext","useEffect","jsxs","Fragment","Map","cn","RotateCcw","ExternalLink"],"mappings":"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","file":"chunk-5D2OCOPQ.cjs","sourcesContent":["'use client'\n\nimport {\n createContext,\n useContext,\n useState,\n useRef,\n useMemo,\n useCallback,\n type ReactNode,\n} from 'react'\nimport type { MapRef } from 'react-map-gl/maplibre'\nimport type { MapContextValue, MapViewport, MarkerData } from '../types'\n\nconst MapContext = createContext<MapContextValue | null>(null)\n\nexport interface MapProviderProps {\n children: ReactNode\n initialViewport?: Partial<MapViewport>\n}\n\nconst DEFAULT_VIEWPORT: MapViewport = {\n longitude: 115.1889,\n latitude: -8.4095,\n zoom: 10,\n bearing: 0,\n pitch: 0,\n}\n\nexport function MapProvider({ children, initialViewport }: MapProviderProps) {\n const mapRef = useRef<MapRef | null>(null)\n const initialViewportRef = useRef<MapViewport>({\n ...DEFAULT_VIEWPORT,\n ...initialViewport,\n })\n const [viewport, setViewportState] = useState<MapViewport>(initialViewportRef.current)\n const [markers, setMarkers] = useState<MarkerData[]>([])\n const [selectedMarker, setSelectedMarker] = useState<MarkerData | null>(null)\n const [hoveredFeature, setHoveredFeature] = useState<GeoJSON.Feature | null>(null)\n const [isLoaded, setIsLoaded] = useState(false)\n\n const setViewport = useCallback((newViewport: Partial<MapViewport>) => {\n setViewportState((prev) => ({ ...prev, ...newViewport }))\n }, [])\n\n const resetToInitial = useCallback(() => {\n const map = mapRef.current\n if (map) {\n map.flyTo({\n center: [initialViewportRef.current.longitude, initialViewportRef.current.latitude],\n zoom: initialViewportRef.current.zoom,\n bearing: initialViewportRef.current.bearing ?? 0,\n pitch: initialViewportRef.current.pitch ?? 0,\n duration: 1000,\n })\n }\n }, [])\n\n const value = useMemo<MapContextValue>(\n () => ({\n mapRef,\n viewport,\n setViewport,\n initialViewport: initialViewportRef.current,\n resetToInitial,\n markers,\n setMarkers,\n selectedMarker,\n setSelectedMarker,\n hoveredFeature,\n setHoveredFeature,\n isLoaded,\n setIsLoaded,\n }),\n [viewport, setViewport, resetToInitial, markers, selectedMarker, hoveredFeature, isLoaded]\n )\n\n return <MapContext.Provider value={value}>{children}</MapContext.Provider>\n}\n\nexport function useMapContext(): MapContextValue {\n const context = useContext(MapContext)\n if (!context) {\n throw new Error('useMapContext must be used within a MapProvider')\n }\n return context\n}\n\nexport { MapContext }\n","export const MAP_STYLES = {\n light: 'https://basemaps.cartocdn.com/gl/positron-gl-style/style.json',\n dark: 'https://basemaps.cartocdn.com/gl/dark-matter-gl-style/style.json',\n streets: 'https://basemaps.cartocdn.com/gl/voyager-gl-style/style.json',\n satellite: 'https://api.maptiler.com/maps/satellite/style.json',\n} as const\n\nexport type MapStyleKey = keyof typeof MAP_STYLES\n\nexport function getMapStyle(key: MapStyleKey | string): string {\n if (key in MAP_STYLES) {\n return MAP_STYLES[key as MapStyleKey]\n }\n return key\n}\n","'use client'\n\nimport { useCallback, useEffect, useRef, type ReactNode } from 'react'\nimport Map, { type ViewStateChangeEvent } from 'react-map-gl/maplibre'\nimport { ExternalLink, RotateCcw } from 'lucide-react'\nimport { cn } from '@djangocfg/ui-core/lib'\nimport 'maplibre-gl/dist/maplibre-gl.css'\n\nimport { MapProvider, useMapContext } from '../context'\nimport { MAP_STYLES } from '../styles'\nimport type { MapViewport, MapStyleKey } from '../types'\n\nexport interface MapContainerProps {\n children?: ReactNode\n initialViewport?: Partial<MapViewport>\n mapStyle?: MapStyleKey | string\n interactiveLayerIds?: string[]\n className?: string\n style?: React.CSSProperties\n cursor?: string\n attributionControl?: boolean\n reuseMaps?: boolean\n /** URL to open in external maps app (shows \"Open in Maps\" button if provided) */\n openInMapsUrl?: string\n /** Label for the open in maps button */\n openInMapsLabel?: string\n /** Auto-reset to initial viewport after N ms of inactivity (0 = disabled) */\n autoResetDelay?: number\n /** Show reset button */\n showResetButton?: boolean\n}\n\ninterface MapInnerProps extends Omit<MapContainerProps, 'initialViewport'> {}\n\nfunction MapInner({\n children,\n mapStyle = 'light',\n interactiveLayerIds,\n style,\n cursor,\n attributionControl = true,\n reuseMaps = true,\n openInMapsUrl,\n openInMapsLabel = 'Open in Maps',\n autoResetDelay = 0,\n showResetButton = false,\n}: MapInnerProps) {\n const { mapRef, viewport, setViewport, setIsLoaded, resetToInitial, initialViewport } = useMapContext()\n const resetTimerRef = useRef<NodeJS.Timeout | null>(null)\n const isInteractingRef = useRef(false)\n\n // Check if viewport has changed from initial\n const hasViewportChanged =\n Math.abs(viewport.longitude - initialViewport.longitude) > 0.0001 ||\n Math.abs(viewport.latitude - initialViewport.latitude) > 0.0001 ||\n Math.abs(viewport.zoom - initialViewport.zoom) > 0.1\n\n const handleMove = useCallback(\n (evt: ViewStateChangeEvent) => {\n setViewport({\n longitude: evt.viewState.longitude,\n latitude: evt.viewState.latitude,\n zoom: evt.viewState.zoom,\n bearing: evt.viewState.bearing,\n pitch: evt.viewState.pitch,\n })\n },\n [setViewport]\n )\n\n const handleMoveStart = useCallback(() => {\n isInteractingRef.current = true\n // Clear any pending reset timer\n if (resetTimerRef.current) {\n clearTimeout(resetTimerRef.current)\n resetTimerRef.current = null\n }\n }, [])\n\n const handleMoveEnd = useCallback(() => {\n isInteractingRef.current = false\n // Start auto-reset timer if enabled\n if (autoResetDelay > 0) {\n resetTimerRef.current = setTimeout(() => {\n if (!isInteractingRef.current) {\n resetToInitial()\n }\n }, autoResetDelay)\n }\n }, [autoResetDelay, resetToInitial])\n\n const handleLoad = useCallback(() => {\n setIsLoaded(true)\n }, [setIsLoaded])\n\n // Cleanup timer on unmount\n useEffect(() => {\n return () => {\n if (resetTimerRef.current) {\n clearTimeout(resetTimerRef.current)\n }\n }\n }, [])\n\n const resolvedStyle = mapStyle in MAP_STYLES\n ? MAP_STYLES[mapStyle as MapStyleKey]\n : mapStyle\n\n return (\n <>\n <Map\n ref={mapRef}\n {...viewport}\n onMove={handleMove}\n onMoveStart={handleMoveStart}\n onMoveEnd={handleMoveEnd}\n onLoad={handleLoad}\n mapStyle={resolvedStyle}\n interactiveLayerIds={interactiveLayerIds}\n attributionControl={attributionControl ? {} : false}\n reuseMaps={reuseMaps}\n style={{\n width: '100%',\n height: '100%',\n ...style,\n }}\n cursor={cursor}\n >\n {children}\n </Map>\n\n {/* Map overlay buttons */}\n <div className=\"absolute bottom-3 right-3 flex items-center gap-2\">\n {/* Reset button */}\n {showResetButton && hasViewportChanged && (\n <button\n type=\"button\"\n onClick={resetToInitial}\n className={cn(\n 'inline-flex items-center gap-2 px-3 py-2 rounded-lg',\n 'bg-background/95 backdrop-blur-sm text-foreground text-sm font-medium',\n 'hover:bg-background transition-colors shadow-sm border border-border'\n )}\n >\n <RotateCcw className=\"w-4 h-4\" />\n Reset\n </button>\n )}\n\n {/* Open in Maps button */}\n {openInMapsUrl && (\n <a\n href={openInMapsUrl}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n className={cn(\n 'inline-flex items-center gap-2 px-4 py-2 rounded-lg',\n 'bg-background/95 backdrop-blur-sm text-foreground text-sm font-medium',\n 'hover:bg-background transition-colors shadow-sm border border-border'\n )}\n >\n <ExternalLink className=\"w-4 h-4\" />\n {openInMapsLabel}\n </a>\n )}\n </div>\n </>\n )\n}\n\nexport function MapContainer({\n children,\n initialViewport,\n className,\n ...props\n}: MapContainerProps) {\n return (\n <div className={className} style={{ width: '100%', height: '100%', position: 'relative' }}>\n <MapProvider initialViewport={initialViewport}>\n <MapInner {...props}>{children}</MapInner>\n </MapProvider>\n </div>\n )\n}\n\n/**\n * Use this when you need the map inside an existing MapProvider\n */\nexport function MapView(props: Omit<MapContainerProps, 'initialViewport'>) {\n return <MapInner {...props} />\n}\n"]}
|
|
@@ -209,6 +209,6 @@ function MapView(props) {
|
|
|
209
209
|
}
|
|
210
210
|
__name(MapView, "MapView");
|
|
211
211
|
|
|
212
|
-
export { MapContainer, MapView };
|
|
213
|
-
//# sourceMappingURL=
|
|
214
|
-
//# sourceMappingURL=
|
|
212
|
+
export { MapContainer, MapView, useMapContext };
|
|
213
|
+
//# sourceMappingURL=chunk-7CWGZPO3.mjs.map
|
|
214
|
+
//# sourceMappingURL=chunk-7CWGZPO3.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/tools/Map/context/MapContext.tsx","../src/tools/Map/styles/index.ts","../src/tools/Map/components/MapContainer.tsx"],"names":["useRef","useCallback","jsx"],"mappings":"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","file":"chunk-7CWGZPO3.mjs","sourcesContent":["'use client'\n\nimport {\n createContext,\n useContext,\n useState,\n useRef,\n useMemo,\n useCallback,\n type ReactNode,\n} from 'react'\nimport type { MapRef } from 'react-map-gl/maplibre'\nimport type { MapContextValue, MapViewport, MarkerData } from '../types'\n\nconst MapContext = createContext<MapContextValue | null>(null)\n\nexport interface MapProviderProps {\n children: ReactNode\n initialViewport?: Partial<MapViewport>\n}\n\nconst DEFAULT_VIEWPORT: MapViewport = {\n longitude: 115.1889,\n latitude: -8.4095,\n zoom: 10,\n bearing: 0,\n pitch: 0,\n}\n\nexport function MapProvider({ children, initialViewport }: MapProviderProps) {\n const mapRef = useRef<MapRef | null>(null)\n const initialViewportRef = useRef<MapViewport>({\n ...DEFAULT_VIEWPORT,\n ...initialViewport,\n })\n const [viewport, setViewportState] = useState<MapViewport>(initialViewportRef.current)\n const [markers, setMarkers] = useState<MarkerData[]>([])\n const [selectedMarker, setSelectedMarker] = useState<MarkerData | null>(null)\n const [hoveredFeature, setHoveredFeature] = useState<GeoJSON.Feature | null>(null)\n const [isLoaded, setIsLoaded] = useState(false)\n\n const setViewport = useCallback((newViewport: Partial<MapViewport>) => {\n setViewportState((prev) => ({ ...prev, ...newViewport }))\n }, [])\n\n const resetToInitial = useCallback(() => {\n const map = mapRef.current\n if (map) {\n map.flyTo({\n center: [initialViewportRef.current.longitude, initialViewportRef.current.latitude],\n zoom: initialViewportRef.current.zoom,\n bearing: initialViewportRef.current.bearing ?? 0,\n pitch: initialViewportRef.current.pitch ?? 0,\n duration: 1000,\n })\n }\n }, [])\n\n const value = useMemo<MapContextValue>(\n () => ({\n mapRef,\n viewport,\n setViewport,\n initialViewport: initialViewportRef.current,\n resetToInitial,\n markers,\n setMarkers,\n selectedMarker,\n setSelectedMarker,\n hoveredFeature,\n setHoveredFeature,\n isLoaded,\n setIsLoaded,\n }),\n [viewport, setViewport, resetToInitial, markers, selectedMarker, hoveredFeature, isLoaded]\n )\n\n return <MapContext.Provider value={value}>{children}</MapContext.Provider>\n}\n\nexport function useMapContext(): MapContextValue {\n const context = useContext(MapContext)\n if (!context) {\n throw new Error('useMapContext must be used within a MapProvider')\n }\n return context\n}\n\nexport { MapContext }\n","export const MAP_STYLES = {\n light: 'https://basemaps.cartocdn.com/gl/positron-gl-style/style.json',\n dark: 'https://basemaps.cartocdn.com/gl/dark-matter-gl-style/style.json',\n streets: 'https://basemaps.cartocdn.com/gl/voyager-gl-style/style.json',\n satellite: 'https://api.maptiler.com/maps/satellite/style.json',\n} as const\n\nexport type MapStyleKey = keyof typeof MAP_STYLES\n\nexport function getMapStyle(key: MapStyleKey | string): string {\n if (key in MAP_STYLES) {\n return MAP_STYLES[key as MapStyleKey]\n }\n return key\n}\n","'use client'\n\nimport { useCallback, useEffect, useRef, type ReactNode } from 'react'\nimport Map, { type ViewStateChangeEvent } from 'react-map-gl/maplibre'\nimport { ExternalLink, RotateCcw } from 'lucide-react'\nimport { cn } from '@djangocfg/ui-core/lib'\nimport 'maplibre-gl/dist/maplibre-gl.css'\n\nimport { MapProvider, useMapContext } from '../context'\nimport { MAP_STYLES } from '../styles'\nimport type { MapViewport, MapStyleKey } from '../types'\n\nexport interface MapContainerProps {\n children?: ReactNode\n initialViewport?: Partial<MapViewport>\n mapStyle?: MapStyleKey | string\n interactiveLayerIds?: string[]\n className?: string\n style?: React.CSSProperties\n cursor?: string\n attributionControl?: boolean\n reuseMaps?: boolean\n /** URL to open in external maps app (shows \"Open in Maps\" button if provided) */\n openInMapsUrl?: string\n /** Label for the open in maps button */\n openInMapsLabel?: string\n /** Auto-reset to initial viewport after N ms of inactivity (0 = disabled) */\n autoResetDelay?: number\n /** Show reset button */\n showResetButton?: boolean\n}\n\ninterface MapInnerProps extends Omit<MapContainerProps, 'initialViewport'> {}\n\nfunction MapInner({\n children,\n mapStyle = 'light',\n interactiveLayerIds,\n style,\n cursor,\n attributionControl = true,\n reuseMaps = true,\n openInMapsUrl,\n openInMapsLabel = 'Open in Maps',\n autoResetDelay = 0,\n showResetButton = false,\n}: MapInnerProps) {\n const { mapRef, viewport, setViewport, setIsLoaded, resetToInitial, initialViewport } = useMapContext()\n const resetTimerRef = useRef<NodeJS.Timeout | null>(null)\n const isInteractingRef = useRef(false)\n\n // Check if viewport has changed from initial\n const hasViewportChanged =\n Math.abs(viewport.longitude - initialViewport.longitude) > 0.0001 ||\n Math.abs(viewport.latitude - initialViewport.latitude) > 0.0001 ||\n Math.abs(viewport.zoom - initialViewport.zoom) > 0.1\n\n const handleMove = useCallback(\n (evt: ViewStateChangeEvent) => {\n setViewport({\n longitude: evt.viewState.longitude,\n latitude: evt.viewState.latitude,\n zoom: evt.viewState.zoom,\n bearing: evt.viewState.bearing,\n pitch: evt.viewState.pitch,\n })\n },\n [setViewport]\n )\n\n const handleMoveStart = useCallback(() => {\n isInteractingRef.current = true\n // Clear any pending reset timer\n if (resetTimerRef.current) {\n clearTimeout(resetTimerRef.current)\n resetTimerRef.current = null\n }\n }, [])\n\n const handleMoveEnd = useCallback(() => {\n isInteractingRef.current = false\n // Start auto-reset timer if enabled\n if (autoResetDelay > 0) {\n resetTimerRef.current = setTimeout(() => {\n if (!isInteractingRef.current) {\n resetToInitial()\n }\n }, autoResetDelay)\n }\n }, [autoResetDelay, resetToInitial])\n\n const handleLoad = useCallback(() => {\n setIsLoaded(true)\n }, [setIsLoaded])\n\n // Cleanup timer on unmount\n useEffect(() => {\n return () => {\n if (resetTimerRef.current) {\n clearTimeout(resetTimerRef.current)\n }\n }\n }, [])\n\n const resolvedStyle = mapStyle in MAP_STYLES\n ? MAP_STYLES[mapStyle as MapStyleKey]\n : mapStyle\n\n return (\n <>\n <Map\n ref={mapRef}\n {...viewport}\n onMove={handleMove}\n onMoveStart={handleMoveStart}\n onMoveEnd={handleMoveEnd}\n onLoad={handleLoad}\n mapStyle={resolvedStyle}\n interactiveLayerIds={interactiveLayerIds}\n attributionControl={attributionControl ? {} : false}\n reuseMaps={reuseMaps}\n style={{\n width: '100%',\n height: '100%',\n ...style,\n }}\n cursor={cursor}\n >\n {children}\n </Map>\n\n {/* Map overlay buttons */}\n <div className=\"absolute bottom-3 right-3 flex items-center gap-2\">\n {/* Reset button */}\n {showResetButton && hasViewportChanged && (\n <button\n type=\"button\"\n onClick={resetToInitial}\n className={cn(\n 'inline-flex items-center gap-2 px-3 py-2 rounded-lg',\n 'bg-background/95 backdrop-blur-sm text-foreground text-sm font-medium',\n 'hover:bg-background transition-colors shadow-sm border border-border'\n )}\n >\n <RotateCcw className=\"w-4 h-4\" />\n Reset\n </button>\n )}\n\n {/* Open in Maps button */}\n {openInMapsUrl && (\n <a\n href={openInMapsUrl}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n className={cn(\n 'inline-flex items-center gap-2 px-4 py-2 rounded-lg',\n 'bg-background/95 backdrop-blur-sm text-foreground text-sm font-medium',\n 'hover:bg-background transition-colors shadow-sm border border-border'\n )}\n >\n <ExternalLink className=\"w-4 h-4\" />\n {openInMapsLabel}\n </a>\n )}\n </div>\n </>\n )\n}\n\nexport function MapContainer({\n children,\n initialViewport,\n className,\n ...props\n}: MapContainerProps) {\n return (\n <div className={className} style={{ width: '100%', height: '100%', position: 'relative' }}>\n <MapProvider initialViewport={initialViewport}>\n <MapInner {...props}>{children}</MapInner>\n </MapProvider>\n </div>\n )\n}\n\n/**\n * Use this when you need the map inside an existing MapProvider\n */\nexport function MapView(props: Omit<MapContainerProps, 'initialViewport'>) {\n return <MapInner {...props} />\n}\n"]}
|
|
@@ -210,6 +210,50 @@ function newSegmentId() {
|
|
|
210
210
|
return `seg_${Date.now().toString(36)}_${counter.toString(36)}`;
|
|
211
211
|
}
|
|
212
212
|
chunkOLISEQHS_cjs.__name(newSegmentId, "newSegmentId");
|
|
213
|
+
var SCOPES = [
|
|
214
|
+
"engine",
|
|
215
|
+
"dictation",
|
|
216
|
+
"slot",
|
|
217
|
+
"composer",
|
|
218
|
+
"mic",
|
|
219
|
+
"push",
|
|
220
|
+
"error"
|
|
221
|
+
];
|
|
222
|
+
var STORAGE_KEY = "djangocfg:speech-debug";
|
|
223
|
+
function readStorageFlag() {
|
|
224
|
+
if (!lib.isBrowser) return false;
|
|
225
|
+
try {
|
|
226
|
+
return window.localStorage.getItem(STORAGE_KEY) === "1";
|
|
227
|
+
} catch {
|
|
228
|
+
return false;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
chunkOLISEQHS_cjs.__name(readStorageFlag, "readStorageFlag");
|
|
232
|
+
var cache = /* @__PURE__ */ new Map();
|
|
233
|
+
function buildLogger(enabled) {
|
|
234
|
+
const root = consola.consola.withTag("speech");
|
|
235
|
+
const subs = Object.fromEntries(
|
|
236
|
+
SCOPES.map((scope) => [scope, root.withTag(scope)])
|
|
237
|
+
);
|
|
238
|
+
if (!enabled) {
|
|
239
|
+
for (const scope of SCOPES) {
|
|
240
|
+
if (scope === "error") continue;
|
|
241
|
+
subs[scope].level = -999;
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
return { ...subs, enabled };
|
|
245
|
+
}
|
|
246
|
+
chunkOLISEQHS_cjs.__name(buildLogger, "buildLogger");
|
|
247
|
+
function getSpeechLogger(debug) {
|
|
248
|
+
const enabled = debug ?? (lib.isDev || readStorageFlag());
|
|
249
|
+
let logger = cache.get(enabled);
|
|
250
|
+
if (!logger) {
|
|
251
|
+
logger = buildLogger(enabled);
|
|
252
|
+
cache.set(enabled, logger);
|
|
253
|
+
}
|
|
254
|
+
return logger;
|
|
255
|
+
}
|
|
256
|
+
chunkOLISEQHS_cjs.__name(getSpeechLogger, "getSpeechLogger");
|
|
213
257
|
var sttLogger = consola.consola.withTag("ui-tools:speech");
|
|
214
258
|
|
|
215
259
|
// src/tools/SpeechRecognition/core/reducer.ts
|
|
@@ -538,6 +582,7 @@ function useResolvedLanguage(explicit) {
|
|
|
538
582
|
chunkOLISEQHS_cjs.__name(useResolvedLanguage, "useResolvedLanguage");
|
|
539
583
|
|
|
540
584
|
// src/tools/SpeechRecognition/hooks/useSpeechRecognition.ts
|
|
585
|
+
var log = getSpeechLogger();
|
|
541
586
|
function useSpeechRecognition(config = {}) {
|
|
542
587
|
const prefs = useSpeechPrefs();
|
|
543
588
|
const language = useResolvedLanguage(config.language);
|
|
@@ -551,8 +596,10 @@ function useSpeechRecognition(config = {}) {
|
|
|
551
596
|
const cbRef = react.useRef(config);
|
|
552
597
|
cbRef.current = config;
|
|
553
598
|
react.useEffect(() => {
|
|
599
|
+
log.engine.debug("subscribe", { engineId: engine.id });
|
|
554
600
|
const offs = [
|
|
555
601
|
engine.on("partial", (text, segmentId) => {
|
|
602
|
+
log.engine.debug("partial", { len: text.length, segmentId });
|
|
556
603
|
dispatch({ type: "PARTIAL", text, segmentId });
|
|
557
604
|
const seg = {
|
|
558
605
|
id: segmentId,
|
|
@@ -563,6 +610,11 @@ function useSpeechRecognition(config = {}) {
|
|
|
563
610
|
cbRef.current.onPartial?.(text, seg);
|
|
564
611
|
}),
|
|
565
612
|
engine.on("final", (text, segmentId, confidence) => {
|
|
613
|
+
log.engine.info("final", {
|
|
614
|
+
len: text.length,
|
|
615
|
+
segmentId,
|
|
616
|
+
confidence
|
|
617
|
+
});
|
|
566
618
|
dispatch({ type: "FINAL", text, segmentId, confidence });
|
|
567
619
|
const seg = {
|
|
568
620
|
id: segmentId,
|
|
@@ -575,10 +627,12 @@ function useSpeechRecognition(config = {}) {
|
|
|
575
627
|
cbRef.current.onFinal?.(text, seg);
|
|
576
628
|
}),
|
|
577
629
|
engine.on("error", (err) => {
|
|
630
|
+
log.error.error("engine error", err);
|
|
578
631
|
dispatch({ type: "ERROR", error: err });
|
|
579
632
|
cbRef.current.onError?.(err);
|
|
580
633
|
}),
|
|
581
634
|
engine.on("state", (s) => {
|
|
635
|
+
log.engine.debug("state", s);
|
|
582
636
|
if (s === "listening") {
|
|
583
637
|
dispatch({ type: "STARTED" });
|
|
584
638
|
cbRef.current.onStart?.();
|
|
@@ -601,7 +655,7 @@ function useSpeechRecognition(config = {}) {
|
|
|
601
655
|
const { silenceMs, maxMs, silenceThreshold = 0.02 } = config.autoStop ?? {};
|
|
602
656
|
if (maxMs) {
|
|
603
657
|
maxTimer.current = window.setTimeout(() => {
|
|
604
|
-
|
|
658
|
+
log.engine.debug("autoStop max duration hit");
|
|
605
659
|
void engine.stop();
|
|
606
660
|
}, maxMs);
|
|
607
661
|
}
|
|
@@ -610,7 +664,7 @@ function useSpeechRecognition(config = {}) {
|
|
|
610
664
|
if (level < silenceThreshold) {
|
|
611
665
|
if (silenceTimer.current == null) {
|
|
612
666
|
silenceTimer.current = window.setTimeout(() => {
|
|
613
|
-
|
|
667
|
+
log.engine.debug("autoStop silence detected");
|
|
614
668
|
void engine.stop();
|
|
615
669
|
}, silenceMs);
|
|
616
670
|
}
|
|
@@ -642,7 +696,7 @@ function useSpeechRecognition(config = {}) {
|
|
|
642
696
|
deviceId: config.deviceId ?? prefs.deviceId ?? void 0
|
|
643
697
|
});
|
|
644
698
|
} catch (cause) {
|
|
645
|
-
|
|
699
|
+
log.engine.warn("start threw", cause);
|
|
646
700
|
}
|
|
647
701
|
}, [engine, language, config.interim, config.deviceId, prefs.deviceId, state.status]);
|
|
648
702
|
const stop = react.useCallback(async () => {
|
|
@@ -831,7 +885,8 @@ function DictationField({
|
|
|
831
885
|
chunkOLISEQHS_cjs.__name(DictationField, "DictationField");
|
|
832
886
|
|
|
833
887
|
exports.DictationField = DictationField;
|
|
888
|
+
exports.getSpeechLogger = getSpeechLogger;
|
|
834
889
|
exports.useResolvedLanguage = useResolvedLanguage;
|
|
835
890
|
exports.useSpeechPrefs = useSpeechPrefs;
|
|
836
|
-
//# sourceMappingURL=chunk-
|
|
837
|
-
//# sourceMappingURL=chunk-
|
|
891
|
+
//# sourceMappingURL=chunk-ADEN3UA4.cjs.map
|
|
892
|
+
//# sourceMappingURL=chunk-ADEN3UA4.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/tools/SpeechRecognition/components/DictationButton.tsx","../src/tools/SpeechRecognition/components/ErrorBanner.tsx","../src/tools/SpeechRecognition/components/MicMeter.tsx","../src/tools/SpeechRecognition/components/PushToTalkHint.tsx","../src/tools/SpeechRecognition/core/transcript.ts","../src/tools/SpeechRecognition/core/ids.ts","../src/tools/SpeechRecognition/core/logger.ts","../src/tools/SpeechRecognition/core/reducer.ts","../src/tools/SpeechRecognition/core/engine/index.ts","../src/tools/SpeechRecognition/core/engine/webspeech.ts","../src/tools/SpeechRecognition/store/prefsStore.ts","../src/tools/SpeechRecognition/hooks/useMicLevel.ts","../src/tools/SpeechRecognition/core/language.ts","../src/tools/SpeechRecognition/hooks/useResolvedLanguage.ts","../src/tools/SpeechRecognition/hooks/useSpeechRecognition.ts","../src/tools/SpeechRecognition/hooks/useDictation.ts","../src/tools/SpeechRecognition/hooks/usePushToTalk.ts","../src/tools/SpeechRecognition/widgets/DictationField.tsx"],"names":["jsxs","cn","jsx","Loader2","MicOff","Mic","__name","AlertTriangle","isBrowser","consola","isDev","create","persist","createJSONStorage","useState","useRef","useEffect","useLocaleOptional","useMemo","useReducer","useCallback"],"mappings":";;;;;;;;;;;;AA2BA,IAAM,QAAA,GAAsE;AAAA,EAC1E,EAAA,EAAI,iCAAA;AAAA,EACJ,EAAA,EAAI,mCAAA;AAAA,EACJ,EAAA,EAAI;AACN,CAAA;AAOO,SAAS,eAAA,CAAgB;AAAA,EAC9B,MAAA;AAAA,EACA,OAAA;AAAA,EACA,WAAA,GAAc,IAAA;AAAA,EACd,IAAA,GAAO,IAAA;AAAA,EACP,SAAA;AAAA,EACA,KAAA;AAAA,EACA,SAAA;AAAA,EACA,QAAA;AAAA,EACA,aAAA;AAAA,EACA;AACF,CAAA,EAA6C;AAC3C,EAAA,MAAM,SAAA,GAAY,MAAA,KAAW,WAAA,IAAe,MAAA,KAAW,UAAA;AACvD,EAAA,MAAM,WAAW,MAAA,KAAW,UAAA;AAC5B,EAAA,MAAM,MAAM,CAAC,WAAA;AAEb,EAAA,uBACEA,eAAA;AAAA,IAAC,QAAA;AAAA,IAAA;AAAA,MACC,IAAA,EAAK,QAAA;AAAA,MACL,OAAA;AAAA,MACA,UAAU,QAAA,IAAY,GAAA;AAAA,MACtB,cAAA,EAAc,SAAA;AAAA,MACd,YAAA,EACE,SAAA,KAAc,SAAA,GAAY,gBAAA,GAAmB,MAAM,yBAAA,GAA4B,iBAAA,CAAA;AAAA,MAEjF,SAAA,EAAWC,MAAA;AAAA,QACT,iFAAA;AAAA,QACA,qGAAA;AAAA,QACA,iDAAA;AAAA,QACA,SAAS,IAAI,CAAA;AAAA,QACb,YACI,oEAAA,GACA,wDAAA;AAAA,QACJ;AAAA,OACF;AAAA,MACA,KAAA;AAAA,MAEC,QAAA,EAAA;AAAA,QAAA,SAAA,oBACCC,cAAA;AAAA,UAAC,MAAA;AAAA,UAAA;AAAA,YACC,aAAA,EAAW,IAAA;AAAA,YACX,SAAA,EAAU;AAAA;AAAA,SACZ;AAAA,QAED,2BACCA,cAAA,CAACC,mBAAA,EAAA,EAAQ,WAAU,cAAA,EAAe,CAAA,GAChC,MACF,aAAA,oBAAiBD,cAAA,CAACE,kBAAA,EAAA,EAAO,CAAA,GACvB,YACF,aAAA,oBAAiBF,cAAA,CAACG,mBAAI,CAAA,GAEtB,QAAA,mCAAaA,eAAA,EAAA,EAAI;AAAA;AAAA;AAAA,GAErB;AAEJ;AAtDgBC,wBAAA,CAAA,eAAA,EAAA,iBAAA,CAAA;AC5BhB,IAAM,QAAA,GAAqD;AAAA,EACzD,WAAA,EAAa,qDAAA;AAAA,EACb,mBAAA,EAAqB,6EAAA;AAAA,EACrB,eAAA,EAAiB,sBAAA;AAAA,EACjB,OAAA,EAAS,8CAAA;AAAA,EACT,OAAA,EAAS,8BAAA;AAAA,EACT,WAAA,EAAa,oCAAA;AAAA,EACb,QAAA,EAAU,uDAAA;AAAA,EACV,MAAA,EAAQ,sCAAA;AAAA,EACR,OAAA,EAAS;AACX,CAAA;AAQO,SAAS,WAAA,CAAY,EAAE,KAAA,EAAO,SAAA,EAAW,WAAU,EAAgD;AACxG,EAAA,IAAI,CAAC,OAAO,OAAO,IAAA;AACnB,EAAA,uBACEN,eAAAA;AAAA,IAAC,KAAA;AAAA,IAAA;AAAA,MACC,IAAA,EAAK,OAAA;AAAA,MACL,SAAA,EAAWC,MAAAA;AAAA,QACT,qHAAA;AAAA,QACA;AAAA,OACF;AAAA,MAEA,QAAA,EAAA;AAAA,wBAAAC,cAAAA,CAACK,yBAAA,EAAA,EAAc,SAAA,EAAU,6BAAA,EAA8B,CAAA;AAAA,wBACvDL,cAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,QAAA,EAAU,mBAAS,KAAA,CAAM,IAAI,CAAA,IAAK,KAAA,CAAM,OAAA,EAAQ,CAAA;AAAA,QAC9D,6BACCA,cAAAA;AAAA,UAAC,QAAA;AAAA,UAAA;AAAA,YACC,IAAA,EAAK,QAAA;AAAA,YACL,OAAA,EAAS,SAAA;AAAA,YACT,SAAA,EAAU,kCAAA;AAAA,YACX,QAAA,EAAA;AAAA;AAAA;AAED;AAAA;AAAA,GAEJ;AAEJ;AAvBgBI,wBAAA,CAAA,WAAA,EAAA,aAAA,CAAA;ACHT,SAAS,QAAA,CAAS;AAAA,EACvB,KAAA;AAAA,EACA,IAAA,GAAO,EAAA;AAAA,EACP,GAAA,GAAM,CAAA;AAAA,EACN,QAAA,GAAW,CAAA;AAAA,EACX,MAAA,GAAS,EAAA;AAAA,EACT;AACF,CAAA,EAAsC;AACpC,EAAA,MAAM,OAAA,GAAU,KAAK,GAAA,CAAI,CAAA,EAAG,KAAK,GAAA,CAAI,CAAA,EAAG,KAAK,CAAC,CAAA;AAC9C,EAAA,uBACEJ,cAAAA;AAAA,IAAC,KAAA;AAAA,IAAA;AAAA,MACC,IAAA,EAAK,OAAA;AAAA,MACL,eAAA,EAAe,CAAA;AAAA,MACf,eAAA,EAAe,CAAA;AAAA,MACf,eAAA,EAAe,OAAA;AAAA,MACf,SAAA,EAAWD,MAAAA,CAAG,0BAAA,EAA4B,SAAS,CAAA;AAAA,MACnD,KAAA,EAAO,EAAE,GAAA,EAAK,MAAA,EAAO;AAAA,MAEpB,QAAA,EAAA,KAAA,CAAM,IAAA,CAAK,EAAE,MAAA,EAAQ,IAAA,EAAM,CAAA,CAAE,GAAA,CAAI,CAAC,CAAA,EAAG,CAAA,KAAM;AAC1C,QAAA,MAAM,KAAA,GAAA,CAAS,IAAI,CAAA,IAAK,IAAA;AACxB,QAAA,MAAM,MAAA,GAAS,OAAA,IAAW,KAAA,GAAQ,GAAA,GAAM,IAAA;AAExC,QAAA,MAAM,WAAA,GAAc,CAAA,GAAI,IAAA,CAAK,GAAA,CAAI,CAAA,GAAA,CAAK,IAAA,GAAO,CAAA,IAAK,CAAC,CAAA,IAAA,CAAM,IAAA,GAAO,CAAA,IAAK,CAAA,IAAK,CAAA,CAAA;AAC1E,QAAA,MAAM,OAAO,IAAA,CAAK,GAAA,CAAI,GAAG,MAAA,IAAU,IAAA,GAAO,cAAc,IAAA,CAAK,CAAA;AAC7D,QAAA,uBACEC,cAAAA;AAAA,UAAC,MAAA;AAAA,UAAA;AAAA,YAEC,SAAA,EAAWD,MAAAA;AAAA,cACT,8BAAA;AAAA,cACA,SAAS,YAAA,GAAe;AAAA,aAC1B;AAAA,YACA,KAAA,EAAO,EAAE,KAAA,EAAO,QAAA,EAAU,QAAQ,IAAA;AAAK,WAAA;AAAA,UALlC;AAAA,SAMP;AAAA,MAEJ,CAAC;AAAA;AAAA,GACH;AAEJ;AArCgBK,wBAAA,CAAA,QAAA,EAAA,UAAA,CAAA;ACTT,SAAS,cAAA,CAAe,EAAE,KAAA,EAAO,SAAA,EAAU,EAA4C;AAC5F,EAAA,MAAM,YAAY,KAAA,CACf,KAAA,CAAM,GAAG,CAAA,CACT,IAAI,CAAC,CAAA,KAAM,CAAA,CAAE,IAAA,GAAO,WAAA,EAAa,CAAA,CACjC,GAAA,CAAI,CAAC,CAAA,KAAM;AACV,IAAA,QAAQ,CAAA;AAAG,MACT,KAAK,KAAA;AAAA,MACL,KAAK,MAAA;AACH,QAAA,OAAO,QAAA;AAAA,MACT,KAAK,KAAA;AACH,QAAA,OAAO,QAAA;AAAA,MACT,KAAK,OAAA;AACH,QAAA,OAAO,QAAA;AAAA,MACT,KAAK,MAAA;AACH,QAAA,OAAO,QAAA;AAAA,MACT;AACE,QAAA,OAAO,CAAA,CAAE,MAAA,KAAW,CAAA,GAAI,CAAA,CAAE,aAAY,GAAI,CAAA;AAAA;AAC9C,EACF,CAAC,CAAA,CACA,IAAA,CAAK,EAAE,CAAA;AACV,EAAA,uBACEN,eAAAA;AAAA,IAAC,MAAA;AAAA,IAAA;AAAA,MACC,SAAA,EAAWC,MAAAA;AAAA,QACT,kEAAA;AAAA,QACA;AAAA,OACF;AAAA,MACD,QAAA,EAAA;AAAA,QAAA,MAAA;AAAA,wBAECC,cAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,2FACZ,QAAA,EAAA,SAAA,EACH,CAAA;AAAA,QAAM;AAAA;AAAA;AAAA,GAER;AAEJ;AAlCgBI,wBAAA,CAAA,cAAA,EAAA,gBAAA,CAAA;;;ACdT,IAAM,gBAAA,GAA+B;AAAA,EAC1C,OAAA,EAAS,EAAA;AAAA,EACT,KAAA,EAAO,EAAA;AAAA,EACP,UAAU;AACZ,CAAA;AAEO,SAAS,UAAU,QAAA,EAA6B;AACrD,EAAA,IAAI,GAAA,GAAM,EAAA;AACV,EAAA,KAAA,MAAW,OAAO,QAAA,EAAU;AAC1B,IAAA,IAAI,CAAC,IAAI,OAAA,EAAS;AAClB,IAAA,MAAM,IAAA,GAAO,GAAA,CAAI,IAAA,CAAK,IAAA,EAAK;AAC3B,IAAA,IAAI,CAAC,IAAA,EAAM;AACX,IAAA,GAAA,GAAM,GAAA,GAAM,CAAA,EAAG,GAAG,CAAA,CAAA,EAAI,IAAI,CAAA,CAAA,GAAK,IAAA;AAAA,EACjC;AACA,EAAA,OAAO,GAAA;AACT;AATgBA,wBAAA,CAAA,SAAA,EAAA,WAAA,CAAA;AAWT,SAAS,gBAAgB,QAAA,EAAiC;AAC/D,EAAA,MAAM,IAAA,GAAO,QAAA,CAAS,QAAA,CAAS,MAAA,GAAS,CAAC,CAAA;AACzC,EAAA,MAAM,UAAU,IAAA,IAAQ,CAAC,IAAA,CAAK,OAAA,GAAU,KAAK,IAAA,GAAO,EAAA;AACpD,EAAA,OAAO;AAAA,IACL,OAAA;AAAA,IACA,KAAA,EAAO,UAAU,QAAQ,CAAA;AAAA,IACzB;AAAA,GACF;AACF;AARgBA,wBAAA,CAAA,eAAA,EAAA,iBAAA,CAAA;AAcT,SAAS,eAAe,IAAA,EAAsB;AACnD,EAAA,OAAO,IAAA,CAAK,QAAQ,MAAA,EAAQ,GAAG,EAAE,OAAA,CAAQ,cAAA,EAAgB,IAAI,CAAA,CAAE,IAAA,EAAK;AACtE;AAFgBA,wBAAA,CAAA,cAAA,EAAA,gBAAA,CAAA;;;ACjChB,IAAI,OAAA,GAAU,CAAA;AAOP,SAAS,YAAA,GAAuB;AACrC,EAAA,OAAA,GAAA,CAAW,OAAA,GAAU,KAAK,MAAA,CAAO,gBAAA;AACjC,EAAA,OAAO,CAAA,IAAA,EAAO,IAAA,CAAK,GAAA,EAAI,CAAE,QAAA,CAAS,EAAE,CAAC,CAAA,CAAA,EAAI,OAAA,CAAQ,QAAA,CAAS,EAAE,CAAC,CAAA,CAAA;AAC/D;AAHgBA,wBAAA,CAAA,YAAA,EAAA,cAAA,CAAA;ACwChB,IAAM,MAAA,GAA2B;AAAA,EAC/B,QAAA;AAAA,EACA,WAAA;AAAA,EACA,MAAA;AAAA,EACA,UAAA;AAAA,EACA,KAAA;AAAA,EACA,MAAA;AAAA,EACA;AACF,CAAA;AAEA,IAAM,WAAA,GAAc,wBAAA;AAEpB,SAAS,eAAA,GAA2B;AAClC,EAAA,IAAI,CAACE,eAAW,OAAO,KAAA;AACvB,EAAA,IAAI;AACF,IAAA,OAAO,MAAA,CAAO,YAAA,CAAa,OAAA,CAAQ,WAAW,CAAA,KAAM,GAAA;AAAA,EACtD,CAAA,CAAA,MAAQ;AACN,IAAA,OAAO,KAAA;AAAA,EACT;AACF;AAPSF,wBAAA,CAAA,eAAA,EAAA,iBAAA,CAAA;AAST,IAAM,KAAA,uBAAY,GAAA,EAA2B;AAE7C,SAAS,YAAY,OAAA,EAAgC;AACnD,EAAA,MAAM,IAAA,GAAOG,eAAA,CAAQ,OAAA,CAAQ,QAAQ,CAAA;AACrC,EAAA,MAAM,OAAO,MAAA,CAAO,WAAA;AAAA,IAClB,MAAA,CAAO,GAAA,CAAI,CAAC,KAAA,KAAU,CAAC,OAAO,IAAA,CAAK,OAAA,CAAQ,KAAK,CAAC,CAAC;AAAA,GACpD;AAEA,EAAA,IAAI,CAAC,OAAA,EAAS;AACZ,IAAA,KAAA,MAAW,SAAS,MAAA,EAAQ;AAC1B,MAAA,IAAI,UAAU,OAAA,EAAS;AACvB,MAAA,IAAA,CAAK,KAAK,EAAE,KAAA,GAAQ,IAAA;AAAA,IACtB;AAAA,EACF;AAEA,EAAA,OAAO,EAAE,GAAG,IAAA,EAAM,OAAA,EAAQ;AAC5B;AAdSH,wBAAA,CAAA,WAAA,EAAA,aAAA,CAAA;AAsBF,SAAS,gBAAgB,KAAA,EAA+B;AAC7D,EAAA,MAAM,OAAA,GAAU,KAAA,KAAUI,SAAA,IAAS,eAAA,EAAgB,CAAA;AACnD,EAAA,IAAI,MAAA,GAAS,KAAA,CAAM,GAAA,CAAI,OAAO,CAAA;AAC9B,EAAA,IAAI,CAAC,MAAA,EAAQ;AACX,IAAA,MAAA,GAAS,YAAY,OAAO,CAAA;AAC5B,IAAA,KAAA,CAAM,GAAA,CAAI,SAAS,MAAM,CAAA;AAAA,EAC3B;AACA,EAAA,OAAO,MAAA;AACT;AARgBJ,wBAAA,CAAA,eAAA,EAAA,iBAAA,CAAA;AAgBT,IAAM,SAAA,GAAYG,eAAA,CAAQ,OAAA,CAAQ,iBAAiB,CAAA;;;AC9FnD,IAAM,aAAA,GAAkC;AAAA,EAC7C,MAAA,EAAQ,MAAA;AAAA,EACR,UAAU,EAAC;AAAA,EACX,KAAA,EAAO,IAAA;AAAA,EACP,SAAA,EAAW;AACb,CAAA;AAuBA,SAAS,cAAc,KAAA,EAAiC;AACtD,EAAA,OAAO,MAAM,SAAA,GAAY,IAAA,CAAK,GAAA,EAAI,GAAI,MAAM,SAAA,GAAY,CAAA;AAC1D;AAFSH,wBAAA,CAAA,aAAA,EAAA,eAAA,CAAA;AAIT,SAAS,aAAA,CACP,UACA,KAAA,EACW;AACX,EAAA,MAAM,GAAA,GAAM,SAAS,SAAA,CAAU,CAAC,MAAM,CAAA,CAAE,EAAA,KAAO,MAAM,EAAE,CAAA;AACvD,EAAA,IAAI,QAAQ,EAAA,EAAI,OAAO,CAAC,GAAG,UAAU,KAAK,CAAA;AAC1C,EAAA,MAAM,IAAA,GAAO,SAAS,KAAA,EAAM;AAC5B,EAAA,IAAA,CAAK,GAAG,IAAI,EAAE,GAAG,KAAK,GAAG,CAAA,EAAG,GAAG,KAAA,EAAM;AACrC,EAAA,OAAO,IAAA;AACT;AATSA,wBAAA,CAAA,aAAA,EAAA,eAAA,CAAA;AAWF,SAAS,OAAA,CACd,OACA,MAAA,EACkB;AAClB,EAAA,QAAQ,OAAO,IAAA;AAAM,IACnB,KAAK,OAAA;AACH,MAAA,OAAO;AAAA,QACL,GAAG,KAAA;AAAA,QACH,MAAA,EAAQ,UAAA;AAAA,QACR,KAAA,EAAO,IAAA;AAAA,QACP,SAAA,EAAW,KAAK,GAAA;AAAI,OACtB;AAAA,IACF,KAAK,SAAA;AACH,MAAA,OAAO,EAAE,GAAG,KAAA,EAAO,MAAA,EAAQ,WAAA,EAAY;AAAA,IACzC,KAAK,MAAA;AACH,MAAA,OAAO,EAAE,GAAG,KAAA,EAAO,MAAA,EAAQ,UAAA,EAAW;AAAA,IACxC,KAAK,SAAA;AAAA,IACL,KAAK,OAAA;AACH,MAAA,OAAO,EAAE,GAAG,KAAA,EAAO,MAAA,EAAQ,MAAA,EAAO;AAAA,IACpC,KAAK,SAAA,EAAW;AACd,MAAA,MAAM,GAAA,GAAe;AAAA,QACnB,IAAI,MAAA,CAAO,SAAA;AAAA,QACX,MAAM,MAAA,CAAO,IAAA;AAAA,QACb,OAAA,EAAS,KAAA;AAAA,QACT,YAAY,MAAA,CAAO,UAAA;AAAA,QACnB,SAAA,EAAW,cAAc,KAAK;AAAA,OAChC;AACA,MAAA,OAAO,EAAE,GAAG,KAAA,EAAO,QAAA,EAAU,cAAc,KAAA,CAAM,QAAA,EAAU,GAAG,CAAA,EAAE;AAAA,IAClE;AAAA,IACA,KAAK,OAAA,EAAS;AACZ,MAAA,MAAM,GAAA,GAAe;AAAA,QACnB,EAAA,EAAI,MAAA,CAAO,SAAA,IAAa,YAAA,EAAa;AAAA,QACrC,MAAM,MAAA,CAAO,IAAA;AAAA,QACb,OAAA,EAAS,IAAA;AAAA,QACT,YAAY,MAAA,CAAO,UAAA;AAAA,QACnB,SAAA,EAAW,cAAc,KAAK,CAAA;AAAA,QAC9B,OAAA,EAAS,cAAc,KAAK;AAAA,OAC9B;AACA,MAAA,OAAO,EAAE,GAAG,KAAA,EAAO,QAAA,EAAU,cAAc,KAAA,CAAM,QAAA,EAAU,GAAG,CAAA,EAAE;AAAA,IAClE;AAAA,IACA,KAAK,OAAA;AACH,MAAA,OAAO,EAAE,GAAG,KAAA,EAAO,QAAQ,OAAA,EAAS,KAAA,EAAO,OAAO,KAAA,EAAM;AAAA,IAC1D,KAAK,OAAA;AACH,MAAA,OAAO,EAAE,GAAG,aAAA,EAAc;AAAA,IAC5B;AACE,MAAA,OAAO,KAAA;AAAA;AAEb;AA/CgBA,wBAAA,CAAA,OAAA,EAAA,SAAA,CAAA;;;AC7CT,SAAS,eAAA,GAOd;AACA,EAAA,MAAM,SAAA,GAAuB;AAAA,IAC3B,OAAA,sBAAa,GAAA,EAAI;AAAA,IACjB,KAAA,sBAAW,GAAA,EAAI;AAAA,IACf,KAAA,sBAAW,GAAA,EAAI;AAAA,IACf,KAAA,sBAAW,GAAA;AAAI,GACjB;AAEA,EAAA,OAAO;AAAA,IACL,EAAA,CAAG,OAAO,EAAA,EAAI;AACZ,MAAA,MAAM,GAAA,GAAM,UAAU,KAAK,CAAA;AAC3B,MAAA,GAAA,CAAI,IAAI,EAAE,CAAA;AACV,MAAA,OAAO,MAAM;AACX,QAAA,GAAA,CAAI,OAAO,EAAE,CAAA;AAAA,MACf,CAAA;AAAA,IACF,CAAA;AAAA,IACA,IAAA,CAAK,UAAU,IAAA,EAAM;AACnB,MAAA,MAAM,GAAA,GAAM,UAAU,KAAK,CAAA;AAC3B,MAAA,KAAA,MAAW,MAAM,GAAA,EAAK;AACpB,QAAA,IAAI;AACF,UAAC,EAAA,CAAiC,GAAI,IAAkB,CAAA;AAAA,QAC1D,CAAA,CAAA,MAAQ;AAAA,QAER;AAAA,MACF;AAAA,IACF,CAAA;AAAA,IACA,KAAA,GAAQ;AACN,MAAA,KAAA,MAAW,GAAA,IAAO,MAAA,CAAO,IAAA,CAAK,SAAS,CAAA,EAA6B;AAClE,QAAA,SAAA,CAAU,GAAG,EAAE,KAAA,EAAM;AAAA,MACvB;AAAA,IACF;AAAA,GACF;AACF;AAvCgBA,wBAAA,CAAA,eAAA,EAAA,iBAAA,CAAA;;;AC0ChB,SAAS,WAAA,GAA2B;AAClC,EAAA,IAAI,OAAO,MAAA,KAAW,WAAA,EAAa,OAAO,IAAA;AAC1C,EAAA,MAAM,CAAA,GAAI,MAAA;AAIV,EAAA,OAAO,CAAA,CAAE,iBAAA,IAAqB,CAAA,CAAE,uBAAA,IAA2B,IAAA;AAC7D;AAPSA,wBAAA,CAAA,WAAA,EAAA,aAAA,CAAA;AAST,IAAM,SAAA,GAAkD;AAAA,EACtD,WAAA,EAAa,WAAA;AAAA,EACb,OAAA,EAAS,SAAA;AAAA,EACT,eAAA,EAAiB,eAAA;AAAA,EACjB,OAAA,EAAS,SAAA;AAAA,EACT,aAAA,EAAe,mBAAA;AAAA,EACf,qBAAA,EAAuB,mBAAA;AAAA,EACvB,aAAA,EAAe,QAAA;AAAA,EACf,wBAAA,EAA0B;AAC5B,CAAA;AASO,SAAS,qBAAA,CACd,IAAA,GAA+B,EAAC,EACb;AACnB,EAAA,MAAM,OAAO,WAAA,EAAY;AACzB,EAAA,MAAM,MAAM,eAAA,EAAgB;AAC5B,EAAA,IAAI,QAAA,GAA4C,IAAA;AAChD,EAAA,IAAI,gBAAA,GAAkC,IAAA;AAEtC,EAAA,SAAS,QAAA,GAAiB;AACxB,IAAA,IAAI,CAAC,QAAA,EAAU;AACf,IAAA,QAAA,CAAS,QAAA,GAAW,IAAA;AACpB,IAAA,QAAA,CAAS,OAAA,GAAU,IAAA;AACnB,IAAA,QAAA,CAAS,OAAA,GAAU,IAAA;AACnB,IAAA,QAAA,CAAS,KAAA,GAAQ,IAAA;AACjB,IAAA,QAAA,GAAW,IAAA;AACX,IAAA,gBAAA,GAAmB,IAAA;AAAA,EACrB;AARS,EAAAA,wBAAA,CAAA,QAAA,EAAA,UAAA,CAAA;AAUT,EAAA,OAAO;AAAA,IACL,EAAA,EAAI,WAAA;AAAA,IACJ,aAAa,IAAA,KAAS,IAAA;AAAA,IACtB,EAAA,CAAG,OAAO,EAAA,EAAW;AACnB,MAAA,OAAO,GAAA,CAAI,EAAA,CAAG,KAAA,EAAO,EAAE,CAAA;AAAA,IACzB,CAAA;AAAA,IACA,MAAM,MAAM,KAAA,EAA0C;AACpD,MAAA,IAAI,CAAC,IAAA,EAAM;AACT,QAAA,MAAM,GAAA,GAAwB;AAAA,UAC5B,IAAA,EAAM,aAAA;AAAA,UACN,OAAA,EAAS;AAAA,SACX;AACA,QAAA,GAAA,CAAI,IAAA,CAAK,SAAS,GAAG,CAAA;AACrB,QAAA,MAAM,GAAA;AAAA,MACR;AACA,MAAA,IAAI,QAAA,EAAU;AACZ,QAAA,SAAA,CAAU,MAAM,0DAAqD,CAAA;AACrE,QAAA;AAAA,MACF;AAEA,MAAA,GAAA,CAAI,IAAA,CAAK,SAAS,YAAY,CAAA;AAE9B,MAAA,MAAM,GAAA,GAAM,IAAI,IAAA,EAAK;AACrB,MAAA,GAAA,CAAI,OAAO,KAAA,CAAM,QAAA;AACjB,MAAA,GAAA,CAAI,iBAAiB,KAAA,CAAM,OAAA;AAC3B,MAAA,GAAA,CAAI,UAAA,GAAa,KAAK,UAAA,IAAc,IAAA;AACpC,MAAA,GAAA,CAAI,eAAA,GAAkB,KAAK,eAAA,IAAmB,CAAA;AAE9C,MAAA,GAAA,CAAI,UAAU,MAAM;AAClB,QAAA,GAAA,CAAI,IAAA,CAAK,SAAS,WAAW,CAAA;AAAA,MAC/B,CAAA;AACA,MAAA,GAAA,CAAI,QAAQ,MAAM;AAChB,QAAA,GAAA,CAAI,IAAA,CAAK,SAAS,QAAQ,CAAA;AAC1B,QAAA,QAAA,EAAS;AAAA,MACX,CAAA;AACA,MAAA,GAAA,CAAI,OAAA,GAAU,CAAC,CAAA,KAAM;AACnB,QAAA,MAAM,IAAA,GAAO,SAAA,CAAU,CAAA,CAAE,KAAK,CAAA,IAAK,QAAA;AACnC,QAAA,MAAM,GAAA,GAAwB;AAAA,UAC5B,IAAA;AAAA,UACA,OAAA,EAAS,CAAA,CAAE,OAAA,IAAW,CAAA,kBAAA,EAAqB,EAAE,KAAK,CAAA;AAAA,SACpD;AACA,QAAA,GAAA,CAAI,IAAA,CAAK,SAAS,GAAG,CAAA;AAAA,MACvB,CAAA;AACA,MAAA,GAAA,CAAI,QAAA,GAAW,CAAC,CAAA,KAAM;AACpB,QAAA,KAAA,IAAS,CAAA,GAAI,EAAE,WAAA,EAAa,CAAA,GAAI,EAAE,OAAA,CAAQ,MAAA,EAAQ,KAAK,CAAA,EAAG;AACxD,UAAA,MAAM,GAAA,GAAM,CAAA,CAAE,OAAA,CAAQ,CAAC,CAAA;AACvB,UAAA,MAAM,GAAA,GAAM,IAAI,CAAC,CAAA;AACjB,UAAA,MAAM,OAAO,GAAA,CAAI,UAAA;AACjB,UAAA,IAAI,CAAC,gBAAA,EAAkB,gBAAA,GAAmB,YAAA,EAAa;AACvD,UAAA,IAAI,IAAI,OAAA,EAAS;AACf,YAAA,GAAA,CAAI,IAAA,CAAK,OAAA,EAAS,IAAA,EAAM,gBAAA,EAAkB,IAAI,UAAU,CAAA;AACxD,YAAA,gBAAA,GAAmB,IAAA;AAAA,UACrB,CAAA,MAAO;AACL,YAAA,GAAA,CAAI,IAAA,CAAK,SAAA,EAAW,IAAA,EAAM,gBAAgB,CAAA;AAAA,UAC5C;AAAA,QACF;AAAA,MACF,CAAA;AAEA,MAAA,IAAI,MAAM,MAAA,EAAQ;AAChB,QAAA,KAAA,CAAM,MAAA,CAAO,gBAAA,CAAiB,OAAA,EAAS,MAAM;AAC3C,UAAA,GAAA,CAAI,KAAA,EAAM;AAAA,QACZ,CAAC,CAAA;AAAA,MACH;AAEA,MAAA,QAAA,GAAW,GAAA;AACX,MAAA,IAAI;AACF,QAAA,GAAA,CAAI,KAAA,EAAM;AAAA,MACZ,SAAS,KAAA,EAAO;AACd,QAAA,MAAM,GAAA,GAAwB;AAAA,UAC5B,IAAA,EAAM,QAAA;AAAA,UACN,OAAA,EAAS,yCAAA;AAAA,UACT;AAAA,SACF;AACA,QAAA,GAAA,CAAI,IAAA,CAAK,SAAS,GAAG,CAAA;AACrB,QAAA,QAAA,EAAS;AACT,QAAA,MAAM,GAAA;AAAA,MACR;AAAA,IACF,CAAA;AAAA,IACA,MAAM,IAAA,GAAsB;AAC1B,MAAA,IAAI,CAAC,QAAA,EAAU;AACf,MAAA,GAAA,CAAI,IAAA,CAAK,SAAS,SAAS,CAAA;AAC3B,MAAA,QAAA,CAAS,IAAA,EAAK;AAAA,IAChB,CAAA;AAAA,IACA,KAAA,GAAc;AACZ,MAAA,IAAI,CAAC,QAAA,EAAU;AACf,MAAA,QAAA,CAAS,KAAA,EAAM;AAAA,IACjB;AAAA,GACF;AACF;AA1GgBA,wBAAA,CAAA,qBAAA,EAAA,uBAAA,CAAA;AC9DhB,IAAM,QAAA,GAAwB;AAAA,EAC5B,QAAA,EAAU,IAAA;AAAA,EACV,QAAA,EAAU,IAAA;AAAA,EACV,QAAA,EAAU,IAAA;AAAA,EACV,OAAA,EAAS;AACX,CAAA;AAUO,IAAM,iBAAiBK,cAAA,EAAmB;AAAA,EAC/CC,kBAAA;AAAA,IACE,CAAC,GAAA,MAAS;AAAA,MACR,GAAG,QAAA;AAAA,MACH,6BAAaN,wBAAA,CAAA,CAAC,QAAA,KAAa,IAAI,EAAE,QAAA,EAAU,CAAA,EAA9B,aAAA,CAAA;AAAA,MACb,6BAAaA,wBAAA,CAAA,CAAC,QAAA,KAAa,IAAI,EAAE,QAAA,EAAU,CAAA,EAA9B,aAAA,CAAA;AAAA,MACb,6BAAaA,wBAAA,CAAA,CAAC,QAAA,KAAa,IAAI,EAAE,QAAA,EAAU,CAAA,EAA9B,aAAA,CAAA;AAAA,MACb,4BAAYA,wBAAA,CAAA,CAAC,OAAA,KAAY,IAAI,EAAE,OAAA,EAAS,CAAA,EAA5B,YAAA,CAAA;AAAA,MACZ,uBAAOA,wBAAA,CAAA,MAAM,GAAA,CAAI,EAAE,GAAG,QAAA,EAAU,CAAA,EAAzB,OAAA;AAAA,KACT,CAAA;AAAA,IACA;AAAA,MACE,IAAA,EAAM,qBAAA;AAAA,MACN,OAAA,EAASO,4BAAA;AAAA,QAAkB,MACzB,OAAO,MAAA,KAAW,WAAA,GACb,SACD,MAAA,CAAO;AAAA;AACb;AACF;AAEJ;AC5CO,SAAS,YAAY,MAAA,EAAoC;AAC9D,EAAA,MAAM,CAAC,KAAA,EAAO,QAAQ,CAAA,GAAIC,eAAS,CAAC,CAAA;AACpC,EAAA,MAAM,GAAA,GAAMC,aAAsB,IAAI,CAAA;AAEtC,EAAAC,eAAA,CAAU,MAAM;AACd,IAAA,IAAI,CAAC,MAAA,EAAQ;AACX,MAAA,QAAA,CAAS,CAAC,CAAA;AACV,MAAA,OAAO,MAAA;AAAA,IACT;AACA,IAAA,MAAM,EAAA,GACH,MAAA,CAA6D,YAAA,IAC7D,MAAA,CAAmE,kBAAA;AACtE,IAAA,IAAI,CAAC,IAAI,OAAO,MAAA;AAChB,IAAA,MAAM,GAAA,GAAM,IAAI,EAAA,EAAG;AACnB,IAAA,MAAM,MAAA,GAAS,GAAA,CAAI,uBAAA,CAAwB,MAAM,CAAA;AACjD,IAAA,MAAM,QAAA,GAAW,IAAI,cAAA,EAAe;AACpC,IAAA,QAAA,CAAS,OAAA,GAAU,IAAA;AACnB,IAAA,QAAA,CAAS,qBAAA,GAAwB,GAAA;AACjC,IAAA,MAAA,CAAO,QAAQ,QAAQ,CAAA;AACvB,IAAA,MAAM,GAAA,GAAM,IAAI,YAAA,CAAa,QAAA,CAAS,OAAO,CAAA;AAE7C,IAAA,MAAM,uBAAOV,wBAAA,CAAA,MAAY;AACvB,MAAA,QAAA,CAAS,uBAAuB,GAAG,CAAA;AACnC,MAAA,IAAI,GAAA,GAAM,CAAA;AACV,MAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,GAAA,CAAI,MAAA,EAAQ,CAAA,IAAK,CAAA,EAAG,GAAA,IAAO,GAAA,CAAI,CAAC,CAAA,GAAI,GAAA,CAAI,CAAC,CAAA;AAC7D,MAAA,MAAM,GAAA,GAAM,IAAA,CAAK,IAAA,CAAK,GAAA,GAAM,IAAI,MAAM,CAAA;AAEtC,MAAA,QAAA,CAAS,IAAA,CAAK,GAAA,CAAI,CAAA,EAAG,GAAA,GAAM,GAAG,CAAC,CAAA;AAC/B,MAAA,GAAA,CAAI,OAAA,GAAU,sBAAsB,IAAI,CAAA;AAAA,IAC1C,CAAA,EARa,MAAA,CAAA;AASb,IAAA,IAAA,EAAK;AAEL,IAAA,OAAO,MAAM;AACX,MAAA,IAAI,GAAA,CAAI,OAAA,IAAW,IAAA,EAAM,oBAAA,CAAqB,IAAI,OAAO,CAAA;AACzD,MAAA,GAAA,CAAI,OAAA,GAAU,IAAA;AACd,MAAA,MAAA,CAAO,UAAA,EAAW;AAClB,MAAA,QAAA,CAAS,UAAA,EAAW;AACpB,MAAA,KAAK,IAAI,KAAA,EAAM;AAAA,IACjB,CAAA;AAAA,EACF,CAAA,EAAG,CAAC,MAAM,CAAC,CAAA;AAEX,EAAA,OAAO,KAAA;AACT;AA1CgBA,wBAAA,CAAA,WAAA,EAAA,aAAA,CAAA;;;ACGhB,IAAM,YAAA,GAAuC;AAAA,EAC3C,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA,EACJ,EAAA,EAAI;AACN,CAAA;AAWO,SAAS,OAAA,CACd,IAAA,EACA,KAAA,GAAgC,YAAA,EACZ;AACpB,EAAA,IAAI,CAAC,MAAM,OAAO,MAAA;AAClB,EAAA,MAAM,OAAA,GAAU,KAAK,IAAA,EAAK;AAC1B,EAAA,IAAI,CAAC,SAAS,OAAO,MAAA;AACrB,EAAA,IAAI,OAAA,CAAQ,QAAA,CAAS,GAAG,CAAA,EAAG,OAAO,OAAA;AAClC,EAAA,MAAM,KAAA,GAAQ,QAAQ,WAAA,EAAY;AAClC,EAAA,OAAO,KAAA,CAAM,KAAK,CAAA,IAAK,CAAA,EAAG,KAAK,CAAA,CAAA,EAAI,KAAA,CAAM,aAAa,CAAA,CAAA;AACxD;AAVgBA,wBAAA,CAAA,OAAA,EAAA,SAAA,CAAA;AAwBT,SAAS,sBAAsB,IAAA,EAI3B;AACT,EAAA,OACE,QAAQ,IAAA,CAAK,QAAQ,KACrB,OAAA,CAAQ,IAAA,CAAK,KAAK,CAAA,IAClB,OAAA,CAAQ,KAAK,IAAI,CAAA,IACjB,QAAQ,OAAO,SAAA,KAAc,cAAc,SAAA,CAAU,QAAA,GAAW,IAAI,CAAA,IACpE,OAAA;AAEJ;AAZgBA,wBAAA,CAAA,qBAAA,EAAA,uBAAA,CAAA;;;AC9CT,SAAS,oBAAoB,QAAA,EAA2B;AAC7D,EAAA,MAAM,QAAQ,cAAA,EAAe;AAC7B,EAAA,MAAM,SAASW,sBAAA,EAAkB;AACjC,EAAA,OAAO,qBAAA,CAAsB;AAAA,IAC3B,QAAA;AAAA,IACA,OAAO,KAAA,CAAM,QAAA;AAAA,IACb,IAAA,EAAM;AAAA,GACP,CAAA;AACH;AARgBX,wBAAA,CAAA,mBAAA,EAAA,qBAAA,CAAA;;;ACPhB,IAAM,MAAM,eAAA,EAAgB;AAiBrB,SAAS,oBAAA,CACd,MAAA,GAAqC,EAAC,EACV;AAC5B,EAAA,MAAM,QAAQ,cAAA,EAAe;AAC7B,EAAA,MAAM,QAAA,GAAW,mBAAA,CAAoB,MAAA,CAAO,QAAQ,CAAA;AACpD,EAAA,MAAM,MAAA,GAASY,aAAA;AAAA,IACb,MAAM,MAAA,CAAO,MAAA,IAAU,qBAAA,EAAsB;AAAA,IAC7C,CAAC,OAAO,MAAM;AAAA,GAChB;AAEA,EAAA,MAAM,CAAC,KAAA,EAAO,QAAQ,CAAA,GAAIC,gBAAA,CAAW,SAAS,aAAa,CAAA;AAC3D,EAAA,MAAM,CAAC,MAAA,EAAQ,SAAS,CAAA,GAAIL,eAA6B,IAAI,CAAA;AAC7D,EAAA,MAAM,KAAA,GAAQ,YAAY,MAAM,CAAA;AAIhC,EAAA,MAAM,KAAA,GAAQC,aAAO,MAAM,CAAA;AAC3B,EAAA,KAAA,CAAM,OAAA,GAAU,MAAA;AAGhB,EAAAC,gBAAU,MAAM;AACd,IAAA,GAAA,CAAI,OAAO,KAAA,CAAM,WAAA,EAAa,EAAE,QAAA,EAAU,MAAA,CAAO,IAAI,CAAA;AACrD,IAAA,MAAM,IAAA,GAAO;AAAA,MACX,MAAA,CAAO,EAAA,CAAG,SAAA,EAAW,CAAC,MAAM,SAAA,KAAc;AACxC,QAAA,GAAA,CAAI,MAAA,CAAO,MAAM,SAAA,EAAW,EAAE,KAAK,IAAA,CAAK,MAAA,EAAQ,WAAW,CAAA;AAC3D,QAAA,QAAA,CAAS,EAAE,IAAA,EAAM,SAAA,EAAW,IAAA,EAAM,WAAW,CAAA;AAC7C,QAAA,MAAM,GAAA,GAAe;AAAA,UACnB,EAAA,EAAI,SAAA;AAAA,UACJ,IAAA;AAAA,UACA,OAAA,EAAS,KAAA;AAAA,UACT,SAAA,EAAW,KAAK,GAAA;AAAI,SACtB;AACA,QAAA,KAAA,CAAM,OAAA,CAAQ,SAAA,GAAY,IAAA,EAAM,GAAG,CAAA;AAAA,MACrC,CAAC,CAAA;AAAA,MACD,OAAO,EAAA,CAAG,OAAA,EAAS,CAAC,IAAA,EAAM,WAAW,UAAA,KAAe;AAClD,QAAA,GAAA,CAAI,MAAA,CAAO,KAAK,OAAA,EAAS;AAAA,UACvB,KAAK,IAAA,CAAK,MAAA;AAAA,UACV,SAAA;AAAA,UACA;AAAA,SACD,CAAA;AACD,QAAA,QAAA,CAAS,EAAE,IAAA,EAAM,OAAA,EAAS,IAAA,EAAM,SAAA,EAAW,YAAY,CAAA;AACvD,QAAA,MAAM,GAAA,GAAe;AAAA,UACnB,EAAA,EAAI,SAAA;AAAA,UACJ,IAAA;AAAA,UACA,OAAA,EAAS,IAAA;AAAA,UACT,UAAA;AAAA,UACA,SAAA,EAAW,KAAK,GAAA,EAAI;AAAA,UACpB,OAAA,EAAS,KAAK,GAAA;AAAI,SACpB;AACA,QAAA,KAAA,CAAM,OAAA,CAAQ,OAAA,GAAU,IAAA,EAAM,GAAG,CAAA;AAAA,MACnC,CAAC,CAAA;AAAA,MACD,MAAA,CAAO,EAAA,CAAG,OAAA,EAAS,CAAC,GAAA,KAAQ;AAC1B,QAAA,GAAA,CAAI,KAAA,CAAM,KAAA,CAAM,cAAA,EAAgB,GAAG,CAAA;AACnC,QAAA,QAAA,CAAS,EAAE,IAAA,EAAM,OAAA,EAAS,KAAA,EAAO,KAAK,CAAA;AACtC,QAAA,KAAA,CAAM,OAAA,CAAQ,UAAU,GAAG,CAAA;AAAA,MAC7B,CAAC,CAAA;AAAA,MACD,MAAA,CAAO,EAAA,CAAG,OAAA,EAAS,CAAC,CAAA,KAAM;AACxB,QAAA,GAAA,CAAI,MAAA,CAAO,KAAA,CAAM,OAAA,EAAS,CAAC,CAAA;AAC3B,QAAA,IAAI,MAAM,WAAA,EAAa;AACrB,UAAA,QAAA,CAAS,EAAE,IAAA,EAAM,SAAA,EAAW,CAAA;AAC5B,UAAA,KAAA,CAAM,QAAQ,OAAA,IAAU;AACxB,UAAA,SAAA,CAAU,MAAA,CAAO,SAAA,IAAY,IAAK,IAAI,CAAA;AAAA,QACxC,CAAA,MAAA,IAAW,MAAM,QAAA,EAAU;AACzB,UAAA,QAAA,CAAS,EAAE,IAAA,EAAM,SAAA,EAAW,CAAA;AAC5B,UAAA,KAAA,CAAM,QAAQ,MAAA,IAAS;AACvB,UAAA,SAAA,CAAU,IAAI,CAAA;AAAA,QAChB;AAAA,MACF,CAAC;AAAA,KACH;AACA,IAAA,OAAO,MAAM;AACX,MAAA,IAAA,CAAK,OAAA,CAAQ,CAAC,GAAA,KAAQ,GAAA,EAAK,CAAA;AAAA,IAC7B,CAAA;AAAA,EACF,CAAA,EAAG,CAAC,MAAM,CAAC,CAAA;AAGX,EAAA,MAAM,YAAA,GAAeD,aAAsB,IAAI,CAAA;AAC/C,EAAA,MAAM,QAAA,GAAWA,aAAsB,IAAI,CAAA;AAC3C,EAAAC,gBAAU,MAAM;AACd,IAAA,IAAI,KAAA,CAAM,MAAA,KAAW,WAAA,EAAa,OAAO,MAAA;AACzC,IAAA,MAAM,EAAE,WAAW,KAAA,EAAO,gBAAA,GAAmB,MAAK,GAAI,MAAA,CAAO,YAAY,EAAC;AAC1E,IAAA,IAAI,KAAA,EAAO;AACT,MAAA,QAAA,CAAS,OAAA,GAAU,MAAA,CAAO,UAAA,CAAW,MAAM;AACzC,QAAA,GAAA,CAAI,MAAA,CAAO,MAAM,2BAA2B,CAAA;AAC5C,QAAA,KAAK,OAAO,IAAA,EAAK;AAAA,MACnB,GAAG,KAAK,CAAA;AAAA,IACV;AACA,IAAA,IAAI,SAAA,EAAW;AACb,MAAA,MAAM,aAAA,GAAgB,MAAA,CAAO,WAAA,CAAY,MAAM;AAC7C,QAAA,IAAI,QAAQ,gBAAA,EAAkB;AAC5B,UAAA,IAAI,YAAA,CAAa,WAAW,IAAA,EAAM;AAChC,YAAA,YAAA,CAAa,OAAA,GAAU,MAAA,CAAO,UAAA,CAAW,MAAM;AAC7C,cAAA,GAAA,CAAI,MAAA,CAAO,MAAM,2BAA2B,CAAA;AAC5C,cAAA,KAAK,OAAO,IAAA,EAAK;AAAA,YACnB,GAAG,SAAS,CAAA;AAAA,UACd;AAAA,QACF,CAAA,MAAA,IAAW,YAAA,CAAa,OAAA,IAAW,IAAA,EAAM;AACvC,UAAA,YAAA,CAAa,aAAa,OAAO,CAAA;AACjC,UAAA,YAAA,CAAa,OAAA,GAAU,IAAA;AAAA,QACzB;AAAA,MACF,GAAG,GAAG,CAAA;AACN,MAAA,OAAO,MAAM;AACX,QAAA,aAAA,CAAc,aAAa,CAAA;AAC3B,QAAA,IAAI,YAAA,CAAa,OAAA,IAAW,IAAA,EAAM,YAAA,CAAa,aAAa,OAAO,CAAA;AACnE,QAAA,YAAA,CAAa,OAAA,GAAU,IAAA;AACvB,QAAA,IAAI,QAAA,CAAS,OAAA,IAAW,IAAA,EAAM,YAAA,CAAa,SAAS,OAAO,CAAA;AAC3D,QAAA,QAAA,CAAS,OAAA,GAAU,IAAA;AAAA,MACrB,CAAA;AAAA,IACF;AACA,IAAA,OAAO,MAAM;AACX,MAAA,IAAI,QAAA,CAAS,OAAA,IAAW,IAAA,EAAM,YAAA,CAAa,SAAS,OAAO,CAAA;AAC3D,MAAA,QAAA,CAAS,OAAA,GAAU,IAAA;AAAA,IACrB,CAAA;AAAA,EACF,CAAA,EAAG,CAAC,KAAA,CAAM,MAAA,EAAQ,OAAO,QAAA,EAAU,KAAA,EAAO,MAAM,CAAC,CAAA;AAEjD,EAAA,MAAM,KAAA,GAAQI,kBAAY,YAAY;AACpC,IAAA,IAAI,KAAA,CAAM,MAAA,KAAW,WAAA,IAAe,KAAA,CAAM,WAAW,UAAA,EAAY;AACjE,IAAA,QAAA,CAAS,EAAE,IAAA,EAAM,OAAA,EAAS,CAAA;AAC1B,IAAA,IAAI;AACF,MAAA,MAAM,OAAO,KAAA,CAAM;AAAA,QACjB,QAAA;AAAA,QACA,OAAA,EAAS,OAAO,OAAA,IAAW,IAAA;AAAA,QAC3B,QAAA,EAAU,MAAA,CAAO,QAAA,IAAY,KAAA,CAAM,QAAA,IAAY,KAAA;AAAA,OAChD,CAAA;AAAA,IACH,SAAS,KAAA,EAAO;AAEd,MAAA,GAAA,CAAI,MAAA,CAAO,IAAA,CAAK,aAAA,EAAe,KAAK,CAAA;AAAA,IACtC;AAAA,EACF,CAAA,EAAG,CAAC,MAAA,EAAQ,QAAA,EAAU,MAAA,CAAO,OAAA,EAAS,MAAA,CAAO,QAAA,EAAU,KAAA,CAAM,QAAA,EAAU,KAAA,CAAM,MAAM,CAAC,CAAA;AAEpF,EAAA,MAAM,IAAA,GAAOA,kBAAY,YAAY;AACnC,IAAA,IAAI,KAAA,CAAM,MAAA,KAAW,MAAA,IAAU,KAAA,CAAM,WAAW,UAAA,EAAY;AAC5D,IAAA,QAAA,CAAS,EAAE,IAAA,EAAM,MAAA,EAAQ,CAAA;AACzB,IAAA,MAAM,OAAO,IAAA,EAAK;AAAA,EACpB,CAAA,EAAG,CAAC,MAAA,EAAQ,KAAA,CAAM,MAAM,CAAC,CAAA;AAEzB,EAAA,MAAM,KAAA,GAAQA,kBAAY,MAAM;AAC9B,IAAA,MAAA,CAAO,KAAA,EAAM;AACb,IAAA,QAAA,CAAS,EAAE,IAAA,EAAM,OAAA,EAAS,CAAA;AAAA,EAC5B,CAAA,EAAG,CAAC,MAAM,CAAC,CAAA;AAEX,EAAA,MAAM,MAAA,GAASA,kBAAY,YAAY;AACrC,IAAA,IAAI,KAAA,CAAM,MAAA,KAAW,WAAA,IAAe,KAAA,CAAM,WAAW,UAAA,EAAY;AAC/D,MAAA,MAAM,IAAA,EAAK;AAAA,IACb,CAAA,MAAO;AACL,MAAA,MAAM,KAAA,EAAM;AAAA,IACd;AAAA,EACF,GAAG,CAAC,KAAA,CAAM,MAAA,EAAQ,KAAA,EAAO,IAAI,CAAC,CAAA;AAE9B,EAAA,MAAM,KAAA,GAAQA,kBAAY,MAAM;AAC9B,IAAA,QAAA,CAAS,EAAE,IAAA,EAAM,OAAA,EAAS,CAAA;AAAA,EAC5B,CAAA,EAAG,EAAE,CAAA;AAEL,EAAA,MAAM,UAAA,GAAaF,aAAA;AAAA,IACjB,MAAO,MAAM,QAAA,CAAS,MAAA,KAAW,IAAI,gBAAA,GAAmB,eAAA,CAAgB,MAAM,QAAQ,CAAA;AAAA,IACtF,CAAC,MAAM,QAAQ;AAAA,GACjB;AAEA,EAAA,OAAO;AAAA,IACL,QAAQ,KAAA,CAAM,MAAA;AAAA,IACd,aAAa,MAAA,CAAO,WAAA;AAAA,IACpB,UAAA;AAAA,IACA,OAAO,KAAA,CAAM,KAAA;AAAA,IACb,KAAA;AAAA,IACA,KAAA;AAAA,IACA,IAAA;AAAA,IACA,KAAA;AAAA,IACA,MAAA;AAAA,IACA;AAAA,GACF;AACF;AAzKgBZ,wBAAA,CAAA,oBAAA,EAAA,sBAAA,CAAA;;;ACAT,SAAS,aAAa,MAAA,EAAgD;AAC3E,EAAA,MAAM,EAAE,KAAA,EAAO,QAAA,EAAU,YAAY,GAAA,EAAK,GAAG,MAAK,GAAI,MAAA;AAItD,EAAA,MAAM,QAAA,GAAWS,aAAO,KAAK,CAAA;AAC7B,EAAA,QAAA,CAAS,OAAA,GAAU,KAAA;AACnB,EAAA,MAAM,WAAA,GAAcA,aAAO,QAAQ,CAAA;AACnC,EAAA,WAAA,CAAY,OAAA,GAAU,QAAA;AAEtB,EAAA,MAAM,MAAM,oBAAA,CAAqB;AAAA,IAC/B,GAAG,IAAA;AAAA,IACH,OAAA,4CAAU,IAAA,KAAS;AACjB,MAAA,MAAM,KAAA,GAAQ,eAAe,IAAI,CAAA;AACjC,MAAA,IAAI,CAAC,KAAA,EAAO;AACZ,MAAA,MAAM,OAAO,QAAA,CAAS,OAAA;AACtB,MAAA,MAAM,IAAA,GAAO,OAAO,CAAA,EAAG,IAAI,GAAG,SAAS,CAAA,EAAG,KAAK,CAAA,CAAA,GAAK,KAAA;AACpD,MAAA,WAAA,CAAY,QAAQ,IAAI,CAAA;AAAA,IAC1B,CAAA,EANS,SAAA;AAAA,GAOV,CAAA;AAED,EAAAC,gBAAU,MAAM;AACd,IAAA,QAAA,CAAS,OAAA,GAAU,KAAA;AAAA,EACrB,CAAA,EAAG,CAAC,KAAK,CAAC,CAAA;AAEV,EAAA,OAAO;AAAA,IACL,GAAG,GAAA;AAAA,IACH,iBAAiB,GAAA,CAAI;AAAA,GACvB;AACF;AA7BgBV,wBAAA,CAAA,YAAA,EAAA,cAAA,CAAA;AChBhB,IAAM,QAAA,uBAAe,GAAA,CAAI,CAAC,SAAS,MAAA,EAAQ,KAAA,EAAO,MAAA,EAAQ,KAAK,CAAC,CAAA;AAEhE,SAAS,WAAW,KAAA,EAA2D;AAC7E,EAAA,MAAM,KAAA,GAAQ,KAAA,CACX,WAAA,EAAY,CACZ,KAAA,CAAM,GAAG,CAAA,CACT,GAAA,CAAI,CAAC,CAAA,KAAM,CAAA,CAAE,IAAA,EAAM,CAAA;AACtB,EAAA,MAAM,IAAA,uBAAW,GAAA,EAAY;AAC7B,EAAA,IAAI,IAAA,GAAsB,IAAA;AAC1B,EAAA,KAAA,MAAW,QAAQ,KAAA,EAAO;AACxB,IAAA,IAAI,QAAA,CAAS,GAAA,CAAI,IAAI,CAAA,EAAG;AACtB,MAAA,IAAA,CAAK,GAAA,CAAI,IAAA,KAAS,KAAA,GAAQ,MAAA,GAAS,IAAI,CAAA;AAAA,IACzC,CAAA,MAAO;AACL,MAAA,IAAA,GAAO,IAAA;AAAA,IACT;AAAA,EACF;AACA,EAAA,OAAO,EAAE,MAAM,IAAA,EAAK;AACtB;AAfSA,wBAAA,CAAA,UAAA,EAAA,YAAA,CAAA;AAiBT,SAAS,OAAA,CAAQ,CAAA,EAAkB,IAAA,EAAmB,IAAA,EAA8B;AAClF,EAAA,IAAI,KAAK,GAAA,CAAI,OAAO,CAAA,KAAM,CAAA,CAAE,UAAU,OAAO,KAAA;AAC7C,EAAA,IAAI,KAAK,GAAA,CAAI,MAAM,CAAA,KAAM,CAAA,CAAE,SAAS,OAAO,KAAA;AAC3C,EAAA,IAAI,KAAK,GAAA,CAAI,KAAK,CAAA,KAAM,CAAA,CAAE,QAAQ,OAAO,KAAA;AAEzC,EAAA,IAAI,IAAA,CAAK,GAAA,CAAI,MAAM,CAAA,MAAO,CAAA,CAAE,WAAY,CAAC,CAAA,CAAE,OAAA,IAAW,KAAA,CAAA,EAAS,OAAO,KAAA;AACtE,EAAA,IAAI,QAAQ,CAAA,CAAE,GAAA,CAAI,WAAA,EAAY,KAAM,MAAM,OAAO,KAAA;AACjD,EAAA,OAAO,IAAA;AACT;AARSA,wBAAA,CAAA,OAAA,EAAA,SAAA,CAAA;AAeF,SAAS,aAAA,CACd,aACA,IAAA,EACM;AACN,EAAA,MAAM,EAAE,GAAA,EAAK,OAAA,GAAU,IAAA,EAAK,GAAI,IAAA;AAEhC,EAAAU,gBAAU,MAAM;AACd,IAAA,IAAI,CAAC,OAAA,IAAW,OAAO,MAAA,KAAW,aAAa,OAAO,MAAA;AACtD,IAAA,MAAM,EAAE,IAAA,EAAM,IAAA,EAAK,GAAI,WAAW,GAAG,CAAA;AAErC,IAAA,MAAM,MAAA,6CAAU,CAAA,KAA2B;AACzC,MAAA,IAAI,EAAE,MAAA,EAAQ;AACd,MAAA,IAAI,CAAC,IAAA,IAAQ,IAAA,CAAK,IAAA,KAAS,CAAA,EAAG;AAC9B,MAAA,MAAM,SAAS,CAAA,CAAE,MAAA;AACjB,MAAA,MAAM,UACJ,MAAA,EAAQ,OAAA,KAAY,WACpB,MAAA,EAAQ,OAAA,KAAY,cACpB,MAAA,EAAQ,iBAAA;AACV,MAAA,IAAI,OAAA,IAAW,IAAA,CAAK,IAAA,KAAS,CAAA,EAAG;AAChC,MAAA,IAAI,CAAC,OAAA,CAAQ,CAAA,EAAG,IAAA,EAAM,IAAI,CAAA,EAAG;AAC7B,MAAA,IAAI,WAAA,CAAY,MAAA,KAAW,WAAA,IAAe,WAAA,CAAY,WAAW,UAAA,EAAY;AAC7E,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,KAAK,YAAY,KAAA,EAAM;AAAA,IACzB,CAAA,EAbe,QAAA,CAAA;AAcf,IAAA,MAAM,IAAA,6CAAQ,CAAA,KAA2B;AACvC,MAAA,IAAI,CAAC,OAAA,CAAQ,CAAA,EAAG,IAAA,EAAM,IAAI,CAAA,IAAK,IAAA,IAAQ,CAAA,CAAE,GAAA,CAAI,WAAA,EAAY,KAAM,IAAA,EAAM;AACrE,MAAA,IAAI,WAAA,CAAY,MAAA,KAAW,WAAA,IAAe,WAAA,CAAY,WAAW,UAAA,EAAY;AAC7E,MAAA,KAAK,YAAY,IAAA,EAAK;AAAA,IACxB,CAAA,EAJa,MAAA,CAAA;AAMb,IAAA,MAAA,CAAO,gBAAA,CAAiB,WAAW,MAAM,CAAA;AACzC,IAAA,MAAA,CAAO,gBAAA,CAAiB,SAAS,IAAI,CAAA;AACrC,IAAA,OAAO,MAAM;AACX,MAAA,MAAA,CAAO,mBAAA,CAAoB,WAAW,MAAM,CAAA;AAC5C,MAAA,MAAA,CAAO,mBAAA,CAAoB,SAAS,IAAI,CAAA;AAAA,IAC1C,CAAA;AAAA,EACF,CAAA,EAAG,CAAC,OAAA,EAAS,GAAA,EAAK,WAAW,CAAC,CAAA;AAChC;AArCgBV,wBAAA,CAAA,aAAA,EAAA,eAAA,CAAA;ACRT,SAAS,cAAA,CAAe;AAAA,EAC7B,KAAA;AAAA,EACA,QAAA;AAAA,EACA,MAAA;AAAA,EACA,QAAA;AAAA,EACA,UAAA;AAAA,EACA,WAAA,GAAc,wCAAA;AAAA,EACd,IAAA,GAAO,CAAA;AAAA,EACP,QAAA;AAAA,EACA,WAAA,GAAc,IAAA;AAAA,EACd,SAAA,GAAY,IAAA;AAAA,EACZ,SAAA;AAAA,EACA;AACF,CAAA,EAA4C;AAC1C,EAAA,MAAM,MAAM,YAAA,CAAa;AAAA,IACvB,KAAA;AAAA,IACA,QAAA;AAAA,IACA,MAAA;AAAA,IACA;AAAA,GACD,CAAA;AAED,EAAA,aAAA,CAAc,GAAA,EAAK;AAAA,IACjB,GAAA,EAAK,YAAY,GAAA,IAAO,KAAA;AAAA,IACxB,OAAA,EAAS,CAAC,CAAC,UAAA,IAAc,WAAW,OAAA,KAAY;AAAA,GACjD,CAAA;AAED,EAAA,uBACEN,eAAAA,CAAC,KAAA,EAAA,EAAI,WAAWC,MAAAA,CAAG,qBAAA,EAAuB,SAAS,CAAA,EACjD,QAAA,EAAA;AAAA,oBAAAD,eAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,UAAA,EACb,QAAA,EAAA;AAAA,sBAAAE,cAAAA;AAAA,QAAC,UAAA;AAAA,QAAA;AAAA,UACC,KAAA;AAAA,UACA,UAAU,CAAC,CAAA,KAAM,QAAA,CAAS,CAAA,CAAE,OAAO,KAAK,CAAA;AAAA,UACxC,WAAA;AAAA,UACA,IAAA;AAAA,UACA,QAAA;AAAA,UACA,SAAA,EAAWD,MAAAA;AAAA,YACT,gFAAA;AAAA,YACA,mCAAA;AAAA,YACA,yEAAA;AAAA,YACA,iDAAA;AAAA,YACA;AAAA;AACF;AAAA,OACF;AAAA,MACC,WAAA,IAAe,IAAI,UAAA,CAAW,OAAA,oBAC7BD,eAAAA,CAAC,KAAA,EAAA,EAAI,WAAU,+DAAA,EAAgE,QAAA,EAAA;AAAA,QAAA,SAAA;AAAA,QAC1E,IAAI,UAAA,CAAW;AAAA,OAAA,EACpB;AAAA,KAAA,EAEJ,CAAA;AAAA,oBAEAA,eAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,yBAAA,EACb,QAAA,EAAA;AAAA,sBAAAE,cAAAA;AAAA,QAAC,eAAA;AAAA,QAAA;AAAA,UACC,QAAQ,GAAA,CAAI,MAAA;AAAA,UACZ,aAAa,GAAA,CAAI,WAAA;AAAA,UACjB,OAAA,EAAS,MAAM,KAAK,GAAA,CAAI,eAAA,EAAgB;AAAA,UACxC,IAAA,EAAK,IAAA;AAAA,UACL;AAAA;AAAA,OACF;AAAA,MACC,SAAA,oBAAaA,cAAAA,CAAC,QAAA,EAAA,EAAS,KAAA,EAAO,IAAI,KAAA,EAAO,IAAA,EAAM,EAAA,EAAI,MAAA,EAAQ,EAAA,EAAI,CAAA;AAAA,MAC/D,UAAA,oBAAcA,cAAAA,CAAC,cAAA,EAAA,EAAe,OAAO,UAAA,CAAW,GAAA,EAAK,WAAU,SAAA,EAAU;AAAA,KAAA,EAC5E,CAAA;AAAA,oBAEAA,cAAAA,CAAC,WAAA,EAAA,EAAY,KAAA,EAAO,IAAI,KAAA,EAAO;AAAA,GAAA,EACjC,CAAA;AAEJ;AAjEgBI,wBAAA,CAAA,cAAA,EAAA,gBAAA,CAAA","file":"chunk-ADEN3UA4.cjs","sourcesContent":["'use client';\n\nimport type * as React from 'react';\n\nimport { Loader2, Mic, MicOff } from 'lucide-react';\nimport type { CSSProperties, ReactNode } from 'react';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nimport type { RecognitionStatus } from '../types';\n\nexport interface DictationButtonProps {\n status: RecognitionStatus;\n onClick: () => void;\n isSupported?: boolean;\n size?: 'sm' | 'md' | 'lg';\n className?: string;\n style?: CSSProperties;\n ariaLabel?: string;\n /** Override icon for the idle state. */\n idleIcon?: ReactNode;\n /** Override icon for the listening state. */\n listeningIcon?: ReactNode;\n /** Disable without unmounting. */\n disabled?: boolean;\n}\n\nconst SIZE_CLS: Record<NonNullable<DictationButtonProps['size']>, string> = {\n sm: 'h-8 w-8 [&_svg]:h-4 [&_svg]:w-4',\n md: 'h-10 w-10 [&_svg]:h-5 [&_svg]:w-5',\n lg: 'h-12 w-12 [&_svg]:h-6 [&_svg]:w-6',\n};\n\n/**\n * Round microphone button. Cycles icon by status; shows a soft pulse\n * ring when listening. ARIA-correct so screen readers announce\n * \"recording\" vs \"start dictation\".\n */\nexport function DictationButton({\n status,\n onClick,\n isSupported = true,\n size = 'md',\n className,\n style,\n ariaLabel,\n idleIcon,\n listeningIcon,\n disabled,\n}: DictationButtonProps): React.ReactElement {\n const listening = status === 'listening' || status === 'starting';\n const stopping = status === 'stopping';\n const off = !isSupported;\n\n return (\n <button\n type=\"button\"\n onClick={onClick}\n disabled={disabled || off}\n aria-pressed={listening}\n aria-label={\n ariaLabel ?? (listening ? 'Stop dictation' : off ? 'Dictation not supported' : 'Start dictation')\n }\n className={cn(\n 'relative inline-flex items-center justify-center rounded-full transition-colors',\n 'focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2',\n 'disabled:cursor-not-allowed disabled:opacity-50',\n SIZE_CLS[size],\n listening\n ? 'bg-destructive text-destructive-foreground hover:bg-destructive/90'\n : 'bg-primary text-primary-foreground hover:bg-primary/90',\n className,\n )}\n style={style}\n >\n {listening && (\n <span\n aria-hidden\n className=\"absolute inset-0 rounded-full bg-destructive/40 animate-ping\"\n />\n )}\n {stopping ? (\n <Loader2 className=\"animate-spin\" />\n ) : off ? (\n listeningIcon ?? <MicOff />\n ) : listening ? (\n listeningIcon ?? <Mic />\n ) : (\n idleIcon ?? <Mic />\n )}\n </button>\n );\n}\n","'use client';\n\nimport type * as React from 'react';\n\nimport { AlertTriangle } from 'lucide-react';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nimport type { RecognitionError } from '../types';\n\nconst FRIENDLY: Record<RecognitionError['code'], string> = {\n unsupported: 'Speech recognition isn\\'t available in this browser.',\n 'permission-denied': 'Microphone access was denied. Allow it in your browser settings to dictate.',\n 'no-microphone': 'No microphone found.',\n network: 'Network error talking to the speech service.',\n aborted: 'Recognition was interrupted.',\n 'no-speech': 'No speech was detected. Try again.',\n language: 'The requested language isn\\'t supported by the engine.',\n engine: 'The speech engine reported an error.',\n unknown: 'Something went wrong with the microphone.',\n};\n\nexport interface ErrorBannerProps {\n error: RecognitionError | null;\n className?: string;\n onDismiss?: () => void;\n}\n\nexport function ErrorBanner({ error, className, onDismiss }: ErrorBannerProps): React.ReactElement | null {\n if (!error) return null;\n return (\n <div\n role=\"alert\"\n className={cn(\n 'flex items-start gap-2 rounded-md border border-destructive/40 bg-destructive/10 px-3 py-2 text-xs text-destructive',\n className,\n )}\n >\n <AlertTriangle className=\"mt-0.5 h-3.5 w-3.5 shrink-0\" />\n <div className=\"flex-1\">{FRIENDLY[error.code] ?? error.message}</div>\n {onDismiss && (\n <button\n type=\"button\"\n onClick={onDismiss}\n className=\"text-destructive hover:underline\"\n >\n Dismiss\n </button>\n )}\n </div>\n );\n}\n","'use client';\n\nimport type * as React from 'react';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nexport interface MicMeterProps {\n /** RMS level 0..1 (use the value returned by `useMicLevel`). */\n level: number;\n /** Number of bars rendered. @default 12 */\n bars?: number;\n /** Bar gap in px. @default 2 */\n gap?: number;\n /** Bar width in px. @default 3 */\n barWidth?: number;\n /** Container height in px. @default 24 */\n height?: number;\n className?: string;\n}\n\n/**\n * Discrete-bar VU meter — small, dependency-free, no canvas. Each bar\n * lights up when the current `level` exceeds its threshold. Centre bars\n * are tallest so the meter reads like a classic mic level indicator.\n */\nexport function MicMeter({\n level,\n bars = 12,\n gap = 2,\n barWidth = 3,\n height = 24,\n className,\n}: MicMeterProps): React.ReactElement {\n const clamped = Math.min(1, Math.max(0, level));\n return (\n <div\n role=\"meter\"\n aria-valuemin={0}\n aria-valuemax={1}\n aria-valuenow={clamped}\n className={cn('inline-flex items-center', className)}\n style={{ gap, height }}\n >\n {Array.from({ length: bars }).map((_, i) => {\n const ratio = (i + 1) / bars;\n const active = clamped >= ratio - 0.5 / bars;\n // taller in the middle, shorter at edges\n const heightRatio = 1 - Math.abs(i - (bars - 1) / 2) / ((bars - 1) / 2 || 1);\n const barH = Math.max(2, height * (0.35 + heightRatio * 0.65));\n return (\n <span\n key={i}\n className={cn(\n 'rounded-sm transition-colors',\n active ? 'bg-primary' : 'bg-border',\n )}\n style={{ width: barWidth, height: barH }}\n />\n );\n })}\n </div>\n );\n}\n","'use client';\n\nimport type * as React from 'react';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nexport interface PushToTalkHintProps {\n /** Same chord string passed to `usePushToTalk`. */\n chord: string;\n className?: string;\n}\n\n/**\n * Renders \"Hold ⌥ to talk\" with the chord pretty-printed. Tiny helper\n * meant to live next to a `DictationButton`.\n */\nexport function PushToTalkHint({ chord, className }: PushToTalkHintProps): React.ReactElement {\n const formatted = chord\n .split('+')\n .map((p) => p.trim().toLowerCase())\n .map((p) => {\n switch (p) {\n case 'mod':\n case 'meta':\n return '⌘';\n case 'alt':\n return '⌥';\n case 'shift':\n return '⇧';\n case 'ctrl':\n return '⌃';\n default:\n return p.length === 1 ? p.toUpperCase() : p;\n }\n })\n .join('');\n return (\n <span\n className={cn(\n 'inline-flex items-center gap-1 text-[11px] text-muted-foreground',\n className,\n )}\n >\n Hold\n <kbd className=\"rounded border border-border bg-muted px-1 py-0.5 font-mono text-[10px] text-foreground\">\n {formatted}\n </kbd>\n to talk\n </span>\n );\n}\n","import type { Segment, Transcript } from '../types';\n\nexport const EMPTY_TRANSCRIPT: Transcript = {\n interim: '',\n final: '',\n segments: [],\n};\n\nexport function joinFinal(segments: Segment[]): string {\n let out = '';\n for (const seg of segments) {\n if (!seg.isFinal) continue;\n const text = seg.text.trim();\n if (!text) continue;\n out = out ? `${out} ${text}` : text;\n }\n return out;\n}\n\nexport function buildTranscript(segments: Segment[]): Transcript {\n const last = segments[segments.length - 1];\n const interim = last && !last.isFinal ? last.text : '';\n return {\n interim,\n final: joinFinal(segments),\n segments,\n };\n}\n\n/**\n * Polite text normalisation between concatenated finals — strips double\n * spaces / leading punctuation that some engines emit when the user pauses.\n */\nexport function normaliseFinal(text: string): string {\n return text.replace(/\\s+/g, ' ').replace(/\\s+([,.!?])/g, '$1').trim();\n}\n","let counter = 0;\n\n/**\n * Cheap monotonic id — collisions are fine across sessions, we just need\n * uniqueness within one component lifecycle. Avoids pulling in nanoid for\n * a tool that already keeps the lazy chunk small.\n */\nexport function newSegmentId(): string {\n counter = (counter + 1) % Number.MAX_SAFE_INTEGER;\n return `seg_${Date.now().toString(36)}_${counter.toString(36)}`;\n}\n","/**\n * Speech recognition dev logger.\n *\n * Mirrors `getChatLogger()` in the Chat tool so consumers reason about both\n * surfaces the same way. Silent in production by default; emits `error`\n * level always.\n *\n * Opt-in mechanisms (any one is enough):\n * 1. `NODE_ENV === 'development'` (auto, via `isDev` from `@djangocfg/ui-core`).\n * 2. `localStorage.setItem('djangocfg:speech-debug', '1')` — runtime\n * toggle without a rebuild. Useful for diagnosing a prod bug.\n * 3. Explicit `getSpeechLogger(true)` from a host component.\n *\n * Sub-loggers:\n * engine — engine lifecycle (start/stop/abort/result)\n * dictation — `useDictation` state (anchor, partial → final merge)\n * slot — `VoiceComposerSlot` UI (mount, support gating, button)\n * composer — composer handle registry (focus / setValue / pin caret)\n * mic — `useMicLevel` / `useMicDevices` / device picker\n * push — push-to-talk hotkey events\n * error — caught errors (always emitted, even when disabled)\n */\nimport { consola, type ConsolaInstance } from 'consola';\n\nimport { isBrowser, isDev } from '@djangocfg/ui-core/lib';\n\nexport type SpeechLogScope =\n | 'engine'\n | 'dictation'\n | 'slot'\n | 'composer'\n | 'mic'\n | 'push'\n | 'error';\n\nexport interface SpeechLogger {\n engine: ConsolaInstance;\n dictation: ConsolaInstance;\n slot: ConsolaInstance;\n composer: ConsolaInstance;\n mic: ConsolaInstance;\n push: ConsolaInstance;\n error: ConsolaInstance;\n /** True when this logger is actually emitting. */\n enabled: boolean;\n}\n\nconst SCOPES: SpeechLogScope[] = [\n 'engine',\n 'dictation',\n 'slot',\n 'composer',\n 'mic',\n 'push',\n 'error',\n];\n\nconst STORAGE_KEY = 'djangocfg:speech-debug';\n\nfunction readStorageFlag(): boolean {\n if (!isBrowser) return false;\n try {\n return window.localStorage.getItem(STORAGE_KEY) === '1';\n } catch {\n return false;\n }\n}\n\nconst cache = new Map<boolean, SpeechLogger>();\n\nfunction buildLogger(enabled: boolean): SpeechLogger {\n const root = consola.withTag('speech');\n const subs = Object.fromEntries(\n SCOPES.map((scope) => [scope, root.withTag(scope)]),\n ) as Record<SpeechLogScope, ConsolaInstance>;\n\n if (!enabled) {\n for (const scope of SCOPES) {\n if (scope === 'error') continue;\n subs[scope].level = -999;\n }\n }\n\n return { ...subs, enabled };\n}\n\n/**\n * Get the speech logger.\n *\n * @param debug Explicit override. `undefined` falls back to `isDev` OR\n * `localStorage['djangocfg:speech-debug'] === '1'`.\n */\nexport function getSpeechLogger(debug?: boolean): SpeechLogger {\n const enabled = debug ?? (isDev || readStorageFlag());\n let logger = cache.get(enabled);\n if (!logger) {\n logger = buildLogger(enabled);\n cache.set(enabled, logger);\n }\n return logger;\n}\n\n/**\n * Legacy flat-tag logger. New code should prefer `getSpeechLogger()` for\n * scoped output and the runtime opt-in switch.\n *\n * @deprecated Use `getSpeechLogger()`.\n */\nexport const sttLogger = consola.withTag('ui-tools:speech');\n","import { newSegmentId } from './ids';\nimport type {\n RecognitionError,\n RecognitionStatus,\n Segment,\n} from '../types';\n\nexport interface RecognitionState {\n status: RecognitionStatus;\n segments: Segment[];\n error: RecognitionError | null;\n startedAt: number | null;\n}\n\nexport const INITIAL_STATE: RecognitionState = {\n status: 'idle',\n segments: [],\n error: null,\n startedAt: null,\n};\n\nexport type RecognitionAction =\n | { type: 'START' }\n | { type: 'STARTED' }\n | { type: 'STOP' }\n | { type: 'STOPPED' }\n | { type: 'ABORT' }\n | {\n type: 'PARTIAL';\n text: string;\n segmentId: string;\n confidence?: number;\n }\n | {\n type: 'FINAL';\n text: string;\n segmentId: string;\n confidence?: number;\n }\n | { type: 'ERROR'; error: RecognitionError }\n | { type: 'RESET' };\n\nfunction nowSinceStart(state: RecognitionState): number {\n return state.startedAt ? Date.now() - state.startedAt : 0;\n}\n\nfunction upsertSegment(\n segments: Segment[],\n patch: Segment,\n): Segment[] {\n const idx = segments.findIndex((s) => s.id === patch.id);\n if (idx === -1) return [...segments, patch];\n const next = segments.slice();\n next[idx] = { ...next[idx], ...patch };\n return next;\n}\n\nexport function reducer(\n state: RecognitionState,\n action: RecognitionAction,\n): RecognitionState {\n switch (action.type) {\n case 'START':\n return {\n ...state,\n status: 'starting',\n error: null,\n startedAt: Date.now(),\n };\n case 'STARTED':\n return { ...state, status: 'listening' };\n case 'STOP':\n return { ...state, status: 'stopping' };\n case 'STOPPED':\n case 'ABORT':\n return { ...state, status: 'idle' };\n case 'PARTIAL': {\n const seg: Segment = {\n id: action.segmentId,\n text: action.text,\n isFinal: false,\n confidence: action.confidence,\n startedAt: nowSinceStart(state),\n };\n return { ...state, segments: upsertSegment(state.segments, seg) };\n }\n case 'FINAL': {\n const seg: Segment = {\n id: action.segmentId || newSegmentId(),\n text: action.text,\n isFinal: true,\n confidence: action.confidence,\n startedAt: nowSinceStart(state),\n endedAt: nowSinceStart(state),\n };\n return { ...state, segments: upsertSegment(state.segments, seg) };\n }\n case 'ERROR':\n return { ...state, status: 'error', error: action.error };\n case 'RESET':\n return { ...INITIAL_STATE };\n default:\n return state;\n }\n}\n","/**\n * Tiny event-bus helper shared by every engine. Lets engine authors avoid\n * re-implementing add/remove listener bookkeeping while keeping the\n * public `RecognitionEngine.on(...)` contract identical across engines.\n */\n\nimport type { EngineEventMap, Unsub } from '../../types';\n\ntype Listeners = {\n [K in keyof EngineEventMap]: Set<EngineEventMap[K]>;\n};\n\nexport function createEngineBus(): {\n on: <K extends keyof EngineEventMap>(event: K, cb: EngineEventMap[K]) => Unsub;\n emit: <K extends keyof EngineEventMap>(\n event: K,\n ...args: Parameters<EngineEventMap[K]>\n ) => void;\n clear: () => void;\n} {\n const listeners: Listeners = {\n partial: new Set(),\n final: new Set(),\n error: new Set(),\n state: new Set(),\n };\n\n return {\n on(event, cb) {\n const set = listeners[event] as Set<typeof cb>;\n set.add(cb);\n return () => {\n set.delete(cb);\n };\n },\n emit(event, ...args) {\n const set = listeners[event];\n for (const cb of set) {\n try {\n (cb as (...a: unknown[]) => void)(...(args as unknown[]));\n } catch {\n // listener errors are isolated — never break the engine loop\n }\n }\n },\n clear() {\n for (const key of Object.keys(listeners) as Array<keyof Listeners>) {\n listeners[key].clear();\n }\n },\n };\n}\n","/**\n * Default engine — wraps the browser's `SpeechRecognition` API.\n *\n * Lives behind the same `RecognitionEngine` contract every other engine\n * implements. When the browser doesn't expose `SpeechRecognition`\n * (Firefox, some mobile WebViews) `isSupported` is `false` and `start()`\n * throws an `unsupported` error.\n */\n\nimport { newSegmentId } from '../ids';\nimport { sttLogger } from '../logger';\nimport { createEngineBus } from './index';\nimport type {\n EngineStartOptions,\n RecognitionEngine,\n RecognitionError,\n RecognitionErrorCode,\n Unsub,\n} from '../../types';\n\n// Minimal subset of the Web Speech API we actually rely on. Browsers\n// expose either `SpeechRecognition` (Edge / Safari new) or the older\n// `webkitSpeechRecognition` (Chrome). Both share the same shape.\ninterface BrowserSpeechRecognition extends EventTarget {\n lang: string;\n interimResults: boolean;\n continuous: boolean;\n maxAlternatives: number;\n start(): void;\n stop(): void;\n abort(): void;\n onresult: ((e: BrowserSpeechRecognitionEvent) => void) | null;\n onerror: ((e: BrowserSpeechRecognitionError) => void) | null;\n onstart: (() => void) | null;\n onend: (() => void) | null;\n}\n\ninterface BrowserSpeechRecognitionResult {\n isFinal: boolean;\n 0: { transcript: string; confidence: number };\n}\n\ninterface BrowserSpeechRecognitionEvent extends Event {\n resultIndex: number;\n results: ArrayLike<BrowserSpeechRecognitionResult>;\n}\n\ninterface BrowserSpeechRecognitionError extends Event {\n error: string;\n message?: string;\n}\n\ntype Ctor = new () => BrowserSpeechRecognition;\n\nfunction resolveCtor(): Ctor | null {\n if (typeof window === 'undefined') return null;\n const w = window as unknown as {\n SpeechRecognition?: Ctor;\n webkitSpeechRecognition?: Ctor;\n };\n return w.SpeechRecognition ?? w.webkitSpeechRecognition ?? null;\n}\n\nconst ERROR_MAP: Record<string, RecognitionErrorCode> = {\n 'no-speech': 'no-speech',\n aborted: 'aborted',\n 'audio-capture': 'no-microphone',\n network: 'network',\n 'not-allowed': 'permission-denied',\n 'service-not-allowed': 'permission-denied',\n 'bad-grammar': 'engine',\n 'language-not-supported': 'language',\n};\n\nexport interface WebSpeechEngineOptions {\n /** Whether the underlying recognition should be continuous. Default true. */\n continuous?: boolean;\n /** Max alternatives the engine should request. Default 1. */\n maxAlternatives?: number;\n}\n\nexport function createWebSpeechEngine(\n opts: WebSpeechEngineOptions = {},\n): RecognitionEngine {\n const Ctor = resolveCtor();\n const bus = createEngineBus();\n let instance: BrowserSpeechRecognition | null = null;\n let currentSegmentId: string | null = null;\n\n function teardown(): void {\n if (!instance) return;\n instance.onresult = null;\n instance.onerror = null;\n instance.onstart = null;\n instance.onend = null;\n instance = null;\n currentSegmentId = null;\n }\n\n return {\n id: 'webspeech',\n isSupported: Ctor !== null,\n on(event, cb): Unsub {\n return bus.on(event, cb);\n },\n async start(start: EngineStartOptions): Promise<void> {\n if (!Ctor) {\n const err: RecognitionError = {\n code: 'unsupported',\n message: 'Web Speech API is not available in this browser.',\n };\n bus.emit('error', err);\n throw err;\n }\n if (instance) {\n sttLogger.debug('[webspeech] start() called while running — ignoring');\n return;\n }\n\n bus.emit('state', 'connecting');\n\n const rec = new Ctor();\n rec.lang = start.language;\n rec.interimResults = start.interim;\n rec.continuous = opts.continuous ?? true;\n rec.maxAlternatives = opts.maxAlternatives ?? 1;\n\n rec.onstart = () => {\n bus.emit('state', 'listening');\n };\n rec.onend = () => {\n bus.emit('state', 'closed');\n teardown();\n };\n rec.onerror = (e) => {\n const code = ERROR_MAP[e.error] ?? 'engine';\n const err: RecognitionError = {\n code,\n message: e.message || `Web Speech error: ${e.error}`,\n };\n bus.emit('error', err);\n };\n rec.onresult = (e) => {\n for (let i = e.resultIndex; i < e.results.length; i += 1) {\n const res = e.results[i];\n const alt = res[0];\n const text = alt.transcript;\n if (!currentSegmentId) currentSegmentId = newSegmentId();\n if (res.isFinal) {\n bus.emit('final', text, currentSegmentId, alt.confidence);\n currentSegmentId = null;\n } else {\n bus.emit('partial', text, currentSegmentId);\n }\n }\n };\n\n if (start.signal) {\n start.signal.addEventListener('abort', () => {\n rec.abort();\n });\n }\n\n instance = rec;\n try {\n rec.start();\n } catch (cause) {\n const err: RecognitionError = {\n code: 'engine',\n message: 'Failed to start Web Speech recognition.',\n cause,\n };\n bus.emit('error', err);\n teardown();\n throw err;\n }\n },\n async stop(): Promise<void> {\n if (!instance) return;\n bus.emit('state', 'closing');\n instance.stop();\n },\n abort(): void {\n if (!instance) return;\n instance.abort();\n },\n };\n}\n","'use client';\n\nimport { create } from 'zustand';\nimport { persist, createJSONStorage } from 'zustand/middleware';\n\nexport interface SpeechPrefs {\n /**\n * BCP-47 tag the user explicitly picked (via `<LanguagePicker>` or\n * programmatically). `null` means \"no override\" — `useResolvedLanguage`\n * then falls through to the app i18n locale / `navigator.language`.\n * Storing the picker default as `null` is what lets a host's i18n\n * locale take effect when the user never touched the picker.\n */\n language: string | null;\n deviceId: string | null;\n engineId: string | null;\n earcons: boolean;\n}\n\nconst DEFAULTS: SpeechPrefs = {\n language: null,\n deviceId: null,\n engineId: null,\n earcons: false,\n};\n\ninterface PrefsStore extends SpeechPrefs {\n setLanguage: (v: string | null) => void;\n setDeviceId: (v: string | null) => void;\n setEngineId: (v: string | null) => void;\n setEarcons: (v: boolean) => void;\n reset: () => void;\n}\n\nexport const useSpeechPrefs = create<PrefsStore>()(\n persist(\n (set) => ({\n ...DEFAULTS,\n setLanguage: (language) => set({ language }),\n setDeviceId: (deviceId) => set({ deviceId }),\n setEngineId: (engineId) => set({ engineId }),\n setEarcons: (earcons) => set({ earcons }),\n reset: () => set({ ...DEFAULTS }),\n }),\n {\n name: 'djangocfg-stt:prefs',\n storage: createJSONStorage(() =>\n typeof window === 'undefined'\n ? (undefined as unknown as Storage)\n : window.localStorage,\n ),\n },\n ),\n);\n","'use client';\n\nimport { useEffect, useRef, useState } from 'react';\n\n/**\n * RMS level meter driven by an `AnalyserNode`. Attach a `MediaStream`\n * (the one returned from `startMicCapture`) and read `level` (0..1) for\n * VU meters / mic-pulse animations. Returns 0 when no stream is bound.\n */\nexport function useMicLevel(stream: MediaStream | null): number {\n const [level, setLevel] = useState(0);\n const raf = useRef<number | null>(null);\n\n useEffect(() => {\n if (!stream) {\n setLevel(0);\n return undefined;\n }\n const AC =\n (window as unknown as { AudioContext?: typeof AudioContext }).AudioContext ??\n (window as unknown as { webkitAudioContext?: typeof AudioContext }).webkitAudioContext;\n if (!AC) return undefined;\n const ctx = new AC();\n const source = ctx.createMediaStreamSource(stream);\n const analyser = ctx.createAnalyser();\n analyser.fftSize = 1024;\n analyser.smoothingTimeConstant = 0.7;\n source.connect(analyser);\n const buf = new Float32Array(analyser.fftSize);\n\n const tick = (): void => {\n analyser.getFloatTimeDomainData(buf);\n let sum = 0;\n for (let i = 0; i < buf.length; i += 1) sum += buf[i] * buf[i];\n const rms = Math.sqrt(sum / buf.length);\n // soft compression so loud peaks don't dominate the meter\n setLevel(Math.min(1, rms * 2.5));\n raf.current = requestAnimationFrame(tick);\n };\n tick();\n\n return () => {\n if (raf.current != null) cancelAnimationFrame(raf.current);\n raf.current = null;\n source.disconnect();\n analyser.disconnect();\n void ctx.close();\n };\n }, [stream]);\n\n return level;\n}\n","/**\n * Maps 2-letter ISO 639-1 codes (`en`, `ru`, `ko` — what\n * `@djangocfg/i18n` exposes via `useLocale()`) to BCP-47 tags\n * (`en-US`, `ru-RU`, `ko-KR`) that the Web Speech API and most cloud\n * STT services expect.\n *\n * We keep a small built-in table for the locales we ship translations\n * for; everything else falls through to `<code>-<UPPER(code)>`, which\n * works for the majority of regions. The mapping is also re-exported\n * so consumers can extend it.\n */\n\nconst ISO_TO_BCP47: Record<string, string> = {\n en: 'en-US',\n ru: 'ru-RU',\n ko: 'ko-KR',\n ja: 'ja-JP',\n zh: 'zh-CN',\n de: 'de-DE',\n fr: 'fr-FR',\n it: 'it-IT',\n es: 'es-ES',\n nl: 'nl-NL',\n ar: 'ar-SA',\n tr: 'tr-TR',\n pl: 'pl-PL',\n sv: 'sv-SE',\n no: 'nb-NO',\n da: 'da-DK',\n pt: 'pt-BR',\n};\n\nexport const DEFAULT_ISO_TO_BCP47 = ISO_TO_BCP47;\n\n/**\n * Normalise any of:\n * - BCP-47 (\"en-US\", \"ru-RU\") — passed through.\n * - ISO 639-1 (\"en\", \"ru\") — mapped via the table above, or\n * falls back to `<code>-<UPPER(code)>`.\n * - `null`/`undefined`/empty — returns `undefined`.\n */\nexport function toBCP47(\n code: string | null | undefined,\n table: Record<string, string> = ISO_TO_BCP47,\n): string | undefined {\n if (!code) return undefined;\n const trimmed = code.trim();\n if (!trimmed) return undefined;\n if (trimmed.includes('-')) return trimmed; // already BCP-47\n const lower = trimmed.toLowerCase();\n return table[lower] ?? `${lower}-${lower.toUpperCase()}`;\n}\n\n/**\n * Resolve the language tag for a speech session in priority order:\n * 1. `explicit` prop (always wins) — host-supplied override.\n * 2. `prefs` — value stored in `useSpeechPrefs` (user picked it\n * via `<LanguagePicker>` or programmatically).\n * 3. `i18n` — current i18n locale (2-letter ISO).\n * 4. `navigator.language` — browser default.\n * 5. `'en-US'` — last-resort safety net.\n *\n * All inputs may be ISO-2 or BCP-47; the function normalises before\n * returning.\n */\nexport function resolveSpeechLanguage(opts: {\n explicit?: string;\n prefs?: string | null;\n i18n?: string | null;\n}): string {\n return (\n toBCP47(opts.explicit) ??\n toBCP47(opts.prefs) ??\n toBCP47(opts.i18n) ??\n toBCP47(typeof navigator !== 'undefined' ? navigator.language : null) ??\n 'en-US'\n );\n}\n","'use client';\n\nimport { useLocaleOptional } from '@djangocfg/i18n';\n\nimport { resolveSpeechLanguage } from '../core/language';\nimport { useSpeechPrefs } from '../store/prefsStore';\n\n/**\n * Resolves the BCP-47 language tag a speech session should use.\n *\n * Priority: explicit prop → user-picked `useSpeechPrefs.language` →\n * app i18n locale (when an `<I18nProvider>` is mounted) →\n * `navigator.language` → `en-US`.\n *\n * Uses `useLocaleOptional` (not `useLocale`) so that an unmounted\n * provider doesn't silently inject `'en'`. Without that, the i18n\n * default would shadow the user's real browser language — a Russian\n * speaker would always get `en-US` recognition.\n */\nexport function useResolvedLanguage(explicit?: string): string {\n const prefs = useSpeechPrefs();\n const locale = useLocaleOptional();\n return resolveSpeechLanguage({\n explicit,\n prefs: prefs.language,\n i18n: locale,\n });\n}\n","'use client';\n\nimport { useCallback, useEffect, useMemo, useReducer, useRef, useState } from 'react';\n\nimport {\n EMPTY_TRANSCRIPT,\n INITIAL_STATE,\n buildTranscript,\n getSpeechLogger,\n reducer,\n} from '../core';\n\nconst log = getSpeechLogger();\nimport { createWebSpeechEngine } from '../core/engine/webspeech';\nimport { useSpeechPrefs } from '../store/prefsStore';\nimport type {\n RecognitionEngine,\n Segment,\n UseSpeechRecognitionConfig,\n UseSpeechRecognitionReturn,\n} from '../types';\nimport { useMicLevel } from './useMicLevel';\nimport { useResolvedLanguage } from './useResolvedLanguage';\n\n/**\n * Main entry point. With no config it uses the browser Web Speech API\n * and the persisted language from `useSpeechPrefs`. Pass a custom\n * `engine` to route through Deepgram / Whisper / custom WebSocket.\n */\nexport function useSpeechRecognition(\n config: UseSpeechRecognitionConfig = {},\n): UseSpeechRecognitionReturn {\n const prefs = useSpeechPrefs();\n const language = useResolvedLanguage(config.language);\n const engine = useMemo<RecognitionEngine>(\n () => config.engine ?? createWebSpeechEngine(),\n [config.engine],\n );\n\n const [state, dispatch] = useReducer(reducer, INITIAL_STATE);\n const [stream, setStream] = useState<MediaStream | null>(null);\n const level = useMicLevel(stream);\n\n // Latest-callback refs so engine subscriptions never tear down on\n // every render — same trick the Chat reducer uses.\n const cbRef = useRef(config);\n cbRef.current = config;\n\n // Engine subscription lifecycle.\n useEffect(() => {\n log.engine.debug('subscribe', { engineId: engine.id });\n const offs = [\n engine.on('partial', (text, segmentId) => {\n log.engine.debug('partial', { len: text.length, segmentId });\n dispatch({ type: 'PARTIAL', text, segmentId });\n const seg: Segment = {\n id: segmentId,\n text,\n isFinal: false,\n startedAt: Date.now(),\n };\n cbRef.current.onPartial?.(text, seg);\n }),\n engine.on('final', (text, segmentId, confidence) => {\n log.engine.info('final', {\n len: text.length,\n segmentId,\n confidence,\n });\n dispatch({ type: 'FINAL', text, segmentId, confidence });\n const seg: Segment = {\n id: segmentId,\n text,\n isFinal: true,\n confidence,\n startedAt: Date.now(),\n endedAt: Date.now(),\n };\n cbRef.current.onFinal?.(text, seg);\n }),\n engine.on('error', (err) => {\n log.error.error('engine error', err);\n dispatch({ type: 'ERROR', error: err });\n cbRef.current.onError?.(err);\n }),\n engine.on('state', (s) => {\n log.engine.debug('state', s);\n if (s === 'listening') {\n dispatch({ type: 'STARTED' });\n cbRef.current.onStart?.();\n setStream(engine.getStream?.() ?? null);\n } else if (s === 'closed') {\n dispatch({ type: 'STOPPED' });\n cbRef.current.onStop?.();\n setStream(null);\n }\n }),\n ];\n return () => {\n offs.forEach((off) => off());\n };\n }, [engine]);\n\n // AutoStop driven by silence + maxMs caps.\n const silenceTimer = useRef<number | null>(null);\n const maxTimer = useRef<number | null>(null);\n useEffect(() => {\n if (state.status !== 'listening') return undefined;\n const { silenceMs, maxMs, silenceThreshold = 0.02 } = config.autoStop ?? {};\n if (maxMs) {\n maxTimer.current = window.setTimeout(() => {\n log.engine.debug('autoStop max duration hit');\n void engine.stop();\n }, maxMs);\n }\n if (silenceMs) {\n const checkInterval = window.setInterval(() => {\n if (level < silenceThreshold) {\n if (silenceTimer.current == null) {\n silenceTimer.current = window.setTimeout(() => {\n log.engine.debug('autoStop silence detected');\n void engine.stop();\n }, silenceMs);\n }\n } else if (silenceTimer.current != null) {\n clearTimeout(silenceTimer.current);\n silenceTimer.current = null;\n }\n }, 200);\n return () => {\n clearInterval(checkInterval);\n if (silenceTimer.current != null) clearTimeout(silenceTimer.current);\n silenceTimer.current = null;\n if (maxTimer.current != null) clearTimeout(maxTimer.current);\n maxTimer.current = null;\n };\n }\n return () => {\n if (maxTimer.current != null) clearTimeout(maxTimer.current);\n maxTimer.current = null;\n };\n }, [state.status, config.autoStop, level, engine]);\n\n const start = useCallback(async () => {\n if (state.status === 'listening' || state.status === 'starting') return;\n dispatch({ type: 'START' });\n try {\n await engine.start({\n language,\n interim: config.interim ?? true,\n deviceId: config.deviceId ?? prefs.deviceId ?? undefined,\n });\n } catch (cause) {\n // engine already emitted 'error'; reducer caught it via subscription\n log.engine.warn('start threw', cause);\n }\n }, [engine, language, config.interim, config.deviceId, prefs.deviceId, state.status]);\n\n const stop = useCallback(async () => {\n if (state.status === 'idle' || state.status === 'stopping') return;\n dispatch({ type: 'STOP' });\n await engine.stop();\n }, [engine, state.status]);\n\n const abort = useCallback(() => {\n engine.abort();\n dispatch({ type: 'ABORT' });\n }, [engine]);\n\n const toggle = useCallback(async () => {\n if (state.status === 'listening' || state.status === 'starting') {\n await stop();\n } else {\n await start();\n }\n }, [state.status, start, stop]);\n\n const reset = useCallback(() => {\n dispatch({ type: 'RESET' });\n }, []);\n\n const transcript = useMemo(\n () => (state.segments.length === 0 ? EMPTY_TRANSCRIPT : buildTranscript(state.segments)),\n [state.segments],\n );\n\n return {\n status: state.status,\n isSupported: engine.isSupported,\n transcript,\n error: state.error,\n level,\n start,\n stop,\n abort,\n toggle,\n reset,\n };\n}\n","'use client';\n\nimport { useEffect, useRef } from 'react';\n\nimport { normaliseFinal } from '../core/transcript';\nimport type { UseSpeechRecognitionConfig, UseSpeechRecognitionReturn } from '../types';\nimport { useSpeechRecognition } from './useSpeechRecognition';\n\nexport interface UseDictationConfig\n extends Omit<UseSpeechRecognitionConfig, 'onFinal'> {\n /** Controlled value the dictation is appending to. */\n value: string;\n /** Called with the next value after each final segment lands. */\n onChange: (next: string) => void;\n /** Joiner between the previous value and the new segment. Default ' '. */\n separator?: string;\n}\n\nexport interface UseDictationReturn extends UseSpeechRecognitionReturn {\n /** Convenience — same as `toggle`, named for dictation UIs. */\n toggleDictation: () => Promise<void>;\n}\n\n/**\n * Convenience adapter that pipes final transcript segments straight into\n * a controlled string (`<textarea>` / `<input>` / TipTap). Interim text\n * is left alone — bind `transcript.interim` separately if you want to\n * show a live ghost.\n */\nexport function useDictation(config: UseDictationConfig): UseDictationReturn {\n const { value, onChange, separator = ' ', ...rest } = config;\n\n // Stash latest value in a ref so the onFinal closure always sees the\n // freshest text without forcing the underlying hook to resubscribe.\n const valueRef = useRef(value);\n valueRef.current = value;\n const onChangeRef = useRef(onChange);\n onChangeRef.current = onChange;\n\n const rec = useSpeechRecognition({\n ...rest,\n onFinal: (text) => {\n const clean = normaliseFinal(text);\n if (!clean) return;\n const prev = valueRef.current;\n const next = prev ? `${prev}${separator}${clean}` : clean;\n onChangeRef.current(next);\n },\n });\n\n useEffect(() => {\n valueRef.current = value;\n }, [value]);\n\n return {\n ...rec,\n toggleDictation: rec.toggle,\n };\n}\n","'use client';\n\nimport { useEffect } from 'react';\n\nimport type { UseSpeechRecognitionReturn } from '../types';\n\nexport interface UsePushToTalkOptions {\n /** Key to hold. Combine modifiers with `+`, e.g. `'alt'`, `'mod+alt'`. */\n key: string;\n /** Disable the binding without unmounting. */\n enabled?: boolean;\n}\n\nconst MOD_KEYS = new Set(['shift', 'ctrl', 'alt', 'meta', 'mod']);\n\nfunction parseChord(chord: string): { mods: Set<string>; main: string | null } {\n const parts = chord\n .toLowerCase()\n .split('+')\n .map((s) => s.trim());\n const mods = new Set<string>();\n let main: string | null = null;\n for (const part of parts) {\n if (MOD_KEYS.has(part)) {\n mods.add(part === 'mod' ? 'meta' : part);\n } else {\n main = part;\n }\n }\n return { mods, main };\n}\n\nfunction matches(e: KeyboardEvent, mods: Set<string>, main: string | null): boolean {\n if (mods.has('shift') !== e.shiftKey) return false;\n if (mods.has('ctrl') !== e.ctrlKey) return false;\n if (mods.has('alt') !== e.altKey) return false;\n // 'mod' → meta on mac / ctrl elsewhere; we already normalised to 'meta'.\n if (mods.has('meta') !== (e.metaKey || (!e.metaKey && false))) return false;\n if (main && e.key.toLowerCase() !== main) return false;\n return true;\n}\n\n/**\n * Hold-to-talk wiring. Press → `start()`, release → `stop()`. Ignores\n * repeats and skips keydown inside `<input>` / `<textarea>` unless a\n * modifier is in the chord.\n */\nexport function usePushToTalk(\n recognition: Pick<UseSpeechRecognitionReturn, 'start' | 'stop' | 'status'>,\n opts: UsePushToTalkOptions,\n): void {\n const { key, enabled = true } = opts;\n\n useEffect(() => {\n if (!enabled || typeof window === 'undefined') return undefined;\n const { mods, main } = parseChord(key);\n\n const onDown = (e: KeyboardEvent): void => {\n if (e.repeat) return;\n if (!main && mods.size === 0) return;\n const target = e.target as HTMLElement | null;\n const inField =\n target?.tagName === 'INPUT' ||\n target?.tagName === 'TEXTAREA' ||\n target?.isContentEditable;\n if (inField && mods.size === 0) return;\n if (!matches(e, mods, main)) return;\n if (recognition.status === 'listening' || recognition.status === 'starting') return;\n e.preventDefault();\n void recognition.start();\n };\n const onUp = (e: KeyboardEvent): void => {\n if (!matches(e, mods, main) && main && e.key.toLowerCase() !== main) return;\n if (recognition.status !== 'listening' && recognition.status !== 'starting') return;\n void recognition.stop();\n };\n\n window.addEventListener('keydown', onDown);\n window.addEventListener('keyup', onUp);\n return () => {\n window.removeEventListener('keydown', onDown);\n window.removeEventListener('keyup', onUp);\n };\n }, [enabled, key, recognition]);\n}\n","'use client';\n\nimport type * as React from 'react';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nimport { DictationButton } from '../components/DictationButton';\nimport { ErrorBanner } from '../components/ErrorBanner';\nimport { MicMeter } from '../components/MicMeter';\nimport { PushToTalkHint } from '../components/PushToTalkHint';\nimport { useDictation } from '../hooks/useDictation';\nimport { usePushToTalk } from '../hooks/usePushToTalk';\nimport type { RecognitionEngine } from '../types';\n\nexport interface DictationFieldProps {\n value: string;\n onChange: (next: string) => void;\n /** Custom engine. Defaults to Web Speech via `useSpeechRecognition`. */\n engine?: RecognitionEngine;\n /** Override the language stored in `useSpeechPrefs`. */\n language?: string;\n /** Push-to-talk chord (e.g. `'alt'`, `'mod+alt'`). Disabled when omitted. */\n pushToTalk?: { key: string; enabled?: boolean };\n placeholder?: string;\n rows?: number;\n disabled?: boolean;\n /** Show the interim transcript as a ghost overlay below the textarea. */\n showInterim?: boolean;\n /** Show a small RMS meter inside the toolbar. */\n showMeter?: boolean;\n className?: string;\n textareaClassName?: string;\n}\n\n/**\n * Opinionated textarea + dictation button assembly. Final segments are\n * appended to the controlled `value` automatically. Press-and-hold\n * shortcut is optional; the mic button itself works as a toggle.\n */\nexport function DictationField({\n value,\n onChange,\n engine,\n language,\n pushToTalk,\n placeholder = 'Type or press the mic to dictate…',\n rows = 3,\n disabled,\n showInterim = true,\n showMeter = true,\n className,\n textareaClassName,\n}: DictationFieldProps): React.ReactElement {\n const rec = useDictation({\n value,\n onChange,\n engine,\n language,\n });\n\n usePushToTalk(rec, {\n key: pushToTalk?.key ?? 'alt',\n enabled: !!pushToTalk && pushToTalk.enabled !== false,\n });\n\n return (\n <div className={cn('flex flex-col gap-2', className)}>\n <div className=\"relative\">\n <textarea\n value={value}\n onChange={(e) => onChange(e.target.value)}\n placeholder={placeholder}\n rows={rows}\n disabled={disabled}\n className={cn(\n 'w-full resize-y rounded-md border border-input bg-background px-3 py-2 text-sm',\n 'placeholder:text-muted-foreground',\n 'focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring',\n 'disabled:cursor-not-allowed disabled:opacity-50',\n textareaClassName,\n )}\n />\n {showInterim && rec.transcript.interim && (\n <div className=\"pointer-events-none mt-1 text-xs italic text-muted-foreground\">\n … {rec.transcript.interim}\n </div>\n )}\n </div>\n\n <div className=\"flex items-center gap-2\">\n <DictationButton\n status={rec.status}\n isSupported={rec.isSupported}\n onClick={() => void rec.toggleDictation()}\n size=\"sm\"\n disabled={disabled}\n />\n {showMeter && <MicMeter level={rec.level} bars={10} height={20} />}\n {pushToTalk && <PushToTalkHint chord={pushToTalk.key} className=\"ml-auto\" />}\n </div>\n\n <ErrorBanner error={rec.error} />\n </div>\n );\n}\n"]}
|