@ankorar/nodex 0.0.1
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 +228 -0
- package/package.json +54 -0
- package/src/components/mindMap/Background.tsx +39 -0
- package/src/components/mindMap/Board.tsx +159 -0
- package/src/components/mindMap/CentalNode.tsx +121 -0
- package/src/components/mindMap/DefaultNode.tsx +205 -0
- package/src/components/mindMap/Header.tsx +247 -0
- package/src/components/mindMap/ImageNode.tsx +345 -0
- package/src/components/mindMap/KeyboardHelpDialog.tsx +108 -0
- package/src/components/mindMap/MineMap.tsx +237 -0
- package/src/components/mindMap/NodeStylePopover.tsx +486 -0
- package/src/components/mindMap/Nodes.tsx +113 -0
- package/src/components/mindMap/Nodex.tsx +65 -0
- package/src/components/mindMap/SaveStatusIndicator.tsx +61 -0
- package/src/components/mindMap/Segments.tsx +270 -0
- package/src/components/mindMap/ZenCard.tsx +41 -0
- package/src/components/ui/dialog.tsx +141 -0
- package/src/components/ui/popover.tsx +46 -0
- package/src/components/ui/select.tsx +192 -0
- package/src/components/ui/toggle-group.tsx +83 -0
- package/src/components/ui/toggle.tsx +45 -0
- package/src/config/rootKeyBinds.ts +191 -0
- package/src/config/shortCuts.ts +28 -0
- package/src/contexts/MindMapNodeEditorContext.tsx +47 -0
- package/src/handlers/rootKeyBinds/handleAltEKeyBind.ts +6 -0
- package/src/handlers/rootKeyBinds/handleAltHKeyBind.ts +6 -0
- package/src/handlers/rootKeyBinds/handleAltWKeyBind.ts +6 -0
- package/src/handlers/rootKeyBinds/handleAltZKeyBind.ts +6 -0
- package/src/handlers/rootKeyBinds/handleArrowHorizontalRootKeyBind.ts +46 -0
- package/src/handlers/rootKeyBinds/handleArrowVerticalRootKeyBind.ts +44 -0
- package/src/handlers/rootKeyBinds/handleBackEspaceKeyBind.ts +12 -0
- package/src/handlers/rootKeyBinds/handleERootKeyBind.ts +16 -0
- package/src/handlers/rootKeyBinds/handleEnterRootKeyBind.ts +35 -0
- package/src/handlers/rootKeyBinds/handleEscapeKeyBind.ts +24 -0
- package/src/handlers/rootKeyBinds/handleEspaceKeyBind.ts +11 -0
- package/src/handlers/rootKeyBinds/handleMoveByWorldKeyBind.ts +6 -0
- package/src/handlers/rootKeyBinds/handleRedoRootKeyBind.ts +23 -0
- package/src/handlers/rootKeyBinds/handleTabRootKeyBind.ts +49 -0
- package/src/handlers/rootKeyBinds/handleTransformNodeKeyBind.ts +39 -0
- package/src/handlers/rootKeyBinds/handleUndoRootKeyBind.ts +23 -0
- package/src/handlers/rootKeyBinds/handleZoonByKeyBind.ts +31 -0
- package/src/helpers/centerNode.ts +19 -0
- package/src/helpers/getNodeSide.ts +16 -0
- package/src/hooks/mindMap/useHelpers.tsx +9 -0
- package/src/hooks/mindMap/useMindMapDebounce.ts +47 -0
- package/src/hooks/mindMap/useMindMapHistoryDebounce.ts +69 -0
- package/src/hooks/mindMap/useMindMapNode.tsx +203 -0
- package/src/hooks/mindMap/useMindMapNodeEditor.ts +91 -0
- package/src/hooks/mindMap/useMindMapNodeMouseHandlers.ts +24 -0
- package/src/hooks/mindMap/useRootKeyBindHandlers.ts +49 -0
- package/src/hooks/mindMap/useRootMouseHandlers.ts +124 -0
- package/src/hooks/mindMap/useUpdateCenter.ts +54 -0
- package/src/index.ts +76 -0
- package/src/lib/utils.ts +6 -0
- package/src/state/mindMap.ts +793 -0
- package/src/state/mindMapHistory.ts +96 -0
- package/src/styles.input.css +95 -0
- package/src/utils/exportMindMapAsHighQualityImage.ts +327 -0
- package/src/utils/exportMindMapAsMarkdown.ts +102 -0
- package/src/utils/exportMindMapAsPdf.ts +241 -0
- package/src/utils/getMindMapPreviewDataUrl.ts +60 -0
- package/styles.css +2 -0
|
@@ -0,0 +1,241 @@
|
|
|
1
|
+
import { jsPDF } from "jspdf";
|
|
2
|
+
import type { MindMapNode } from "../state/mindMap";
|
|
3
|
+
|
|
4
|
+
export type ExportPdfOptions = {
|
|
5
|
+
/** Filename for download (without extension). */
|
|
6
|
+
filename?: string;
|
|
7
|
+
/** Page orientation. */
|
|
8
|
+
orientation?: "portrait" | "landscape";
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
function nodeToPlainText(node: MindMapNode): string {
|
|
12
|
+
const trimmed = node.text.trim();
|
|
13
|
+
if (node.type === "image") {
|
|
14
|
+
if (!trimmed) return "[imagem]";
|
|
15
|
+
try {
|
|
16
|
+
new URL(trimmed);
|
|
17
|
+
return `[imagem] ${trimmed}`;
|
|
18
|
+
} catch {
|
|
19
|
+
return trimmed || "[imagem]";
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
return trimmed || "(vazio)";
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/** One content block = one node's text (can have multiple paragraphs). */
|
|
26
|
+
type ContentBlock = { level: number; paragraphs: string[] };
|
|
27
|
+
|
|
28
|
+
function walkToBlocks(nodes: MindMapNode[], level: number): ContentBlock[] {
|
|
29
|
+
const blocks: ContentBlock[] = [];
|
|
30
|
+
const visible = nodes.filter((n) => n.isVisible);
|
|
31
|
+
const sorted = [...visible].sort((a, b) => a.sequence - b.sequence);
|
|
32
|
+
|
|
33
|
+
for (const node of sorted) {
|
|
34
|
+
const content = nodeToPlainText(node);
|
|
35
|
+
const paragraphs = content
|
|
36
|
+
.split(/\n+/)
|
|
37
|
+
.map((p) => p.trim())
|
|
38
|
+
.filter(Boolean);
|
|
39
|
+
if (paragraphs.length === 0) paragraphs.push(" ");
|
|
40
|
+
blocks.push({ level, paragraphs });
|
|
41
|
+
if (node.childrens.length > 0) {
|
|
42
|
+
blocks.push(...walkToBlocks(node.childrens, level + 1));
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return blocks;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function getDocumentStructure(nodes: MindMapNode[]): { title: string; blocks: ContentBlock[] } {
|
|
49
|
+
const roots = nodes.filter((n) => n.isVisible);
|
|
50
|
+
if (roots.length === 0) return { title: "", blocks: [] };
|
|
51
|
+
|
|
52
|
+
const sortedRoots = [...roots].sort((a, b) => a.sequence - b.sequence);
|
|
53
|
+
const root = sortedRoots[0];
|
|
54
|
+
const title = root.text.trim() || "Mapa mental";
|
|
55
|
+
const blocks = root.childrens.length > 0 ? walkToBlocks(root.childrens, 1) : [];
|
|
56
|
+
return { title, blocks };
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
// —— Diagramação: grelha e ritmo tipográfico ——
|
|
60
|
+
const MARGIN_X = 56;
|
|
61
|
+
const MARGIN_TOP = 56;
|
|
62
|
+
const MARGIN_BOTTOM = 64;
|
|
63
|
+
const FOOTER_Y_OFFSET = 40;
|
|
64
|
+
const CONTENT_START = 80; // após área do título
|
|
65
|
+
|
|
66
|
+
// Título (nó central)
|
|
67
|
+
const TITLE_FONT_SIZE = 24;
|
|
68
|
+
const TITLE_LINE_HEIGHT = 1.08;
|
|
69
|
+
const TITLE_BOTTOM_RULE_WIDTH = 120;
|
|
70
|
+
const TITLE_RULE_THICKNESS = 2;
|
|
71
|
+
const SPACE_AFTER_TITLE = 48;
|
|
72
|
+
|
|
73
|
+
// Blocos de conteúdo
|
|
74
|
+
const BLOCK_VERTICAL_RHYTHM = 28; // espaço entre blocos de mesmo nível
|
|
75
|
+
const BLOCK_VERTICAL_RHYTHM_MAJOR = 36; // antes de bloco nível 1 (separação de “capítulo”)
|
|
76
|
+
const PARAGRAPH_GAP = 10; // entre parágrafos dentro do bloco
|
|
77
|
+
const ACCENT_BAR_WIDTH = 5;
|
|
78
|
+
const BAR_TO_TEXT = 14;
|
|
79
|
+
const INDENT_PER_LEVEL = 28;
|
|
80
|
+
|
|
81
|
+
const BLOCK_HEADING_SIZE = 12; // primeiro parágrafo do bloco
|
|
82
|
+
const BLOCK_BODY_SIZE = 10;
|
|
83
|
+
const BLOCK_HEADING_LEADING = 1.15;
|
|
84
|
+
const BLOCK_BODY_LEADING = 1.2;
|
|
85
|
+
const FONT_SIZES = [BLOCK_HEADING_SIZE, 11, 10, 9, 9] as const;
|
|
86
|
+
|
|
87
|
+
// Paleta
|
|
88
|
+
const COLOR_TITLE = { r: 15, g: 23, b: 42 };
|
|
89
|
+
const COLOR_LEVEL_1_HEAD = { r: 30, g: 41, b: 59 };
|
|
90
|
+
const COLOR_LEVEL_1_BODY = { r: 51, g: 65, b: 85 };
|
|
91
|
+
const COLOR_LEVEL_2_PLUS = { r: 71, g: 85, b: 105 };
|
|
92
|
+
const COLOR_ACCENT = { r: 59, g: 130, b: 246 };
|
|
93
|
+
const COLOR_RULE = { r: 59, g: 130, b: 246 };
|
|
94
|
+
const COLOR_FOOTER = { r: 148, g: 163, b: 184 };
|
|
95
|
+
|
|
96
|
+
function setTextColor(doc: jsPDF, c: { r: number; g: number; b: number }): void {
|
|
97
|
+
doc.setTextColor(c.r, c.g, c.b);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
function setDrawColor(doc: jsPDF, c: { r: number; g: number; b: number }): void {
|
|
101
|
+
doc.setDrawColor(c.r, c.g, c.b);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
function setFillColor(doc: jsPDF, c: { r: number; g: number; b: number }): void {
|
|
105
|
+
doc.setFillColor(c.r, c.g, c.b);
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
function drawFooter(doc: jsPDF, pageNum: number, pageW: number, pageH: number): void {
|
|
109
|
+
setTextColor(doc, COLOR_FOOTER);
|
|
110
|
+
doc.setFontSize(9);
|
|
111
|
+
doc.setFont("helvetica", "normal");
|
|
112
|
+
const str = String(pageNum);
|
|
113
|
+
doc.text(str, pageW - MARGIN_X - doc.getTextWidth(str), pageH - FOOTER_Y_OFFSET);
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
function ensurePage(
|
|
117
|
+
doc: jsPDF,
|
|
118
|
+
y: number,
|
|
119
|
+
maxY: number,
|
|
120
|
+
pageNum: { current: number },
|
|
121
|
+
pageW: number,
|
|
122
|
+
pageH: number,
|
|
123
|
+
): number {
|
|
124
|
+
if (y <= maxY) return y;
|
|
125
|
+
drawFooter(doc, pageNum.current, pageW, pageH);
|
|
126
|
+
doc.addPage();
|
|
127
|
+
pageNum.current += 1;
|
|
128
|
+
return MARGIN_TOP;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Exports the mind map as a PDF. Central node = document title; every other node = a text block with heading + body. Expert typesetting: rhythm, measure, hierarchy.
|
|
133
|
+
*/
|
|
134
|
+
export function exportMindMapAsPdf(
|
|
135
|
+
nodes: MindMapNode[],
|
|
136
|
+
options: ExportPdfOptions = {},
|
|
137
|
+
): void {
|
|
138
|
+
const { title, blocks } = getDocumentStructure(nodes);
|
|
139
|
+
if (!title && blocks.length === 0) return;
|
|
140
|
+
|
|
141
|
+
const filename = options.filename ?? `mind-map-${Date.now()}`;
|
|
142
|
+
const orientation = options.orientation ?? "portrait";
|
|
143
|
+
|
|
144
|
+
const doc = new jsPDF({
|
|
145
|
+
orientation,
|
|
146
|
+
unit: "pt",
|
|
147
|
+
format: "a4",
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
const pageW = doc.internal.pageSize.getWidth();
|
|
151
|
+
const pageH = doc.internal.pageSize.getHeight();
|
|
152
|
+
const contentWidth = pageW - MARGIN_X * 2;
|
|
153
|
+
const maxY = pageH - MARGIN_BOTTOM - FOOTER_Y_OFFSET;
|
|
154
|
+
const pageNum = { current: 1 };
|
|
155
|
+
let y = MARGIN_TOP;
|
|
156
|
+
|
|
157
|
+
// —— Título (nó central) ——
|
|
158
|
+
if (title) {
|
|
159
|
+
doc.setFont("helvetica", "bold");
|
|
160
|
+
doc.setFontSize(TITLE_FONT_SIZE);
|
|
161
|
+
setTextColor(doc, COLOR_TITLE);
|
|
162
|
+
const titleLines = doc.splitTextToSize(title, contentWidth);
|
|
163
|
+
const titleLead = TITLE_FONT_SIZE * TITLE_LINE_HEIGHT;
|
|
164
|
+
for (const line of titleLines) {
|
|
165
|
+
doc.text(line, MARGIN_X, y + TITLE_FONT_SIZE * 0.9);
|
|
166
|
+
y += titleLead;
|
|
167
|
+
}
|
|
168
|
+
y += 12;
|
|
169
|
+
setDrawColor(doc, COLOR_RULE);
|
|
170
|
+
doc.setLineWidth(TITLE_RULE_THICKNESS);
|
|
171
|
+
doc.line(MARGIN_X, y, MARGIN_X + TITLE_BOTTOM_RULE_WIDTH, y);
|
|
172
|
+
y += SPACE_AFTER_TITLE;
|
|
173
|
+
} else {
|
|
174
|
+
y = CONTENT_START;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// —— Blocos de texto (cada nó = um bloco com parágrafos) ——
|
|
178
|
+
for (let i = 0; i < blocks.length; i++) {
|
|
179
|
+
const block = blocks[i];
|
|
180
|
+
const prevLevel = i > 0 ? blocks[i - 1].level : 0;
|
|
181
|
+
const { level, paragraphs } = block;
|
|
182
|
+
const isLevel1 = level === 1;
|
|
183
|
+
|
|
184
|
+
// Ritmo vertical: mais espaço antes de novo “capítulo” (nível 1)
|
|
185
|
+
if (isLevel1 && prevLevel >= 1) y += BLOCK_VERTICAL_RHYTHM_MAJOR;
|
|
186
|
+
else if (i > 0) y += BLOCK_VERTICAL_RHYTHM;
|
|
187
|
+
|
|
188
|
+
const indent = MARGIN_X + (level - 1) * INDENT_PER_LEVEL;
|
|
189
|
+
const textAreaWidth = contentWidth - (level - 1) * INDENT_PER_LEVEL - ACCENT_BAR_WIDTH - BAR_TO_TEXT;
|
|
190
|
+
const textX = indent + ACCENT_BAR_WIDTH + BAR_TO_TEXT;
|
|
191
|
+
|
|
192
|
+
const headingSize = FONT_SIZES[Math.min(level - 1, FONT_SIZES.length - 1)];
|
|
193
|
+
const bodySize = BLOCK_BODY_SIZE;
|
|
194
|
+
const headingLead = headingSize * BLOCK_HEADING_LEADING;
|
|
195
|
+
const bodyLead = bodySize * BLOCK_BODY_LEADING;
|
|
196
|
+
|
|
197
|
+
// Pré-calcular altura do bloco para desenhar a barra antes do texto
|
|
198
|
+
let blockHeight = 0;
|
|
199
|
+
for (let p = 0; p < paragraphs.length; p++) {
|
|
200
|
+
const isFirst = p === 0;
|
|
201
|
+
doc.setFontSize(isFirst ? headingSize : bodySize);
|
|
202
|
+
const lead = isFirst ? headingLead : bodyLead;
|
|
203
|
+
const lines = doc.splitTextToSize(paragraphs[p], textAreaWidth);
|
|
204
|
+
blockHeight += lines.length * lead;
|
|
205
|
+
if (p < paragraphs.length - 1) blockHeight += PARAGRAPH_GAP;
|
|
206
|
+
}
|
|
207
|
+
const barHeight = Math.min(blockHeight + 4, maxY - y + 6);
|
|
208
|
+
if (barHeight > 0) {
|
|
209
|
+
setFillColor(doc, COLOR_ACCENT);
|
|
210
|
+
doc.rect(indent, y - 2, ACCENT_BAR_WIDTH, barHeight, "F");
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
doc.setFont("helvetica", "bold");
|
|
214
|
+
doc.setFontSize(headingSize);
|
|
215
|
+
setTextColor(doc, level === 1 ? COLOR_LEVEL_1_HEAD : COLOR_LEVEL_2_PLUS);
|
|
216
|
+
|
|
217
|
+
let blockY = y;
|
|
218
|
+
|
|
219
|
+
for (let p = 0; p < paragraphs.length; p++) {
|
|
220
|
+
const isFirst = p === 0;
|
|
221
|
+
const fontSize = isFirst ? headingSize : bodySize;
|
|
222
|
+
const leading = isFirst ? headingLead : bodyLead;
|
|
223
|
+
doc.setFontSize(fontSize);
|
|
224
|
+
doc.setFont("helvetica", isFirst ? "bold" : "normal");
|
|
225
|
+
setTextColor(doc, isFirst && level === 1 ? COLOR_LEVEL_1_HEAD : isFirst ? COLOR_LEVEL_2_PLUS : COLOR_LEVEL_1_BODY);
|
|
226
|
+
|
|
227
|
+
const lines = doc.splitTextToSize(paragraphs[p], textAreaWidth);
|
|
228
|
+
for (const line of lines) {
|
|
229
|
+
blockY = ensurePage(doc, blockY, maxY, pageNum, pageW, pageH);
|
|
230
|
+
doc.text(line, textX, blockY + fontSize * 0.85);
|
|
231
|
+
blockY += leading;
|
|
232
|
+
}
|
|
233
|
+
if (p < paragraphs.length - 1) blockY += PARAGRAPH_GAP;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
y = blockY;
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
drawFooter(doc, pageNum.current, pageW, pageH);
|
|
240
|
+
doc.save(`${filename}.pdf`);
|
|
241
|
+
}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import type { MindMapNode } from "../state/mindMap";
|
|
2
|
+
import { useMindMapState } from "../state/mindMap";
|
|
3
|
+
import {
|
|
4
|
+
getMineMapProjection,
|
|
5
|
+
MINE_MAP_HEIGHT,
|
|
6
|
+
MINE_MAP_WIDTH,
|
|
7
|
+
} from "../components/mindMap/MineMap";
|
|
8
|
+
|
|
9
|
+
function escapeXml(unsafe: string): string {
|
|
10
|
+
return unsafe
|
|
11
|
+
.replace(/&/g, "&")
|
|
12
|
+
.replace(/</g, "<")
|
|
13
|
+
.replace(/>/g, ">")
|
|
14
|
+
.replace(/"/g, """)
|
|
15
|
+
.replace(/'/g, "'");
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Builds the MineMap (minimap) SVG as a data URL for preview thumbnails.
|
|
20
|
+
* Only the node circles, no viewport rect, transparent background.
|
|
21
|
+
*/
|
|
22
|
+
export function getMindMapPreviewDataUrl(nodes: MindMapNode[]): string | null {
|
|
23
|
+
if (nodes.length === 0) return null;
|
|
24
|
+
|
|
25
|
+
const state = useMindMapState.getState();
|
|
26
|
+
const root = typeof document !== "undefined"
|
|
27
|
+
? document.querySelector<HTMLElement>("[data-nodex-root]")
|
|
28
|
+
: null;
|
|
29
|
+
const rootSize = root
|
|
30
|
+
? (() => {
|
|
31
|
+
const r = root.getBoundingClientRect();
|
|
32
|
+
return { w: r.width, h: r.height };
|
|
33
|
+
})()
|
|
34
|
+
: { w: 0, h: 0 };
|
|
35
|
+
|
|
36
|
+
const { nodesToRender } = getMineMapProjection(
|
|
37
|
+
nodes,
|
|
38
|
+
state.offset,
|
|
39
|
+
state.scale,
|
|
40
|
+
rootSize,
|
|
41
|
+
);
|
|
42
|
+
|
|
43
|
+
if (nodesToRender.length === 0) return null;
|
|
44
|
+
|
|
45
|
+
const circles = nodesToRender
|
|
46
|
+
.map(
|
|
47
|
+
(n) =>
|
|
48
|
+
`<circle cx="${n.x}" cy="${n.y}" r="${n.r}" fill="${escapeXml(n.color)}"/>`,
|
|
49
|
+
)
|
|
50
|
+
.join("\n");
|
|
51
|
+
|
|
52
|
+
const svg = `<?xml version="1.0" encoding="UTF-8"?>
|
|
53
|
+
<svg xmlns="http://www.w3.org/2000/svg" width="${MINE_MAP_WIDTH}" height="${MINE_MAP_HEIGHT}" viewBox="0 0 ${MINE_MAP_WIDTH} ${MINE_MAP_HEIGHT}">
|
|
54
|
+
${circles}
|
|
55
|
+
</svg>`;
|
|
56
|
+
|
|
57
|
+
if (typeof btoa === "undefined") return null;
|
|
58
|
+
const encoded = btoa(unescape(encodeURIComponent(svg)));
|
|
59
|
+
return `data:image/svg+xml;base64,${encoded}`;
|
|
60
|
+
}
|
package/styles.css
ADDED
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
/*! tailwindcss v4.1.0 | MIT License | https://tailwindcss.com */
|
|
2
|
+
@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){@layer base{*,:before,:after,::backdrop{--tw-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--tw-scale-x:1;--tw-scale-y:1;--tw-scale-z:1;--tw-rotate-x:rotateX(0);--tw-rotate-y:rotateY(0);--tw-rotate-z:rotateZ(0);--tw-skew-x:skewX(0);--tw-skew-y:skewY(0);--tw-space-y-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-outline-style:solid;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial;--tw-animation-delay:0s;--tw-animation-direction:normal;--tw-animation-duration:initial;--tw-animation-fill-mode:none;--tw-animation-iteration-count:1;--tw-enter-blur:0;--tw-enter-opacity:1;--tw-enter-rotate:0;--tw-enter-scale:1;--tw-enter-translate-x:0;--tw-enter-translate-y:0;--tw-exit-blur:0;--tw-exit-opacity:1;--tw-exit-rotate:0;--tw-exit-scale:1;--tw-exit-translate-x:0;--tw-exit-translate-y:0}}}@layer theme{:root,:host{--font-sans:ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--font-mono:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--color-amber-600:oklch(66.6% .179 58.318);--color-emerald-600:oklch(59.6% .145 163.225);--color-sky-600:oklch(58.8% .158 241.966);--color-slate-50:oklch(98.4% .003 247.858);--color-slate-100:oklch(96.8% .007 247.896);--color-slate-200:oklch(92.9% .013 255.508);--color-slate-300:oklch(86.9% .022 252.894);--color-slate-400:oklch(70.4% .04 256.788);--color-slate-500:oklch(55.4% .046 257.417);--color-slate-600:oklch(44.6% .043 257.281);--color-slate-700:oklch(37.2% .044 257.287);--color-slate-900:oklch(20.8% .042 265.755);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-lg:32rem;--text-xs:.75rem;--text-xs--line-height:calc(1/.75);--text-sm:.875rem;--text-sm--line-height:calc(1.25/.875);--text-base:1rem;--text-base--line-height:calc(1.5/1);--text-lg:1.125rem;--text-lg--line-height:calc(1.75/1.125);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-normal:0em;--tracking-wide:.025em;--radius-xs:.125rem;--radius-sm:.25rem;--radius-md:.375rem;--radius-lg:.5rem;--radius-xl:.75rem;--radius-2xl:1rem;--animate-spin:spin 1s linear infinite;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4,0,.2,1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono)}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:color-mix(in oklab,currentColor 50%,transparent)}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.pointer-events-none{pointer-events:none}.visible{visibility:visible}.sr-only{clip:rect(0,0,0,0);white-space:nowrap;border-width:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.inset-0{inset:calc(var(--spacing)*0)}.top-0{top:calc(var(--spacing)*0)}.top-1\/2{top:50%}.top-2{top:calc(var(--spacing)*2)}.top-4{top:calc(var(--spacing)*4)}.top-\[50\%\]{top:50%}.-right-1\.5{right:calc(var(--spacing)*-1.5)}.-right-2\.5{right:calc(var(--spacing)*-2.5)}.-right-3{right:calc(var(--spacing)*-3)}.right-0{right:calc(var(--spacing)*0)}.right-2{right:calc(var(--spacing)*2)}.right-4{right:calc(var(--spacing)*4)}.bottom-4{bottom:calc(var(--spacing)*4)}.-left-1\.5{left:calc(var(--spacing)*-1.5)}.-left-2\.5{left:calc(var(--spacing)*-2.5)}.-left-3{left:calc(var(--spacing)*-3)}.-left-3\.5{left:calc(var(--spacing)*-3.5)}.left-0{left:calc(var(--spacing)*0)}.left-3{left:calc(var(--spacing)*3)}.left-\[50\%\]{left:50%}.z-50{z-index:50}.-mx-1{margin-inline:calc(var(--spacing)*-1)}.my-1{margin-block:calc(var(--spacing)*1)}.mt-3{margin-top:calc(var(--spacing)*3)}.-mr-6{margin-right:calc(var(--spacing)*-6)}.ml-1\.5{margin-left:calc(var(--spacing)*1.5)}.ml-100{margin-left:calc(var(--spacing)*100)}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline{display:inline}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.size-3\.5{width:calc(var(--spacing)*3.5);height:calc(var(--spacing)*3.5)}.size-4{width:calc(var(--spacing)*4);height:calc(var(--spacing)*4)}.h-3{height:calc(var(--spacing)*3)}.h-4{height:calc(var(--spacing)*4)}.h-5{height:calc(var(--spacing)*5)}.h-6{height:calc(var(--spacing)*6)}.h-8{height:calc(var(--spacing)*8)}.h-9{height:calc(var(--spacing)*9)}.h-10{height:calc(var(--spacing)*10)}.h-\[6px\]{height:6px}.h-\[520px\]{height:520px}.h-\[calc\(100dvh-11rem\)\]{height:calc(100dvh - 11rem)}.h-\[var\(--radix-select-trigger-height\)\]{height:var(--radix-select-trigger-height)}.h-full{height:100%}.h-px{height:1px}.max-h-\(--radix-select-content-available-height\){max-height:var(--radix-select-content-available-height)}.max-h-8{max-height:calc(var(--spacing)*8)}.max-h-16{max-height:calc(var(--spacing)*16)}.min-h-0{min-height:calc(var(--spacing)*0)}.min-h-8{min-height:calc(var(--spacing)*8)}.min-h-\[38rem\]{min-height:38rem}.min-h-\[480px\]{min-height:480px}.w-3{width:calc(var(--spacing)*3)}.w-4{width:calc(var(--spacing)*4)}.w-5{width:calc(var(--spacing)*5)}.w-6{width:calc(var(--spacing)*6)}.w-8{width:calc(var(--spacing)*8)}.w-52{width:calc(var(--spacing)*52)}.w-72{width:calc(var(--spacing)*72)}.w-auto{width:auto}.w-fit{width:fit-content}.w-full{width:100%}.max-w-8{max-width:calc(var(--spacing)*8)}.max-w-\[520px\]{max-width:520px}.max-w-\[calc\(100\%-2rem\)\]{max-width:calc(100% - 2rem)}.max-w-px{max-width:1px}.min-w-0{min-width:calc(var(--spacing)*0)}.min-w-8{min-width:calc(var(--spacing)*8)}.min-w-9{min-width:calc(var(--spacing)*9)}.min-w-10{min-width:calc(var(--spacing)*10)}.min-w-\[8rem\]{min-width:8rem}.min-w-\[var\(--radix-select-trigger-width\)\]{min-width:var(--radix-select-trigger-width)}.min-w-fit{min-width:fit-content}.flex-1{flex:1}.shrink-0{flex-shrink:0}.origin-\(--radix-popover-content-transform-origin\){transform-origin:var(--radix-popover-content-transform-origin)}.origin-\(--radix-select-content-transform-origin\){transform-origin:var(--radix-select-content-transform-origin)}.translate-x-\[-50\%\]{--tw-translate-x:-50%;translate:var(--tw-translate-x)var(--tw-translate-y)}.-translate-y-1\/2{--tw-translate-y:calc(calc(1/2*100%)*-1);translate:var(--tw-translate-x)var(--tw-translate-y)}.translate-y-0{--tw-translate-y:calc(var(--spacing)*0);translate:var(--tw-translate-x)var(--tw-translate-y)}.translate-y-2{--tw-translate-y:calc(var(--spacing)*2);translate:var(--tw-translate-x)var(--tw-translate-y)}.translate-y-\[-50\%\]{--tw-translate-y:-50%;translate:var(--tw-translate-x)var(--tw-translate-y)}.scale-100{--tw-scale-x:100%;--tw-scale-y:100%;--tw-scale-z:100%;scale:var(--tw-scale-x)var(--tw-scale-y)}.transform{transform:var(--tw-rotate-x)var(--tw-rotate-y)var(--tw-rotate-z)var(--tw-skew-x)var(--tw-skew-y)}.animate-spin{animation:var(--animate-spin)}.cursor-default{cursor:default}.cursor-grab{cursor:grab}.resize{resize:both}.scroll-my-1{scroll-margin-block:calc(var(--spacing)*1)}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing)*1)}.gap-2{gap:calc(var(--spacing)*2)}.gap-3{gap:calc(var(--spacing)*3)}.gap-4{gap:calc(var(--spacing)*4)}.gap-10{gap:calc(var(--spacing)*10)}.gap-\[--spacing\(var\(--gap\)\)\]{gap:calc(var(--spacing)*var(--gap))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*2)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*2)*calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-x-hidden{overflow-x:hidden}.overflow-y-auto{overflow-y:auto}.rounded-2xl{border-radius:var(--radius-2xl)}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.rounded-sm{border-radius:var(--radius-sm)}.rounded-xl{border-radius:var(--radius-xl)}.rounded-xs{border-radius:var(--radius-xs)}.rounded-tl-none{border-top-left-radius:0}.rounded-tr-none{border-top-right-radius:0}.rounded-br-none{border-bottom-right-radius:0}.rounded-bl-none{border-bottom-left-radius:0}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-r-0{border-right-style:var(--tw-border-style);border-right-width:0}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-input{border-color:var(--input)}.border-slate-100{border-color:var(--color-slate-100)}.border-slate-200{border-color:var(--color-slate-200)}.border-slate-300{border-color:var(--color-slate-300)}.bg-background{background-color:var(--background)}.bg-black\/50{background-color:#00000080}@supports (color:color-mix(in lab, red, red)){.bg-black\/50{background-color:color-mix(in oklab,var(--color-black)50%,transparent)}}.bg-border{background-color:var(--border)}.bg-popover{background-color:var(--popover)}.bg-slate-50{background-color:var(--color-slate-50)}.bg-slate-100{background-color:var(--color-slate-100)}.bg-transparent{background-color:#0000}.bg-white{background-color:var(--color-white)}.bg-white\/40{background-color:#fff6}@supports (color:color-mix(in lab, red, red)){.bg-white\/40{background-color:color-mix(in oklab,var(--color-white)40%,transparent)}}.bg-white\/50{background-color:#ffffff80}@supports (color:color-mix(in lab, red, red)){.bg-white\/50{background-color:color-mix(in oklab,var(--color-white)50%,transparent)}}.bg-white\/90{background-color:#ffffffe6}@supports (color:color-mix(in lab, red, red)){.bg-white\/90{background-color:color-mix(in oklab,var(--color-white)90%,transparent)}}.object-cover{object-fit:cover}.p-0{padding:calc(var(--spacing)*0)}.p-1{padding:calc(var(--spacing)*1)}.p-2{padding:calc(var(--spacing)*2)}.p-4{padding:calc(var(--spacing)*4)}.p-6{padding:calc(var(--spacing)*6)}.px-1\.5{padding-inline:calc(var(--spacing)*1.5)}.px-2{padding-inline:calc(var(--spacing)*2)}.px-2\.5{padding-inline:calc(var(--spacing)*2.5)}.px-3{padding-inline:calc(var(--spacing)*3)}.px-4{padding-inline:calc(var(--spacing)*4)}.py-1{padding-block:calc(var(--spacing)*1)}.py-1\.5{padding-block:calc(var(--spacing)*1.5)}.py-2{padding-block:calc(var(--spacing)*2)}.py-3{padding-block:calc(var(--spacing)*3)}.pr-6{padding-right:calc(var(--spacing)*6)}.pr-8{padding-right:calc(var(--spacing)*8)}.pb-2{padding-bottom:calc(var(--spacing)*2)}.pl-2{padding-left:calc(var(--spacing)*2)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.font-sans{font-family:var(--font-sans)}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[9px\]{font-size:9px}.text-\[10px\]{font-size:10px}.leading-none{--tw-leading:1;line-height:1}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-normal{--tw-tracking:var(--tracking-normal);letter-spacing:var(--tracking-normal)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.whitespace-nowrap{white-space:nowrap}.whitespace-pre{white-space:pre}.text-amber-600{color:var(--color-amber-600)}.text-emerald-600{color:var(--color-emerald-600)}.text-muted-foreground{color:var(--muted-foreground)}.text-popover-foreground{color:var(--popover-foreground)}.text-sky-600{color:var(--color-sky-600)}.text-slate-400{color:var(--color-slate-400)}.text-slate-500{color:var(--color-slate-500)}.text-slate-600{color:var(--color-slate-600)}.text-slate-700{color:var(--color-slate-700)}.text-slate-900{color:var(--color-slate-900)}.normal-case{text-transform:none}.uppercase{text-transform:uppercase}.italic{font-style:italic}.not-italic{font-style:normal}.opacity-0{opacity:0}.opacity-50{opacity:.5}.opacity-70{opacity:.7}.opacity-100{opacity:1}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a),0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-md{--tw-shadow:0 4px 6px -1px var(--tw-shadow-color,#0000001a),0 2px 4px -2px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a),0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-xs{--tw-shadow:0 1px 2px 0 var(--tw-shadow-color,#0000000d);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.ring-offset-background{--tw-ring-offset-color:var(--background)}.outline-hidden{--tw-outline-style:none;outline-style:none}@media (forced-colors:active){.outline-hidden{outline-offset:2px;outline:2px solid #0000}}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.backdrop-blur{--tw-backdrop-blur:blur(8px);-webkit-backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,)}.transition{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to,opacity,box-shadow,transform,translate,scale,rotate,filter,-webkit-backdrop-filter,backdrop-filter;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-\[color\,box-shadow\]{transition-property:color,box-shadow;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-150{--tw-duration:.15s;transition-duration:.15s}.duration-200{--tw-duration:.2s;transition-duration:.2s}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}.select-text{-webkit-user-select:text;user-select:text}.last\:border-none:last-child{--tw-border-style:none;border-style:none}.last\:pb-0:last-child{padding-bottom:calc(var(--spacing)*0)}@media (hover:hover){.hover\:bg-accent:hover{background-color:var(--accent)}.hover\:bg-muted:hover{background-color:var(--muted)}.hover\:bg-slate-50:hover{background-color:var(--color-slate-50)}.hover\:bg-slate-100:hover{background-color:var(--color-slate-100)}.hover\:text-accent-foreground:hover{color:var(--accent-foreground)}.hover\:text-muted-foreground:hover{color:var(--muted-foreground)}.hover\:text-slate-700:hover{color:var(--color-slate-700)}.hover\:opacity-100:hover{opacity:1}}.focus\:z-10:focus{z-index:10}.focus\:bg-accent:focus{background-color:var(--accent)}.focus\:text-accent-foreground:focus{color:var(--accent-foreground)}.focus\:ring-2:focus{--tw-ring-shadow:var(--tw-ring-inset,)0 0 0 calc(2px + var(--tw-ring-offset-width))var(--tw-ring-color,currentColor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus\:ring-ring:focus{--tw-ring-color:var(--ring)}.focus\:ring-offset-2:focus{--tw-ring-offset-width:2px;--tw-ring-offset-shadow:var(--tw-ring-inset,)0 0 0 var(--tw-ring-offset-width)var(--tw-ring-offset-color)}.focus\:outline-hidden:focus{--tw-outline-style:none;outline-style:none}@media (forced-colors:active){.focus\:outline-hidden:focus{outline-offset:2px;outline:2px solid #0000}}.focus-visible\:z-10:focus-visible{z-index:10}.focus-visible\:border-ring:focus-visible{border-color:var(--ring)}.focus-visible\:ring-\[3px\]:focus-visible{--tw-ring-shadow:var(--tw-ring-inset,)0 0 0 calc(3px + var(--tw-ring-offset-width))var(--tw-ring-color,currentColor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus-visible\:ring-ring\/50:focus-visible{--tw-ring-color:color-mix(in oklab,var(--ring)50%,transparent)}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-50:disabled{opacity:.5}.aria-invalid\:border-destructive[aria-invalid=true]{border-color:var(--destructive)}.aria-invalid\:ring-destructive\/20[aria-invalid=true]{--tw-ring-color:color-mix(in oklab,var(--destructive)20%,transparent)}.data-\[bold\=true\]\:font-bold[data-bold=true]{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.data-\[bold\=true\]\:font-semibold[data-bold=true]{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.data-\[disabled\]\:pointer-events-none[data-disabled]{pointer-events:none}.data-\[disabled\]\:opacity-50[data-disabled]{opacity:.5}.data-\[italic\=true\]\:italic[data-italic=true]{font-style:italic}.data-\[left\=false\]\:left-0[data-left=false]{left:calc(var(--spacing)*0)}.data-\[placeholder\]\:text-muted-foreground[data-placeholder]{color:var(--muted-foreground)}.data-\[selected\=true\]\:hidden[data-selected=true]{display:none}.data-\[read-only\=false\]\:data-\[selected\=true\]\:data-\[editing\=false\]\:flex[data-read-only=false][data-selected=true][data-editing=false]{display:flex}.data-\[side\=bottom\]\:translate-y-1[data-side=bottom]{--tw-translate-y:calc(var(--spacing)*1);translate:var(--tw-translate-x)var(--tw-translate-y)}.data-\[side\=bottom\]\:slide-in-from-top-2[data-side=bottom]{--tw-enter-translate-y:calc(2*var(--spacing)*-1)}.data-\[side\=left\]\:-translate-x-1[data-side=left]{--tw-translate-x:calc(var(--spacing)*-1);translate:var(--tw-translate-x)var(--tw-translate-y)}.data-\[side\=left\]\:slide-in-from-right-2[data-side=left]{--tw-enter-translate-x:calc(2*var(--spacing))}.data-\[side\=right\]\:translate-x-1[data-side=right]{--tw-translate-x:calc(var(--spacing)*1);translate:var(--tw-translate-x)var(--tw-translate-y)}.data-\[side\=right\]\:slide-in-from-left-2[data-side=right]{--tw-enter-translate-x:calc(2*var(--spacing)*-1)}.data-\[side\=top\]\:-translate-y-1[data-side=top]{--tw-translate-y:calc(var(--spacing)*-1);translate:var(--tw-translate-x)var(--tw-translate-y)}.data-\[side\=top\]\:slide-in-from-bottom-2[data-side=top]{--tw-enter-translate-y:calc(2*var(--spacing))}.data-\[size\=default\]\:h-9[data-size=default]{height:calc(var(--spacing)*9)}.data-\[size\=sm\]\:h-8[data-size=sm]{height:calc(var(--spacing)*8)}:is(.\*\:data-\[slot\=select-value\]\:line-clamp-1>*)[data-slot=select-value]{-webkit-line-clamp:1;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}:is(.\*\:data-\[slot\=select-value\]\:flex>*)[data-slot=select-value]{display:flex}:is(.\*\:data-\[slot\=select-value\]\:items-center>*)[data-slot=select-value]{align-items:center}:is(.\*\:data-\[slot\=select-value\]\:gap-2>*)[data-slot=select-value]{gap:calc(var(--spacing)*2)}.data-\[spacing\=0\]\:rounded-none[data-spacing="0"]{border-radius:0}.data-\[spacing\=0\]\:shadow-none[data-spacing="0"]{--tw-shadow:0 0 #0000;box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.data-\[spacing\=0\]\:first\:rounded-l-md[data-spacing="0"]:first-child{border-top-left-radius:var(--radius-md);border-bottom-left-radius:var(--radius-md)}.data-\[spacing\=0\]\:last\:rounded-r-md[data-spacing="0"]:last-child{border-top-right-radius:var(--radius-md);border-bottom-right-radius:var(--radius-md)}.data-\[state\=closed\]\:animate-out[data-state=closed]{animation:exit var(--tw-animation-duration,var(--tw-duration,.15s))var(--tw-ease,ease)var(--tw-animation-delay,0s)var(--tw-animation-iteration-count,1)var(--tw-animation-direction,normal)var(--tw-animation-fill-mode,none)}.data-\[state\=closed\]\:fade-out-0[data-state=closed]{--tw-exit-opacity:0}.data-\[state\=closed\]\:zoom-out-95[data-state=closed]{--tw-exit-scale:.95}.data-\[state\=on\]\:bg-accent[data-state=on]{background-color:var(--accent)}.data-\[state\=on\]\:text-accent-foreground[data-state=on]{color:var(--accent-foreground)}.data-\[state\=open\]\:animate-in[data-state=open]{animation:enter var(--tw-animation-duration,var(--tw-duration,.15s))var(--tw-ease,ease)var(--tw-animation-delay,0s)var(--tw-animation-iteration-count,1)var(--tw-animation-direction,normal)var(--tw-animation-fill-mode,none)}.data-\[state\=open\]\:bg-accent[data-state=open]{background-color:var(--accent)}.data-\[state\=open\]\:text-muted-foreground[data-state=open]{color:var(--muted-foreground)}.data-\[state\=open\]\:fade-in-0[data-state=open]{--tw-enter-opacity:0}.data-\[state\=open\]\:zoom-in-95[data-state=open]{--tw-enter-scale:.95}.data-\[spacing\=0\]\:data-\[variant\=outline\]\:border-l-0[data-spacing="0"][data-variant=outline]{border-left-style:var(--tw-border-style);border-left-width:0}.data-\[spacing\=0\]\:data-\[variant\=outline\]\:first\:border-l[data-spacing="0"][data-variant=outline]:first-child{border-left-style:var(--tw-border-style);border-left-width:1px}.data-\[spacing\=default\]\:data-\[variant\=outline\]\:shadow-xs[data-spacing=default][data-variant=outline]{--tw-shadow:0 1px 2px 0 var(--tw-shadow-color,#0000000d);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.data-\[zen\=true\]\:pointer-events-none[data-zen=true]{pointer-events:none}.data-\[zen\=true\]\:max-h-0[data-zen=true]{max-height:calc(var(--spacing)*0)}.data-\[zen\=true\]\:-translate-y-2[data-zen=true]{--tw-translate-y:calc(var(--spacing)*-2);translate:var(--tw-translate-x)var(--tw-translate-y)}.data-\[zen\=true\]\:translate-y-0[data-zen=true]{--tw-translate-y:calc(var(--spacing)*0);translate:var(--tw-translate-x)var(--tw-translate-y)}.data-\[zen\=true\]\:scale-90[data-zen=true]{--tw-scale-x:90%;--tw-scale-y:90%;--tw-scale-z:90%;scale:var(--tw-scale-x)var(--tw-scale-y)}.data-\[zen\=true\]\:opacity-0[data-zen=true]{opacity:0}.data-\[zen\=true\]\:opacity-100[data-zen=true]{opacity:1}@media (min-width:40rem){.sm\:max-w-lg{max-width:var(--container-lg)}.sm\:flex-row{flex-direction:row}.sm\:justify-end{justify-content:flex-end}.sm\:text-left{text-align:left}}.dark\:bg-input\/30:is(.dark *){background-color:color-mix(in oklab,var(--input)30%,transparent)}@media (hover:hover){.dark\:hover\:bg-input\/50:is(.dark *):hover{background-color:color-mix(in oklab,var(--input)50%,transparent)}}.dark\:aria-invalid\:ring-destructive\/40:is(.dark *)[aria-invalid=true]{--tw-ring-color:color-mix(in oklab,var(--destructive)40%,transparent)}.\[\&_svg\]\:pointer-events-none svg{pointer-events:none}.\[\&_svg\]\:shrink-0 svg{flex-shrink:0}.\[\&_svg\:not\(\[class\*\=\'size-\'\]\)\]\:size-4 svg:not([class*=size-]){width:calc(var(--spacing)*4);height:calc(var(--spacing)*4)}.\[\&_svg\:not\(\[class\*\=\'text-\'\]\)\]\:text-muted-foreground svg:not([class*=text-]){color:var(--muted-foreground)}:is(.\*\:\[span\]\:last\:flex>*):is(span):last-child{display:flex}:is(.\*\:\[span\]\:last\:items-center>*):is(span):last-child{align-items:center}:is(.\*\:\[span\]\:last\:gap-2>*):is(span):last-child{gap:calc(var(--spacing)*2)}.scrollbar{scrollbar-width:thin;scrollbar-color:#94a3b8 transparent}.scrollbar::-webkit-scrollbar{width:10px;height:10px}.scrollbar::-webkit-scrollbar-track{background:0 0}.scrollbar::-webkit-scrollbar-thumb{background-color:#94a3b8;background-clip:padding-box;border:2px solid #0000;border-radius:9999px}.scrollbar::-webkit-scrollbar-thumb:hover{background-color:#64748b}}@property --tw-animation-delay{syntax:"*";inherits:false;initial-value:0s}@property --tw-animation-direction{syntax:"*";inherits:false;initial-value:normal}@property --tw-animation-duration{syntax:"*";inherits:false}@property --tw-animation-fill-mode{syntax:"*";inherits:false;initial-value:none}@property --tw-animation-iteration-count{syntax:"*";inherits:false;initial-value:1}@property --tw-enter-blur{syntax:"*";inherits:false;initial-value:0}@property --tw-enter-opacity{syntax:"*";inherits:false;initial-value:1}@property --tw-enter-rotate{syntax:"*";inherits:false;initial-value:0}@property --tw-enter-scale{syntax:"*";inherits:false;initial-value:1}@property --tw-enter-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-enter-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-exit-blur{syntax:"*";inherits:false;initial-value:0}@property --tw-exit-opacity{syntax:"*";inherits:false;initial-value:1}@property --tw-exit-rotate{syntax:"*";inherits:false;initial-value:0}@property --tw-exit-scale{syntax:"*";inherits:false;initial-value:1}@property --tw-exit-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-exit-translate-y{syntax:"*";inherits:false;initial-value:0}:root{--background:oklch(1 0 0);--foreground:oklch(.145 0 0);--card:oklch(1 0 0);--card-foreground:oklch(.145 0 0);--popover:oklch(1 0 0);--popover-foreground:oklch(.145 0 0);--primary:oklch(.205 0 0);--primary-foreground:oklch(.985 0 0);--secondary:oklch(.97 0 0);--secondary-foreground:oklch(.205 0 0);--muted:oklch(.97 0 0);--muted-foreground:oklch(.556 0 0);--accent:oklch(.97 0 0);--accent-foreground:oklch(.205 0 0);--destructive:oklch(.577 .245 27.325);--destructive-foreground:oklch(.577 .245 27.325);--border:oklch(.922 0 0);--input:oklch(.922 0 0);--ring:oklch(.708 0 0)}.dark{--background:oklch(.145 0 0);--foreground:oklch(.985 0 0);--card:oklch(.145 0 0);--card-foreground:oklch(.985 0 0);--popover:oklch(.145 0 0);--popover-foreground:oklch(.985 0 0);--primary:oklch(.985 0 0);--primary-foreground:oklch(.205 0 0);--secondary:oklch(.269 0 0);--secondary-foreground:oklch(.985 0 0);--muted:oklch(.269 0 0);--muted-foreground:oklch(.708 0 0);--accent:oklch(.269 0 0);--accent-foreground:oklch(.985 0 0);--destructive:oklch(.396 .141 25.723);--destructive-foreground:oklch(.637 .237 25.331);--border:oklch(.269 0 0);--input:oklch(.269 0 0);--ring:oklch(.439 0 0)}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@property --tw-rotate-x{syntax:"*";inherits:false;initial-value:rotateX(0)}@property --tw-rotate-y{syntax:"*";inherits:false;initial-value:rotateY(0)}@property --tw-rotate-z{syntax:"*";inherits:false;initial-value:rotateZ(0)}@property --tw-skew-x{syntax:"*";inherits:false;initial-value:skewX(0)}@property --tw-skew-y{syntax:"*";inherits:false;initial-value:skewY(0)}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-outline-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(360deg)}}@keyframes enter{0%{opacity:var(--tw-enter-opacity,1);transform:translate3d(var(--tw-enter-translate-x,0),var(--tw-enter-translate-y,0),0)scale3d(var(--tw-enter-scale,1),var(--tw-enter-scale,1),var(--tw-enter-scale,1))rotate(var(--tw-enter-rotate,0));filter:blur(var(--tw-enter-blur,0))}}@keyframes exit{to{opacity:var(--tw-exit-opacity,1);transform:translate3d(var(--tw-exit-translate-x,0),var(--tw-exit-translate-y,0),0)scale3d(var(--tw-exit-scale,1),var(--tw-exit-scale,1),var(--tw-exit-scale,1))rotate(var(--tw-exit-rotate,0));filter:blur(var(--tw-exit-blur,0))}}
|