@tomehq/theme 0.2.1 → 0.2.2

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.
@@ -0,0 +1,1468 @@
1
+ // src/entry.tsx
2
+ import { useState as useState3, useEffect as useEffect3, useCallback as useCallback3 } from "react";
3
+ import { createRoot } from "react-dom/client";
4
+
5
+ // src/Shell.tsx
6
+ import { useState as useState2, useEffect as useEffect2, useRef as useRef2, useCallback as useCallback2 } from "react";
7
+
8
+ // src/presets.ts
9
+ var THEME_PRESETS = {
10
+ amber: {
11
+ dark: {
12
+ bg: "#09090b",
13
+ sf: "#111114",
14
+ sfH: "#18181c",
15
+ bd: "#1e1e24",
16
+ tx: "#e4e4e7",
17
+ tx2: "#a1a1aa",
18
+ txM: "#919199",
19
+ ac: "#e8a845",
20
+ acD: "rgba(232,168,69,0.12)",
21
+ acT: "#fbbf24",
22
+ cdBg: "#0c0c0f",
23
+ cdTx: "#c4c4cc",
24
+ sbBg: "#0c0c0e",
25
+ hdBg: "rgba(9,9,11,0.85)"
26
+ },
27
+ light: {
28
+ bg: "#fafaf9",
29
+ sf: "#ffffff",
30
+ sfH: "#f5f5f4",
31
+ bd: "#e7e5e4",
32
+ tx: "#1c1917",
33
+ tx2: "#57534e",
34
+ txM: "#706b66",
35
+ ac: "#96640a",
36
+ acD: "rgba(150,100,10,0.08)",
37
+ acT: "#7a5208",
38
+ cdBg: "#f5f3f0",
39
+ cdTx: "#2c2520",
40
+ sbBg: "#f5f5f3",
41
+ hdBg: "rgba(250,250,249,0.85)"
42
+ },
43
+ fonts: { heading: "Instrument Serif", body: "DM Sans", code: "JetBrains Mono" }
44
+ },
45
+ editorial: {
46
+ dark: {
47
+ bg: "#080c1f",
48
+ sf: "#0e1333",
49
+ sfH: "#141940",
50
+ bd: "#1a2050",
51
+ tx: "#e8e6f0",
52
+ tx2: "#b5b1c8",
53
+ txM: "#9490ae",
54
+ ac: "#ff6b4a",
55
+ acD: "rgba(255,107,74,0.1)",
56
+ acT: "#ff8a70",
57
+ cdBg: "#0a0e27",
58
+ cdTx: "#b8b4cc",
59
+ sbBg: "#0a0e27",
60
+ hdBg: "rgba(8,12,31,0.9)"
61
+ },
62
+ light: {
63
+ bg: "#f6f4f0",
64
+ sf: "#ffffff",
65
+ sfH: "#eeece6",
66
+ bd: "#ddd9d0",
67
+ tx: "#1a1716",
68
+ tx2: "#4a443e",
69
+ txM: "#706960",
70
+ ac: "#b83d22",
71
+ acD: "rgba(184,61,34,0.07)",
72
+ acT: "#9c3019",
73
+ cdBg: "#edeae4",
74
+ cdTx: "#3a3530",
75
+ sbBg: "#f0ede8",
76
+ hdBg: "rgba(246,244,240,0.92)"
77
+ },
78
+ fonts: { heading: "Cormorant Garamond", body: "Bricolage Grotesque", code: "Fira Code" }
79
+ }
80
+ };
81
+
82
+ // src/AiChat.tsx
83
+ import { useState, useRef, useEffect, useCallback } from "react";
84
+ import { jsx, jsxs } from "react/jsx-runtime";
85
+ var ChatIcon = () => /* @__PURE__ */ jsx(
86
+ "svg",
87
+ {
88
+ width: 22,
89
+ height: 22,
90
+ viewBox: "0 0 24 24",
91
+ fill: "none",
92
+ stroke: "currentColor",
93
+ strokeWidth: "1.5",
94
+ strokeLinecap: "round",
95
+ strokeLinejoin: "round",
96
+ children: /* @__PURE__ */ jsx("path", { d: "M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" })
97
+ }
98
+ );
99
+ var CloseIcon = () => /* @__PURE__ */ jsx(
100
+ "svg",
101
+ {
102
+ width: 18,
103
+ height: 18,
104
+ viewBox: "0 0 24 24",
105
+ fill: "none",
106
+ stroke: "currentColor",
107
+ strokeWidth: "1.5",
108
+ strokeLinecap: "round",
109
+ strokeLinejoin: "round",
110
+ children: /* @__PURE__ */ jsx("path", { d: "M18 6L6 18M6 6l12 12" })
111
+ }
112
+ );
113
+ var SendIcon = () => /* @__PURE__ */ jsx(
114
+ "svg",
115
+ {
116
+ width: 16,
117
+ height: 16,
118
+ viewBox: "0 0 24 24",
119
+ fill: "none",
120
+ stroke: "currentColor",
121
+ strokeWidth: "2",
122
+ strokeLinecap: "round",
123
+ strokeLinejoin: "round",
124
+ children: /* @__PURE__ */ jsx("path", { d: "M22 2L11 13M22 2l-7 20-4-9-9-4z" })
125
+ }
126
+ );
127
+ function buildSystemPrompt(context) {
128
+ let prompt = "You are a helpful documentation assistant. Answer questions accurately based on the documentation provided below. If the answer isn't in the documentation, say so clearly. Keep answers concise and reference specific sections when possible.";
129
+ if (context) {
130
+ const trimmed = context.length > 1e5 ? context.slice(0, 1e5) + "\n\n[Documentation truncated...]" : context;
131
+ prompt += `
132
+
133
+ Documentation:
134
+ ${trimmed}`;
135
+ }
136
+ return prompt;
137
+ }
138
+ async function callOpenAI(messages, apiKey, model, context) {
139
+ const res = await fetch("https://api.openai.com/v1/chat/completions", {
140
+ method: "POST",
141
+ headers: {
142
+ "Content-Type": "application/json",
143
+ "Authorization": `Bearer ${apiKey}`
144
+ },
145
+ body: JSON.stringify({
146
+ model,
147
+ messages: [
148
+ { role: "system", content: buildSystemPrompt(context) },
149
+ ...messages.map((m) => ({ role: m.role, content: m.content }))
150
+ ]
151
+ })
152
+ });
153
+ if (!res.ok) {
154
+ const err = await res.text();
155
+ throw new Error(`OpenAI API error (${res.status}): ${err}`);
156
+ }
157
+ const data = await res.json();
158
+ return data.choices?.[0]?.message?.content || "No response.";
159
+ }
160
+ async function callAnthropic(messages, apiKey, model, context) {
161
+ const res = await fetch("https://api.anthropic.com/v1/messages", {
162
+ method: "POST",
163
+ headers: {
164
+ "Content-Type": "application/json",
165
+ "x-api-key": apiKey,
166
+ "anthropic-version": "2023-06-01",
167
+ "anthropic-dangerous-direct-browser-access": "true"
168
+ },
169
+ body: JSON.stringify({
170
+ model,
171
+ max_tokens: 1024,
172
+ system: buildSystemPrompt(context),
173
+ messages: messages.map((m) => ({ role: m.role, content: m.content }))
174
+ })
175
+ });
176
+ if (!res.ok) {
177
+ const err = await res.text();
178
+ throw new Error(`Anthropic API error (${res.status}): ${err}`);
179
+ }
180
+ const data = await res.json();
181
+ return data.content?.[0]?.text || "No response.";
182
+ }
183
+ function getDefaultModel(provider) {
184
+ if (provider === "openai") return "gpt-4o-mini";
185
+ return "claude-sonnet-4-20250514";
186
+ }
187
+ function AiChat({ provider, model, apiKey, context }) {
188
+ const [open, setOpen] = useState(false);
189
+ const [messages, setMessages] = useState([]);
190
+ const [input, setInput] = useState("");
191
+ const [loading, setLoading] = useState(false);
192
+ const [error, setError] = useState(null);
193
+ const messagesEndRef = useRef(null);
194
+ const inputRef = useRef(null);
195
+ const resolvedKey = apiKey || (typeof window !== "undefined" ? window.__TOME_AI_KEY__ : void 0);
196
+ const resolvedModel = model || getDefaultModel(provider);
197
+ useEffect(() => {
198
+ messagesEndRef.current?.scrollIntoView({ behavior: "smooth" });
199
+ }, [messages]);
200
+ useEffect(() => {
201
+ if (open) {
202
+ setTimeout(() => inputRef.current?.focus(), 100);
203
+ }
204
+ }, [open]);
205
+ const sendMessage = useCallback(async () => {
206
+ const text = input.trim();
207
+ if (!text || loading) return;
208
+ if (!resolvedKey) return;
209
+ const userMsg = { role: "user", content: text };
210
+ const updatedMessages = [...messages, userMsg];
211
+ setMessages(updatedMessages);
212
+ setInput("");
213
+ setLoading(true);
214
+ setError(null);
215
+ try {
216
+ let response;
217
+ if (provider === "openai") {
218
+ response = await callOpenAI(updatedMessages, resolvedKey, resolvedModel, context);
219
+ } else {
220
+ response = await callAnthropic(updatedMessages, resolvedKey, resolvedModel, context);
221
+ }
222
+ setMessages((prev) => [...prev, { role: "assistant", content: response }]);
223
+ } catch (err) {
224
+ setError(err instanceof Error ? err.message : "Failed to get response");
225
+ } finally {
226
+ setLoading(false);
227
+ }
228
+ }, [input, loading, messages, provider, resolvedKey, resolvedModel, context]);
229
+ const handleKeyDown = useCallback(
230
+ (e) => {
231
+ if (e.key === "Enter" && !e.shiftKey) {
232
+ e.preventDefault();
233
+ sendMessage();
234
+ }
235
+ },
236
+ [sendMessage]
237
+ );
238
+ if (!open) {
239
+ return /* @__PURE__ */ jsx(
240
+ "button",
241
+ {
242
+ "data-testid": "ai-chat-button",
243
+ onClick: () => setOpen(true),
244
+ "aria-label": "Open AI chat",
245
+ style: {
246
+ position: "fixed",
247
+ bottom: 24,
248
+ right: 24,
249
+ zIndex: 900,
250
+ width: 48,
251
+ height: 48,
252
+ borderRadius: "50%",
253
+ background: "var(--ac)",
254
+ color: "#fff",
255
+ border: "none",
256
+ cursor: "pointer",
257
+ display: "flex",
258
+ alignItems: "center",
259
+ justifyContent: "center",
260
+ boxShadow: "0 4px 16px rgba(0,0,0,0.25)",
261
+ transition: "transform 0.15s"
262
+ },
263
+ children: /* @__PURE__ */ jsx(ChatIcon, {})
264
+ }
265
+ );
266
+ }
267
+ return /* @__PURE__ */ jsxs(
268
+ "div",
269
+ {
270
+ "data-testid": "ai-chat-panel",
271
+ style: {
272
+ position: "fixed",
273
+ bottom: 24,
274
+ right: 24,
275
+ zIndex: 900,
276
+ width: 380,
277
+ maxWidth: "calc(100vw - 48px)",
278
+ height: 520,
279
+ maxHeight: "calc(100vh - 48px)",
280
+ background: "var(--sf)",
281
+ border: "1px solid var(--bd)",
282
+ borderRadius: 12,
283
+ boxShadow: "0 16px 64px rgba(0,0,0,0.3)",
284
+ display: "flex",
285
+ flexDirection: "column",
286
+ overflow: "hidden",
287
+ fontFamily: "var(--font-body)"
288
+ },
289
+ children: [
290
+ /* @__PURE__ */ jsxs(
291
+ "div",
292
+ {
293
+ style: {
294
+ display: "flex",
295
+ alignItems: "center",
296
+ justifyContent: "space-between",
297
+ padding: "12px 16px",
298
+ borderBottom: "1px solid var(--bd)",
299
+ flexShrink: 0
300
+ },
301
+ children: [
302
+ /* @__PURE__ */ jsx(
303
+ "span",
304
+ {
305
+ style: {
306
+ fontSize: 14,
307
+ fontWeight: 600,
308
+ color: "var(--tx)"
309
+ },
310
+ children: "Ask AI"
311
+ }
312
+ ),
313
+ /* @__PURE__ */ jsx(
314
+ "button",
315
+ {
316
+ "data-testid": "ai-chat-close",
317
+ onClick: () => setOpen(false),
318
+ "aria-label": "Close AI chat",
319
+ style: {
320
+ background: "none",
321
+ border: "none",
322
+ color: "var(--txM)",
323
+ cursor: "pointer",
324
+ display: "flex",
325
+ padding: 4
326
+ },
327
+ children: /* @__PURE__ */ jsx(CloseIcon, {})
328
+ }
329
+ )
330
+ ]
331
+ }
332
+ ),
333
+ /* @__PURE__ */ jsxs(
334
+ "div",
335
+ {
336
+ style: {
337
+ flex: 1,
338
+ overflow: "auto",
339
+ padding: "12px 16px"
340
+ },
341
+ children: [
342
+ !resolvedKey && /* @__PURE__ */ jsxs(
343
+ "div",
344
+ {
345
+ "data-testid": "ai-chat-no-key",
346
+ style: {
347
+ textAlign: "center",
348
+ color: "var(--txM)",
349
+ fontSize: 13,
350
+ padding: "24px 8px",
351
+ lineHeight: 1.6
352
+ },
353
+ children: [
354
+ /* @__PURE__ */ jsx("p", { style: { marginBottom: 8, fontWeight: 500, color: "var(--tx)" }, children: "AI not configured" }),
355
+ /* @__PURE__ */ jsxs("p", { style: { marginBottom: 8 }, children: [
356
+ "To enable AI chat, set the ",
357
+ /* @__PURE__ */ jsx("code", { style: {
358
+ fontFamily: "var(--font-code)",
359
+ fontSize: "0.88em",
360
+ background: "var(--cdBg)",
361
+ padding: "0.15em 0.4em",
362
+ borderRadius: 4
363
+ }, children: "apiKeyEnv" }),
364
+ " in ",
365
+ /* @__PURE__ */ jsx("code", { style: {
366
+ fontFamily: "var(--font-code)",
367
+ fontSize: "0.88em",
368
+ background: "var(--cdBg)",
369
+ padding: "0.15em 0.4em",
370
+ borderRadius: 4
371
+ }, children: "tome.config.js" }),
372
+ " and provide the environment variable at build time."
373
+ ] }),
374
+ /* @__PURE__ */ jsxs("p", { style: { fontSize: 11.5, color: "var(--txM)" }, children: [
375
+ "Example: ",
376
+ /* @__PURE__ */ jsx("code", { style: {
377
+ fontFamily: "var(--font-code)",
378
+ fontSize: "0.88em",
379
+ background: "var(--cdBg)",
380
+ padding: "0.15em 0.4em",
381
+ borderRadius: 4
382
+ }, children: "TOME_AI_KEY=sk-... tome build" })
383
+ ] })
384
+ ]
385
+ }
386
+ ),
387
+ messages.map((msg, i) => /* @__PURE__ */ jsx(
388
+ "div",
389
+ {
390
+ "data-testid": `ai-chat-message-${msg.role}`,
391
+ style: {
392
+ marginBottom: 12,
393
+ display: "flex",
394
+ justifyContent: msg.role === "user" ? "flex-end" : "flex-start"
395
+ },
396
+ children: /* @__PURE__ */ jsx(
397
+ "div",
398
+ {
399
+ style: {
400
+ maxWidth: "85%",
401
+ padding: "8px 12px",
402
+ borderRadius: 10,
403
+ fontSize: 13,
404
+ lineHeight: 1.55,
405
+ whiteSpace: "pre-wrap",
406
+ wordBreak: "break-word",
407
+ background: msg.role === "user" ? "var(--ac)" : "var(--cdBg)",
408
+ color: msg.role === "user" ? "#fff" : "var(--tx)"
409
+ },
410
+ children: msg.content
411
+ }
412
+ )
413
+ },
414
+ i
415
+ )),
416
+ loading && /* @__PURE__ */ jsx(
417
+ "div",
418
+ {
419
+ "data-testid": "ai-chat-loading",
420
+ style: {
421
+ display: "flex",
422
+ justifyContent: "flex-start",
423
+ marginBottom: 12
424
+ },
425
+ children: /* @__PURE__ */ jsx(
426
+ "div",
427
+ {
428
+ style: {
429
+ padding: "8px 12px",
430
+ borderRadius: 10,
431
+ fontSize: 13,
432
+ background: "var(--cdBg)",
433
+ color: "var(--txM)"
434
+ },
435
+ children: "Thinking..."
436
+ }
437
+ )
438
+ }
439
+ ),
440
+ error && /* @__PURE__ */ jsx(
441
+ "div",
442
+ {
443
+ "data-testid": "ai-chat-error",
444
+ style: {
445
+ padding: "8px 12px",
446
+ borderRadius: 8,
447
+ fontSize: 12,
448
+ background: "rgba(220,50,50,0.1)",
449
+ color: "#d44",
450
+ marginBottom: 12
451
+ },
452
+ children: error
453
+ }
454
+ ),
455
+ /* @__PURE__ */ jsx("div", { ref: messagesEndRef })
456
+ ]
457
+ }
458
+ ),
459
+ /* @__PURE__ */ jsxs(
460
+ "div",
461
+ {
462
+ style: {
463
+ display: "flex",
464
+ alignItems: "center",
465
+ gap: 8,
466
+ padding: "10px 12px",
467
+ borderTop: "1px solid var(--bd)",
468
+ flexShrink: 0
469
+ },
470
+ children: [
471
+ /* @__PURE__ */ jsx(
472
+ "input",
473
+ {
474
+ ref: inputRef,
475
+ "data-testid": "ai-chat-input",
476
+ value: input,
477
+ onChange: (e) => setInput(e.target.value),
478
+ onKeyDown: handleKeyDown,
479
+ placeholder: resolvedKey ? "Ask a question..." : "API key required",
480
+ disabled: !resolvedKey,
481
+ style: {
482
+ flex: 1,
483
+ background: "var(--cdBg)",
484
+ border: "1px solid var(--bd)",
485
+ borderRadius: 8,
486
+ padding: "8px 12px",
487
+ color: "var(--tx)",
488
+ fontSize: 13,
489
+ fontFamily: "var(--font-body)",
490
+ outline: "none"
491
+ }
492
+ }
493
+ ),
494
+ /* @__PURE__ */ jsx(
495
+ "button",
496
+ {
497
+ "data-testid": "ai-chat-send",
498
+ onClick: sendMessage,
499
+ disabled: !resolvedKey || !input.trim() || loading,
500
+ "aria-label": "Send message",
501
+ style: {
502
+ width: 34,
503
+ height: 34,
504
+ borderRadius: 8,
505
+ background: resolvedKey && input.trim() ? "var(--ac)" : "var(--cdBg)",
506
+ color: resolvedKey && input.trim() ? "#fff" : "var(--txM)",
507
+ border: "none",
508
+ cursor: resolvedKey && input.trim() ? "pointer" : "default",
509
+ display: "flex",
510
+ alignItems: "center",
511
+ justifyContent: "center",
512
+ flexShrink: 0
513
+ },
514
+ children: /* @__PURE__ */ jsx(SendIcon, {})
515
+ }
516
+ )
517
+ ]
518
+ }
519
+ )
520
+ ]
521
+ }
522
+ );
523
+ }
524
+
525
+ // src/Shell.tsx
526
+ import { jsx as jsx2, jsxs as jsxs2 } from "react/jsx-runtime";
527
+ function hexToRgb(hex) {
528
+ const m = /^#([0-9a-f]{6})$/i.exec(hex.trim());
529
+ if (!m) return null;
530
+ const n = parseInt(m[1], 16);
531
+ return [n >> 16 & 255, n >> 8 & 255, n & 255];
532
+ }
533
+ function buildAccentOverride(hex, isDark) {
534
+ const rgb = hexToRgb(hex);
535
+ if (!rgb) return null;
536
+ const [r, g, b] = rgb;
537
+ const acD = `rgba(${r},${g},${b},${isDark ? 0.12 : 0.08})`;
538
+ const factor = isDark ? 1.15 : 0.85;
539
+ const tr = Math.min(255, Math.round(r * factor));
540
+ const tg = Math.min(255, Math.round(g * factor));
541
+ const tb = Math.min(255, Math.round(b * factor));
542
+ const acT = `rgb(${tr},${tg},${tb})`;
543
+ return { ac: hex, acD, acT };
544
+ }
545
+ var Icon = ({ d, size = 16 }) => /* @__PURE__ */ jsx2(
546
+ "svg",
547
+ {
548
+ width: size,
549
+ height: size,
550
+ viewBox: "0 0 24 24",
551
+ fill: "none",
552
+ stroke: "currentColor",
553
+ strokeWidth: "1.5",
554
+ strokeLinecap: "round",
555
+ strokeLinejoin: "round",
556
+ children: /* @__PURE__ */ jsx2("path", { d })
557
+ }
558
+ );
559
+ var SearchIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M11 19a8 8 0 1 0 0-16 8 8 0 0 0 0 16ZM21 21l-4.3-4.3" });
560
+ var ChevRight = () => /* @__PURE__ */ jsx2(Icon, { d: "M9 18l6-6-6-6", size: 14 });
561
+ var ChevDown = () => /* @__PURE__ */ jsx2(Icon, { d: "M6 9l6 6 6-6", size: 14 });
562
+ var MenuIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M3 12h18M3 6h18M3 18h18", size: 20 });
563
+ var XIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M18 6L6 18M6 6l12 12", size: 18 });
564
+ var MoonIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M21 12.79A9 9 0 1 1 11.21 3 7 7 0 0 0 21 12.79z" });
565
+ var SunIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M12 7a5 5 0 1 0 0 10 5 5 0 0 0 0-10Z" });
566
+ var ArrowLeft = () => /* @__PURE__ */ jsx2(Icon, { d: "M19 12H5M12 19l-7-7 7-7", size: 14 });
567
+ var ArrowRight = () => /* @__PURE__ */ jsx2(Icon, { d: "M5 12h14M12 5l7 7-7 7", size: 14 });
568
+ var pagefindInstance = null;
569
+ var PAGEFIND_PATH = "/_pagefind/pagefind.js";
570
+ async function initPagefind() {
571
+ if (pagefindInstance) return pagefindInstance;
572
+ try {
573
+ pagefindInstance = await import(
574
+ /* @vite-ignore */
575
+ PAGEFIND_PATH
576
+ );
577
+ await pagefindInstance.init();
578
+ return pagefindInstance;
579
+ } catch {
580
+ return null;
581
+ }
582
+ }
583
+ var docSearchLoaded = null;
584
+ function loadDocSearch() {
585
+ if (docSearchLoaded) return docSearchLoaded;
586
+ docSearchLoaded = import("@docsearch/react").catch(() => null);
587
+ return docSearchLoaded;
588
+ }
589
+ function AlgoliaSearchModal({
590
+ appId,
591
+ apiKey,
592
+ indexName,
593
+ onNavigate,
594
+ onClose
595
+ }) {
596
+ const [DocSearchComponent, setDocSearchComponent] = useState2(null);
597
+ const [loadFailed, setLoadFailed] = useState2(false);
598
+ useEffect2(() => {
599
+ loadDocSearch().then((mod) => {
600
+ if (mod && mod.DocSearch) {
601
+ setDocSearchComponent(() => mod.DocSearch);
602
+ } else if (mod && mod.default) {
603
+ setDocSearchComponent(() => mod.default);
604
+ } else {
605
+ setLoadFailed(true);
606
+ }
607
+ });
608
+ }, []);
609
+ const extractPageId = useCallback2((url) => {
610
+ try {
611
+ const parsed = new URL(url, "http://localhost");
612
+ const pathname = parsed.pathname;
613
+ return pathname.replace(/^\//, "").replace(/\/index\.html$/, "").replace(/\.html$/, "") || "index";
614
+ } catch {
615
+ return "index";
616
+ }
617
+ }, []);
618
+ if (loadFailed) {
619
+ return /* @__PURE__ */ jsx2("div", { onClick: onClose, style: {
620
+ position: "fixed",
621
+ inset: 0,
622
+ zIndex: 1e3,
623
+ background: "rgba(0,0,0,0.55)",
624
+ backdropFilter: "blur(6px)",
625
+ display: "flex",
626
+ alignItems: "flex-start",
627
+ justifyContent: "center",
628
+ paddingTop: "12vh"
629
+ }, children: /* @__PURE__ */ jsx2("div", { onClick: (e) => e.stopPropagation(), style: {
630
+ background: "var(--sf)",
631
+ border: "1px solid var(--bd)",
632
+ borderRadius: 12,
633
+ width: "100%",
634
+ maxWidth: 520,
635
+ boxShadow: "0 24px 80px rgba(0,0,0,0.4)",
636
+ padding: "32px 18px",
637
+ textAlign: "center",
638
+ color: "var(--txM)",
639
+ fontSize: 14
640
+ }, children: "Algolia DocSearch is not available. Install @docsearch/react to enable it." }) });
641
+ }
642
+ if (!DocSearchComponent) {
643
+ return /* @__PURE__ */ jsx2("div", { onClick: onClose, style: {
644
+ position: "fixed",
645
+ inset: 0,
646
+ zIndex: 1e3,
647
+ background: "rgba(0,0,0,0.55)",
648
+ backdropFilter: "blur(6px)",
649
+ display: "flex",
650
+ alignItems: "flex-start",
651
+ justifyContent: "center",
652
+ paddingTop: "12vh"
653
+ }, children: /* @__PURE__ */ jsx2("div", { style: {
654
+ background: "var(--sf)",
655
+ border: "1px solid var(--bd)",
656
+ borderRadius: 12,
657
+ width: "100%",
658
+ maxWidth: 520,
659
+ boxShadow: "0 24px 80px rgba(0,0,0,0.4)",
660
+ padding: "32px 18px",
661
+ textAlign: "center",
662
+ color: "var(--txM)",
663
+ fontSize: 14
664
+ }, children: "Loading search..." }) });
665
+ }
666
+ return /* @__PURE__ */ jsx2("div", { "data-testid": "algolia-search-modal", children: /* @__PURE__ */ jsx2(
667
+ DocSearchComponent,
668
+ {
669
+ appId,
670
+ apiKey,
671
+ indexName,
672
+ navigator: {
673
+ navigate({ itemUrl }) {
674
+ const pageId = extractPageId(itemUrl);
675
+ onNavigate(pageId);
676
+ }
677
+ },
678
+ hitComponent: ({ hit, children }) => /* @__PURE__ */ jsx2("a", { href: hit.url, onClick: (e) => {
679
+ e.preventDefault();
680
+ const pageId = extractPageId(hit.url);
681
+ onNavigate(pageId);
682
+ }, children })
683
+ }
684
+ ) });
685
+ }
686
+ var VersionIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M12 8v4l3 3m6-3a9 9 0 1 1-18 0 9 9 0 0 1 18 0Z", size: 14 });
687
+ var GlobeIcon = () => /* @__PURE__ */ jsx2(Icon, { d: "M12 21a9 9 0 1 0 0-18 9 9 0 0 0 0 18ZM3.6 9h16.8M3.6 15h16.8M12 3a15 15 0 0 1 4 9 15 15 0 0 1-4 9 15 15 0 0 1-4-9 15 15 0 0 1 4-9Z", size: 14 });
688
+ function Shell({
689
+ config: config2,
690
+ navigation: navigation2,
691
+ currentPageId,
692
+ pageHtml,
693
+ pageComponent,
694
+ mdxComponents,
695
+ pageTitle,
696
+ pageDescription,
697
+ headings,
698
+ onNavigate,
699
+ allPages,
700
+ versioning,
701
+ currentVersion,
702
+ i18n,
703
+ currentLocale,
704
+ docContext: docContext2
705
+ }) {
706
+ const themeMode = config2.theme?.mode || "auto";
707
+ const [isDark, setDark] = useState2(() => {
708
+ if (themeMode === "dark") return true;
709
+ if (themeMode === "light") return false;
710
+ return window.matchMedia?.("(prefers-color-scheme: dark)").matches ?? true;
711
+ });
712
+ const [sbOpen, setSb] = useState2(true);
713
+ const [searchOpen, setSearch] = useState2(false);
714
+ const [versionDropdownOpen, setVersionDropdown] = useState2(false);
715
+ const [localeDropdownOpen, setLocaleDropdown] = useState2(false);
716
+ const isOldVersion = versioning && currentVersion && currentVersion !== versioning.current;
717
+ const [expanded, setExpanded] = useState2(navigation2.map((n) => n.section));
718
+ const contentRef = useRef2(null);
719
+ const [wide, setWide] = useState2(true);
720
+ const preset = config2.theme?.preset || "amber";
721
+ const baseTokens = THEME_PRESETS[preset]?.[isDark ? "dark" : "light"] || THEME_PRESETS.amber.dark;
722
+ const accentOverride = config2.theme?.accent ? buildAccentOverride(config2.theme.accent, isDark) : null;
723
+ const t = accentOverride ? { ...baseTokens, ...accentOverride } : baseTokens;
724
+ const presetFonts = THEME_PRESETS[preset]?.fonts || THEME_PRESETS.amber.fonts;
725
+ const fonts = {
726
+ heading: config2.theme?.fonts?.heading || presetFonts.heading,
727
+ body: config2.theme?.fonts?.body || presetFonts.body,
728
+ code: config2.theme?.fonts?.code || presetFonts.code
729
+ };
730
+ useEffect2(() => {
731
+ if (themeMode !== "auto") return;
732
+ const mq = window.matchMedia("(prefers-color-scheme: dark)");
733
+ const handler = (e) => setDark(e.matches);
734
+ mq.addEventListener("change", handler);
735
+ return () => mq.removeEventListener("change", handler);
736
+ }, [themeMode]);
737
+ useEffect2(() => {
738
+ const c = () => setWide(window.innerWidth > 1100);
739
+ c();
740
+ window.addEventListener("resize", c);
741
+ return () => window.removeEventListener("resize", c);
742
+ }, []);
743
+ useEffect2(() => {
744
+ contentRef.current?.scrollTo(0, 0);
745
+ }, [currentPageId]);
746
+ useEffect2(() => {
747
+ const h = (e) => {
748
+ if ((e.metaKey || e.ctrlKey) && e.key === "k") {
749
+ e.preventDefault();
750
+ setSearch(true);
751
+ }
752
+ if (e.key === "Escape") setSearch(false);
753
+ };
754
+ window.addEventListener("keydown", h);
755
+ return () => window.removeEventListener("keydown", h);
756
+ }, []);
757
+ const allNavPages = navigation2.flatMap((g) => g.pages);
758
+ const idx = allNavPages.findIndex((p) => p.id === currentPageId);
759
+ const prev = idx > 0 ? allNavPages[idx - 1] : null;
760
+ const next = idx < allNavPages.length - 1 ? allNavPages[idx + 1] : null;
761
+ const togSec = (s) => setExpanded((p) => p.includes(s) ? p.filter((x) => x !== s) : [...p, s]);
762
+ const cssVars = {
763
+ "--bg": t.bg,
764
+ "--sf": t.sf,
765
+ "--sfH": t.sfH,
766
+ "--bd": t.bd,
767
+ "--tx": t.tx,
768
+ "--tx2": t.tx2,
769
+ "--txM": t.txM,
770
+ "--ac": t.ac,
771
+ "--acD": t.acD,
772
+ "--acT": t.acT,
773
+ "--cdBg": t.cdBg,
774
+ "--cdTx": t.cdTx,
775
+ "--sbBg": t.sbBg,
776
+ "--hdBg": t.hdBg,
777
+ "--font-heading": `"${fonts.heading}", serif`,
778
+ "--font-body": `"${fonts.body}", sans-serif`,
779
+ "--font-code": `"${fonts.code}", monospace`
780
+ };
781
+ const PageComponent = pageComponent;
782
+ return /* @__PURE__ */ jsxs2("div", { className: "tome-grain", style: { ...cssVars, color: "var(--tx)", background: "var(--bg)", fontFamily: "var(--font-body)", minHeight: "100vh" }, children: [
783
+ searchOpen && config2.search?.provider === "algolia" && config2.search.appId && config2.search.apiKey && config2.search.indexName ? /* @__PURE__ */ jsx2(
784
+ AlgoliaSearchModal,
785
+ {
786
+ appId: config2.search.appId,
787
+ apiKey: config2.search.apiKey,
788
+ indexName: config2.search.indexName,
789
+ onNavigate: (id) => {
790
+ onNavigate(id);
791
+ setSearch(false);
792
+ },
793
+ onClose: () => setSearch(false)
794
+ }
795
+ ) : searchOpen ? /* @__PURE__ */ jsx2(
796
+ SearchModal,
797
+ {
798
+ allPages,
799
+ onNavigate: (id) => {
800
+ onNavigate(id);
801
+ setSearch(false);
802
+ },
803
+ onClose: () => setSearch(false)
804
+ }
805
+ ) : null,
806
+ /* @__PURE__ */ jsxs2("div", { style: { display: "flex", height: "100vh" }, children: [
807
+ /* @__PURE__ */ jsxs2("aside", { style: {
808
+ width: sbOpen ? 270 : 0,
809
+ minWidth: sbOpen ? 270 : 0,
810
+ background: "var(--sbBg)",
811
+ borderRight: "1px solid var(--bd)",
812
+ display: "flex",
813
+ flexDirection: "column",
814
+ transition: "width .2s, min-width .2s",
815
+ overflow: "hidden"
816
+ }, children: [
817
+ /* @__PURE__ */ jsxs2("div", { style: { padding: "18px 20px", display: "flex", alignItems: "baseline", gap: 6, borderBottom: "1px solid var(--bd)" }, children: [
818
+ /* @__PURE__ */ jsx2("span", { style: { fontFamily: "var(--font-heading)", fontSize: 22, fontWeight: 700, fontStyle: "italic" }, children: config2.name }),
819
+ /* @__PURE__ */ jsx2("span", { style: { width: 5, height: 5, borderRadius: "50%", background: "var(--ac)", display: "inline-block" } })
820
+ ] }),
821
+ /* @__PURE__ */ jsx2("div", { style: { padding: "12px 14px" }, children: /* @__PURE__ */ jsxs2("button", { onClick: () => setSearch(true), style: {
822
+ display: "flex",
823
+ alignItems: "center",
824
+ gap: 8,
825
+ width: "100%",
826
+ background: "var(--cdBg)",
827
+ border: "1px solid var(--bd)",
828
+ borderRadius: 2,
829
+ padding: "8px 12px",
830
+ cursor: "pointer",
831
+ color: "var(--txM)",
832
+ fontSize: 12.5,
833
+ fontFamily: "var(--font-body)"
834
+ }, children: [
835
+ /* @__PURE__ */ jsx2(SearchIcon, {}),
836
+ /* @__PURE__ */ jsx2("span", { style: { flex: 1, textAlign: "left" }, children: "Search..." }),
837
+ /* @__PURE__ */ jsx2("kbd", { style: { fontFamily: "var(--font-code)", fontSize: 9, background: "var(--sf)", border: "1px solid var(--bd)", borderRadius: 2, padding: "2px 6px" }, children: "\u2318K" })
838
+ ] }) }),
839
+ /* @__PURE__ */ jsx2("nav", { style: { flex: 1, overflow: "auto", padding: "4px 10px 20px" }, children: navigation2.map((sec) => /* @__PURE__ */ jsxs2("div", { style: { marginBottom: 8 }, children: [
840
+ /* @__PURE__ */ jsxs2("button", { onClick: () => togSec(sec.section), style: {
841
+ display: "flex",
842
+ alignItems: "center",
843
+ gap: 6,
844
+ width: "100%",
845
+ background: "none",
846
+ border: "none",
847
+ padding: "8px 10px",
848
+ cursor: "pointer",
849
+ borderRadius: 2,
850
+ color: "var(--txM)",
851
+ fontSize: 10,
852
+ fontWeight: 600,
853
+ textTransform: "uppercase",
854
+ letterSpacing: ".1em",
855
+ fontFamily: "var(--font-code)"
856
+ }, children: [
857
+ expanded.includes(sec.section) ? /* @__PURE__ */ jsx2(ChevDown, {}) : /* @__PURE__ */ jsx2(ChevRight, {}),
858
+ sec.section
859
+ ] }),
860
+ expanded.includes(sec.section) && /* @__PURE__ */ jsx2("div", { style: { marginLeft: 8, borderLeft: "1px solid var(--bd)", paddingLeft: 0 }, children: sec.pages.map((p) => {
861
+ const active = currentPageId === p.id;
862
+ return /* @__PURE__ */ jsx2("button", { onClick: () => onNavigate(p.id), style: {
863
+ display: "flex",
864
+ alignItems: "center",
865
+ gap: 10,
866
+ width: "100%",
867
+ textAlign: "left",
868
+ background: "none",
869
+ border: "none",
870
+ borderRadius: 0,
871
+ borderLeft: active ? "2px solid var(--ac)" : "2px solid transparent",
872
+ padding: "7px 14px",
873
+ cursor: "pointer",
874
+ color: active ? "var(--ac)" : "var(--tx2)",
875
+ fontSize: 13,
876
+ fontWeight: active ? 500 : 400,
877
+ fontFamily: "var(--font-body)",
878
+ transition: "all .12s"
879
+ }, children: p.title }, p.id);
880
+ }) })
881
+ ] }, sec.section)) }),
882
+ /* @__PURE__ */ jsxs2("div", { style: { padding: "12px 16px", borderTop: "1px solid var(--bd)", display: "flex", alignItems: "center", justifyContent: "space-between" }, children: [
883
+ themeMode === "auto" ? /* @__PURE__ */ jsx2("button", { onClick: () => setDark((d) => !d), style: { background: "none", border: "none", color: "var(--txM)", cursor: "pointer", display: "flex" }, children: isDark ? /* @__PURE__ */ jsx2(SunIcon, {}) : /* @__PURE__ */ jsx2(MoonIcon, {}) }) : /* @__PURE__ */ jsx2("div", {}),
884
+ /* @__PURE__ */ jsx2("span", { style: { fontFamily: "var(--font-code)", fontSize: 10, color: "var(--txM)" }, children: "v0.1.0" })
885
+ ] })
886
+ ] }),
887
+ /* @__PURE__ */ jsxs2("div", { style: { flex: 1, display: "flex", flexDirection: "column", overflow: "hidden" }, children: [
888
+ /* @__PURE__ */ jsxs2("header", { style: {
889
+ display: "flex",
890
+ alignItems: "center",
891
+ gap: 12,
892
+ padding: "10px 24px",
893
+ borderBottom: "1px solid var(--bd)",
894
+ background: "var(--hdBg)",
895
+ backdropFilter: "blur(12px)"
896
+ }, children: [
897
+ /* @__PURE__ */ jsx2("button", { onClick: () => setSb(!sbOpen), style: { background: "none", border: "none", color: "var(--txM)", cursor: "pointer", display: "flex" }, children: sbOpen ? /* @__PURE__ */ jsx2(XIcon, {}) : /* @__PURE__ */ jsx2(MenuIcon, {}) }),
898
+ /* @__PURE__ */ jsx2("div", { style: { display: "flex", alignItems: "center", gap: 8, fontFamily: "var(--font-code)", fontSize: 11, color: "var(--txM)", letterSpacing: ".03em", flex: 1 }, children: navigation2.map((s) => {
899
+ const f = s.pages.find((p) => p.id === currentPageId);
900
+ if (!f) return null;
901
+ return /* @__PURE__ */ jsxs2("span", { style: { display: "flex", alignItems: "center", gap: 8 }, children: [
902
+ /* @__PURE__ */ jsx2("span", { children: s.section }),
903
+ /* @__PURE__ */ jsx2(ChevRight, {}),
904
+ /* @__PURE__ */ jsx2("span", { style: { color: "var(--ac)" }, children: f.title })
905
+ ] }, s.section);
906
+ }) }),
907
+ versioning && /* @__PURE__ */ jsxs2("div", { style: { position: "relative" }, children: [
908
+ /* @__PURE__ */ jsxs2(
909
+ "button",
910
+ {
911
+ "data-testid": "version-switcher",
912
+ onClick: () => setVersionDropdown((o) => !o),
913
+ style: {
914
+ display: "flex",
915
+ alignItems: "center",
916
+ gap: 6,
917
+ background: "var(--sf)",
918
+ border: "1px solid var(--bd)",
919
+ borderRadius: 2,
920
+ padding: "5px 10px",
921
+ cursor: "pointer",
922
+ color: "var(--tx2)",
923
+ fontSize: 12,
924
+ fontFamily: "var(--font-code)"
925
+ },
926
+ children: [
927
+ /* @__PURE__ */ jsx2(VersionIcon, {}),
928
+ currentVersion || versioning.current,
929
+ /* @__PURE__ */ jsx2(ChevDown, {})
930
+ ]
931
+ }
932
+ ),
933
+ versionDropdownOpen && /* @__PURE__ */ jsx2(
934
+ "div",
935
+ {
936
+ "data-testid": "version-dropdown",
937
+ style: {
938
+ position: "absolute",
939
+ top: "100%",
940
+ right: 0,
941
+ marginTop: 4,
942
+ background: "var(--sf)",
943
+ border: "1px solid var(--bd)",
944
+ borderRadius: 2,
945
+ boxShadow: "0 8px 32px rgba(0,0,0,0.2)",
946
+ overflow: "hidden",
947
+ zIndex: 100,
948
+ minWidth: 120
949
+ },
950
+ children: versioning.versions.map((v) => /* @__PURE__ */ jsxs2(
951
+ "button",
952
+ {
953
+ onClick: () => {
954
+ setVersionDropdown(false);
955
+ const targetUrl = v === versioning.current ? "/" : `/${v}`;
956
+ window.location.href = targetUrl;
957
+ },
958
+ style: {
959
+ display: "block",
960
+ width: "100%",
961
+ textAlign: "left",
962
+ background: v === (currentVersion || versioning.current) ? "var(--acD)" : "none",
963
+ border: "none",
964
+ padding: "8px 14px",
965
+ cursor: "pointer",
966
+ color: v === (currentVersion || versioning.current) ? "var(--ac)" : "var(--tx2)",
967
+ fontSize: 12,
968
+ fontFamily: "var(--font-code)",
969
+ fontWeight: v === versioning.current ? 600 : 400
970
+ },
971
+ children: [
972
+ v,
973
+ v === versioning.current ? " (latest)" : ""
974
+ ]
975
+ },
976
+ v
977
+ ))
978
+ }
979
+ )
980
+ ] }),
981
+ i18n && i18n.locales.length > 1 && /* @__PURE__ */ jsxs2("div", { style: { position: "relative" }, children: [
982
+ /* @__PURE__ */ jsxs2(
983
+ "button",
984
+ {
985
+ "data-testid": "language-switcher",
986
+ onClick: () => setLocaleDropdown((o) => !o),
987
+ style: {
988
+ display: "flex",
989
+ alignItems: "center",
990
+ gap: 6,
991
+ background: "var(--sf)",
992
+ border: "1px solid var(--bd)",
993
+ borderRadius: 2,
994
+ padding: "5px 10px",
995
+ cursor: "pointer",
996
+ color: "var(--tx2)",
997
+ fontSize: 12,
998
+ fontFamily: "var(--font-body)"
999
+ },
1000
+ children: [
1001
+ /* @__PURE__ */ jsx2(GlobeIcon, {}),
1002
+ i18n.localeNames?.[currentLocale || i18n.defaultLocale] || currentLocale || i18n.defaultLocale,
1003
+ /* @__PURE__ */ jsx2(ChevDown, {})
1004
+ ]
1005
+ }
1006
+ ),
1007
+ localeDropdownOpen && /* @__PURE__ */ jsx2(
1008
+ "div",
1009
+ {
1010
+ "data-testid": "language-dropdown",
1011
+ style: {
1012
+ position: "absolute",
1013
+ top: "100%",
1014
+ right: 0,
1015
+ marginTop: 4,
1016
+ background: "var(--sf)",
1017
+ border: "1px solid var(--bd)",
1018
+ borderRadius: 2,
1019
+ boxShadow: "0 8px 32px rgba(0,0,0,0.2)",
1020
+ overflow: "hidden",
1021
+ zIndex: 100,
1022
+ minWidth: 120
1023
+ },
1024
+ children: i18n.locales.map((locale) => {
1025
+ const isActive = locale === (currentLocale || i18n.defaultLocale);
1026
+ const displayName = i18n.localeNames?.[locale] || locale;
1027
+ const activeLocale = currentLocale || i18n.defaultLocale;
1028
+ let basePageId = currentPageId;
1029
+ if (activeLocale !== i18n.defaultLocale && currentPageId.startsWith(`${activeLocale}/`)) {
1030
+ basePageId = currentPageId.slice(activeLocale.length + 1);
1031
+ }
1032
+ const targetId = locale === i18n.defaultLocale ? basePageId : `${locale}/${basePageId}`;
1033
+ return /* @__PURE__ */ jsx2(
1034
+ "button",
1035
+ {
1036
+ onClick: () => {
1037
+ setLocaleDropdown(false);
1038
+ onNavigate(targetId);
1039
+ },
1040
+ style: {
1041
+ display: "block",
1042
+ width: "100%",
1043
+ textAlign: "left",
1044
+ background: isActive ? "var(--acD)" : "none",
1045
+ border: "none",
1046
+ padding: "8px 14px",
1047
+ cursor: "pointer",
1048
+ color: isActive ? "var(--ac)" : "var(--tx2)",
1049
+ fontSize: 12,
1050
+ fontFamily: "var(--font-body)",
1051
+ fontWeight: isActive ? 600 : 400
1052
+ },
1053
+ children: displayName
1054
+ },
1055
+ locale
1056
+ );
1057
+ })
1058
+ }
1059
+ )
1060
+ ] })
1061
+ ] }),
1062
+ isOldVersion && /* @__PURE__ */ jsxs2(
1063
+ "div",
1064
+ {
1065
+ "data-testid": "old-version-banner",
1066
+ style: {
1067
+ display: "flex",
1068
+ alignItems: "center",
1069
+ justifyContent: "center",
1070
+ gap: 8,
1071
+ background: "var(--acD)",
1072
+ borderBottom: "1px solid var(--bd)",
1073
+ padding: "8px 24px",
1074
+ fontSize: 13,
1075
+ color: "var(--tx2)"
1076
+ },
1077
+ children: [
1078
+ /* @__PURE__ */ jsxs2("span", { children: [
1079
+ "You're viewing docs for ",
1080
+ currentVersion,
1081
+ "."
1082
+ ] }),
1083
+ /* @__PURE__ */ jsx2(
1084
+ "button",
1085
+ {
1086
+ onClick: () => {
1087
+ window.location.href = "/";
1088
+ },
1089
+ style: {
1090
+ background: "none",
1091
+ border: "none",
1092
+ color: "var(--ac)",
1093
+ cursor: "pointer",
1094
+ fontWeight: 600,
1095
+ fontSize: 13,
1096
+ fontFamily: "var(--font-body)",
1097
+ textDecoration: "underline"
1098
+ },
1099
+ children: "Switch to latest."
1100
+ }
1101
+ )
1102
+ ]
1103
+ }
1104
+ ),
1105
+ /* @__PURE__ */ jsxs2("div", { ref: contentRef, style: { flex: 1, overflow: "auto", display: "flex" }, children: [
1106
+ /* @__PURE__ */ jsxs2("main", { style: { flex: 1, maxWidth: 760, padding: "40px 48px 80px", margin: "0 auto" }, children: [
1107
+ /* @__PURE__ */ jsx2("h1", { style: { fontFamily: "var(--font-heading)", fontSize: 38, fontWeight: 400, fontStyle: "italic", lineHeight: 1.15, marginBottom: 8 }, children: pageTitle }),
1108
+ pageDescription && /* @__PURE__ */ jsx2("p", { style: { fontSize: 16, color: "var(--tx2)", lineHeight: 1.6, marginBottom: 32 }, children: pageDescription }),
1109
+ /* @__PURE__ */ jsx2("div", { style: { borderTop: "1px solid var(--bd)", paddingTop: 28 }, children: PageComponent ? /* @__PURE__ */ jsx2("div", { className: "tome-content", children: /* @__PURE__ */ jsx2(PageComponent, { components: mdxComponents || {} }) }) : /* @__PURE__ */ jsx2(
1110
+ "div",
1111
+ {
1112
+ className: "tome-content",
1113
+ dangerouslySetInnerHTML: { __html: pageHtml || "" }
1114
+ }
1115
+ ) }),
1116
+ /* @__PURE__ */ jsxs2("div", { style: { display: "flex", justifyContent: "space-between", marginTop: 48, paddingTop: 24, borderTop: "1px solid var(--bd)", gap: 16 }, children: [
1117
+ prev ? /* @__PURE__ */ jsxs2("button", { onClick: () => onNavigate(prev.id), style: {
1118
+ display: "flex",
1119
+ alignItems: "center",
1120
+ gap: 8,
1121
+ background: "none",
1122
+ border: "1px solid var(--bd)",
1123
+ borderRadius: 2,
1124
+ padding: "10px 16px",
1125
+ cursor: "pointer",
1126
+ color: "var(--tx2)",
1127
+ fontSize: 13,
1128
+ fontFamily: "var(--font-body)",
1129
+ transition: "border-color .15s, color .15s"
1130
+ }, children: [
1131
+ /* @__PURE__ */ jsx2(ArrowLeft, {}),
1132
+ " ",
1133
+ prev.title
1134
+ ] }) : /* @__PURE__ */ jsx2("div", {}),
1135
+ next ? /* @__PURE__ */ jsxs2("button", { onClick: () => onNavigate(next.id), style: {
1136
+ display: "flex",
1137
+ alignItems: "center",
1138
+ gap: 8,
1139
+ background: "none",
1140
+ border: "1px solid var(--bd)",
1141
+ borderRadius: 2,
1142
+ padding: "10px 16px",
1143
+ cursor: "pointer",
1144
+ color: "var(--tx2)",
1145
+ fontSize: 13,
1146
+ fontFamily: "var(--font-body)",
1147
+ transition: "border-color .15s, color .15s"
1148
+ }, children: [
1149
+ next.title,
1150
+ " ",
1151
+ /* @__PURE__ */ jsx2(ArrowRight, {})
1152
+ ] }) : /* @__PURE__ */ jsx2("div", {})
1153
+ ] })
1154
+ ] }),
1155
+ headings.length > 0 && wide && /* @__PURE__ */ jsxs2("aside", { style: { width: 200, padding: "40px 16px 40px 0", position: "sticky", top: 0, alignSelf: "flex-start", flexShrink: 0 }, children: [
1156
+ /* @__PURE__ */ jsx2("div", { style: { fontSize: 10, fontWeight: 600, textTransform: "uppercase", letterSpacing: ".1em", color: "var(--txM)", marginBottom: 12, fontFamily: "var(--font-code)" }, children: "On this page" }),
1157
+ /* @__PURE__ */ jsx2("div", { style: { borderLeft: "1px solid var(--bd)", paddingLeft: 0 }, children: headings.map((h, i) => /* @__PURE__ */ jsx2("a", { href: `#${h.id}`, style: {
1158
+ display: "block",
1159
+ fontSize: 12,
1160
+ color: "var(--txM)",
1161
+ textDecoration: "none",
1162
+ padding: "4px 12px",
1163
+ paddingLeft: 12 + (h.depth - 2) * 12,
1164
+ lineHeight: 1.4,
1165
+ transition: "color .12s"
1166
+ }, children: h.text }, i)) })
1167
+ ] })
1168
+ ] })
1169
+ ] })
1170
+ ] }),
1171
+ config2.ai?.enabled && /* @__PURE__ */ jsx2(
1172
+ AiChat,
1173
+ {
1174
+ provider: config2.ai.provider || "anthropic",
1175
+ model: config2.ai.model,
1176
+ apiKey: typeof __TOME_AI_API_KEY__ !== "undefined" && __TOME_AI_API_KEY__ ? __TOME_AI_API_KEY__ : void 0,
1177
+ context: docContext2?.map((d) => `## ${d.title}
1178
+ ${d.content}`).join("\n\n") ?? allPages.map((p) => `- ${p.title}${p.description ? ": " + p.description : ""}`).join("\n")
1179
+ }
1180
+ )
1181
+ ] });
1182
+ }
1183
+ function SearchModal({ allPages, onNavigate, onClose }) {
1184
+ const [q, setQ] = useState2("");
1185
+ const [results, setResults] = useState2([]);
1186
+ const [selected, setSelected] = useState2(0);
1187
+ const [pagefindReady, setPagefindReady] = useState2(null);
1188
+ const inputRef = useRef2(null);
1189
+ const debounceRef = useRef2();
1190
+ useEffect2(() => {
1191
+ initPagefind().then((pf) => setPagefindReady(!!pf));
1192
+ setTimeout(() => inputRef.current?.focus(), 50);
1193
+ }, []);
1194
+ const fallbackSearch = useCallback2((query) => {
1195
+ if (!query.trim()) return [];
1196
+ const ql = query.toLowerCase();
1197
+ return allPages.filter((p) => p.title.toLowerCase().includes(ql) || (p.description || "").toLowerCase().includes(ql)).slice(0, 8).map((p) => ({ id: p.id, title: p.title, excerpt: p.description }));
1198
+ }, [allPages]);
1199
+ const doSearch = useCallback2(async (query) => {
1200
+ if (!query.trim()) {
1201
+ setResults([]);
1202
+ setSelected(0);
1203
+ return;
1204
+ }
1205
+ const pf = pagefindInstance;
1206
+ if (pf) {
1207
+ try {
1208
+ const search = await pf.search(query);
1209
+ const items = [];
1210
+ for (const result of search.results.slice(0, 8)) {
1211
+ const data = await result.data();
1212
+ const url = data.url || "";
1213
+ const id = url.replace(/^\//, "").replace(/\/index\.html$/, "").replace(/\.html$/, "") || "index";
1214
+ items.push({
1215
+ id,
1216
+ title: data.meta?.title || id,
1217
+ excerpt: data.excerpt || void 0
1218
+ });
1219
+ }
1220
+ setResults(items);
1221
+ setSelected(0);
1222
+ return;
1223
+ } catch {
1224
+ }
1225
+ }
1226
+ setResults(fallbackSearch(query));
1227
+ setSelected(0);
1228
+ }, [fallbackSearch]);
1229
+ useEffect2(() => {
1230
+ if (debounceRef.current) clearTimeout(debounceRef.current);
1231
+ debounceRef.current = setTimeout(() => doSearch(q), 120);
1232
+ return () => {
1233
+ if (debounceRef.current) clearTimeout(debounceRef.current);
1234
+ };
1235
+ }, [q, doSearch]);
1236
+ const handleKeyDown = useCallback2((e) => {
1237
+ if (e.key === "ArrowDown") {
1238
+ e.preventDefault();
1239
+ setSelected((i) => Math.min(i + 1, results.length - 1));
1240
+ } else if (e.key === "ArrowUp") {
1241
+ e.preventDefault();
1242
+ setSelected((i) => Math.max(i - 1, 0));
1243
+ } else if (e.key === "Enter" && results.length > 0) {
1244
+ e.preventDefault();
1245
+ onNavigate(results[selected].id);
1246
+ }
1247
+ }, [results, selected, onNavigate]);
1248
+ return /* @__PURE__ */ jsx2("div", { onClick: onClose, style: {
1249
+ position: "fixed",
1250
+ inset: 0,
1251
+ zIndex: 1e3,
1252
+ background: "rgba(0,0,0,0.55)",
1253
+ backdropFilter: "blur(6px)",
1254
+ display: "flex",
1255
+ alignItems: "flex-start",
1256
+ justifyContent: "center",
1257
+ paddingTop: "12vh"
1258
+ }, children: /* @__PURE__ */ jsxs2("div", { onClick: (e) => e.stopPropagation(), style: {
1259
+ background: "var(--sf)",
1260
+ border: "1px solid var(--bd)",
1261
+ borderRadius: 2,
1262
+ width: "100%",
1263
+ maxWidth: 520,
1264
+ boxShadow: "0 24px 80px rgba(0,0,0,0.4)",
1265
+ overflow: "hidden"
1266
+ }, children: [
1267
+ /* @__PURE__ */ jsxs2("div", { style: { display: "flex", alignItems: "center", gap: 10, padding: "14px 18px", borderBottom: "1px solid var(--bd)" }, children: [
1268
+ /* @__PURE__ */ jsx2(SearchIcon, {}),
1269
+ /* @__PURE__ */ jsx2(
1270
+ "input",
1271
+ {
1272
+ ref: inputRef,
1273
+ value: q,
1274
+ onChange: (e) => setQ(e.target.value),
1275
+ onKeyDown: handleKeyDown,
1276
+ placeholder: "Search documentation...",
1277
+ style: { flex: 1, background: "none", border: "none", outline: "none", color: "var(--tx)", fontSize: 15, fontFamily: "var(--font-body)" }
1278
+ }
1279
+ ),
1280
+ /* @__PURE__ */ jsx2("kbd", { style: { fontFamily: "var(--font-code)", fontSize: 10, color: "var(--txM)", background: "var(--cdBg)", padding: "2px 6px", borderRadius: 2, border: "1px solid var(--bd)" }, children: "ESC" })
1281
+ ] }),
1282
+ results.length > 0 && /* @__PURE__ */ jsx2("div", { style: { padding: 6, maxHeight: 360, overflow: "auto" }, children: results.map((r, i) => /* @__PURE__ */ jsxs2(
1283
+ "button",
1284
+ {
1285
+ onClick: () => onNavigate(r.id),
1286
+ style: {
1287
+ display: "block",
1288
+ width: "100%",
1289
+ textAlign: "left",
1290
+ background: i === selected ? "var(--acD)" : "none",
1291
+ border: "none",
1292
+ borderRadius: 2,
1293
+ padding: "10px 14px",
1294
+ cursor: "pointer",
1295
+ color: "var(--tx)",
1296
+ fontFamily: "var(--font-body)"
1297
+ },
1298
+ onMouseEnter: () => setSelected(i),
1299
+ children: [
1300
+ /* @__PURE__ */ jsx2("div", { style: { fontWeight: 500, fontSize: 14, marginBottom: 2 }, children: r.title }),
1301
+ r.excerpt && /* @__PURE__ */ jsx2("div", { style: {
1302
+ fontSize: 12,
1303
+ color: "var(--txM)",
1304
+ lineHeight: 1.3
1305
+ }, dangerouslySetInnerHTML: { __html: r.excerpt } })
1306
+ ]
1307
+ },
1308
+ r.id + i
1309
+ )) }),
1310
+ q && !results.length && /* @__PURE__ */ jsx2("div", { style: { padding: "32px 18px", textAlign: "center", color: "var(--txM)", fontSize: 14 }, children: "No results found" }),
1311
+ pagefindReady === false && q && results.length > 0 && /* @__PURE__ */ jsx2("div", { style: { padding: "6px 18px 10px", fontSize: 11, color: "var(--txM)", textAlign: "center" }, children: "Showing title matches. Build your site for full-text search." })
1312
+ ] }) });
1313
+ }
1314
+
1315
+ // src/entry.tsx
1316
+ import config from "virtual:tome/config";
1317
+ import { routes, navigation } from "virtual:tome/routes";
1318
+ import loadPageModule from "virtual:tome/page-loader";
1319
+ import docContext from "virtual:tome/doc-context";
1320
+ import {
1321
+ Callout,
1322
+ Tabs,
1323
+ Card,
1324
+ CardGroup,
1325
+ Steps,
1326
+ Accordion
1327
+ } from "@tome/components";
1328
+ import { Fragment, jsx as jsx3, jsxs as jsxs3 } from "react/jsx-runtime";
1329
+ var MDX_COMPONENTS = {
1330
+ Callout,
1331
+ Tabs,
1332
+ Card,
1333
+ CardGroup,
1334
+ Steps,
1335
+ Accordion
1336
+ };
1337
+ var contentStyles = `
1338
+ @import url('https://fonts.googleapis.com/css2?family=Bricolage+Grotesque:wght@300;400;500;600;700&family=Cormorant+Garamond:ital,wght@0,300;0,400;0,600;0,700;1,300;1,400;1,700&family=Fira+Code:wght@400;500;600&display=swap');
1339
+
1340
+ .tome-content h2 { font-family: var(--font-body); font-size: 1.35em; font-weight: 600; margin-top: 2em; margin-bottom: 0.5em; display: flex; align-items: center; gap: 10px; letter-spacing: 0.01em; }
1341
+ .tome-content h2::before { content: "#"; font-family: var(--font-heading); font-size: 1.2em; font-weight: 300; font-style: italic; color: var(--ac); opacity: 0.5; }
1342
+ .tome-content h3 { font-family: var(--font-body); font-size: 1.15em; font-weight: 600; margin-top: 1.5em; margin-bottom: 0.5em; }
1343
+ .tome-content h4 { font-family: var(--font-body); font-size: 1.05em; font-weight: 600; margin-top: 1.2em; margin-bottom: 0.5em; }
1344
+ .tome-content p { color: var(--tx2); line-height: 1.8; margin-bottom: 1em; font-size: 14.5px; }
1345
+ .tome-content a { color: var(--ac); text-decoration: none; }
1346
+ .tome-content a:hover { text-decoration: underline; }
1347
+ .tome-content .heading-anchor { display: none; }
1348
+ .tome-content ul, .tome-content ol { color: var(--tx2); padding-left: 1.5em; margin-bottom: 1em; }
1349
+ .tome-content li { margin-bottom: 0.3em; line-height: 1.7; }
1350
+ .tome-content code { font-family: var(--font-code); font-size: 0.88em; background: var(--cdBg); padding: 0.15em 0.4em; border-radius: 2px; color: var(--ac); }
1351
+ .tome-content pre { margin-bottom: 1.2em; border-radius: 2px; overflow-x: auto; border: 1px solid var(--bd); }
1352
+ .tome-content pre code { background: none; padding: 1em 1.2em; display: block; font-size: 12.5px; line-height: 1.7; color: var(--cdTx); }
1353
+ .tome-content blockquote { border-left: 3px solid var(--ac); padding: 0.5em 1em; margin: 1em 0; background: var(--acD); border-radius: 0 2px 2px 0; }
1354
+ .tome-content blockquote p { color: var(--tx2); margin: 0; }
1355
+ .tome-content table { width: 100%; border-collapse: collapse; margin-bottom: 1em; }
1356
+ .tome-content th, .tome-content td { padding: 0.5em 0.8em; border: 1px solid var(--bd); text-align: left; font-size: 0.9em; }
1357
+ .tome-content th { background: var(--sf); font-weight: 600; }
1358
+ .tome-content img { max-width: 100%; border-radius: 2px; }
1359
+ .tome-content hr { border: none; border-top: 1px solid var(--bd); margin: 2em 0; }
1360
+
1361
+ /* Selection style */
1362
+ ::selection { background: var(--acD); color: var(--ac); }
1363
+
1364
+ /* Scrollbar style */
1365
+ ::-webkit-scrollbar { width: 5px; height: 5px; }
1366
+ ::-webkit-scrollbar-track { background: transparent; }
1367
+ ::-webkit-scrollbar-thumb { background: var(--bd); border-radius: 10px; }
1368
+
1369
+ /* Grain overlay */
1370
+ .tome-grain::before {
1371
+ content: ""; position: fixed; inset: 0; z-index: 9999; pointer-events: none;
1372
+ opacity: .35;
1373
+ background-image: url("data:image/svg+xml,%3Csvg viewBox='0 0 256 256' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.9' numOctaves='4' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23n)' opacity='0.04'/%3E%3C/svg%3E");
1374
+ background-repeat: repeat; background-size: 256px;
1375
+ }
1376
+
1377
+ /* Shiki dual-theme support */
1378
+ .shiki { background: var(--cdBg) !important; }
1379
+ html.dark .shiki .shiki-light { display: none; }
1380
+ html.light .shiki .shiki-dark { display: none; }
1381
+ `;
1382
+ async function loadPage(id) {
1383
+ try {
1384
+ const route = routes.find((r) => r.id === id);
1385
+ const mod = await loadPageModule(id);
1386
+ if (route?.isMdx && mod.meta) {
1387
+ return {
1388
+ isMdx: true,
1389
+ component: mod.default,
1390
+ frontmatter: mod.meta.frontmatter,
1391
+ headings: mod.meta.headings
1392
+ };
1393
+ }
1394
+ if (!mod.default) return null;
1395
+ return { isMdx: false, ...mod.default };
1396
+ } catch (err) {
1397
+ console.error(`Failed to load page: ${id}`, err);
1398
+ return null;
1399
+ }
1400
+ }
1401
+ function App() {
1402
+ const [currentPageId, setCurrentPageId] = useState3(() => {
1403
+ const hash = window.location.hash.slice(1);
1404
+ if (hash && routes.some((r) => r.id === hash)) return hash;
1405
+ return routes[0]?.id || "index";
1406
+ });
1407
+ const [pageData, setPageData] = useState3(null);
1408
+ const [loading, setLoading] = useState3(true);
1409
+ const navigateTo = useCallback3(async (id) => {
1410
+ setLoading(true);
1411
+ setCurrentPageId(id);
1412
+ window.location.hash = id;
1413
+ const data = await loadPage(id);
1414
+ setPageData(data);
1415
+ setLoading(false);
1416
+ }, []);
1417
+ useEffect3(() => {
1418
+ navigateTo(currentPageId);
1419
+ }, []);
1420
+ useEffect3(() => {
1421
+ const onHashChange = () => {
1422
+ const hash = window.location.hash.slice(1);
1423
+ if (hash && hash !== currentPageId && routes.some((r) => r.id === hash)) {
1424
+ navigateTo(hash);
1425
+ }
1426
+ };
1427
+ window.addEventListener("hashchange", onHashChange);
1428
+ return () => window.removeEventListener("hashchange", onHashChange);
1429
+ }, [currentPageId, navigateTo]);
1430
+ const allPages = routes.map((r) => ({
1431
+ id: r.id,
1432
+ title: r.frontmatter.title,
1433
+ description: r.frontmatter.description
1434
+ }));
1435
+ return /* @__PURE__ */ jsxs3(Fragment, { children: [
1436
+ /* @__PURE__ */ jsx3("style", { children: contentStyles }),
1437
+ /* @__PURE__ */ jsx3(
1438
+ Shell,
1439
+ {
1440
+ config,
1441
+ navigation,
1442
+ currentPageId,
1443
+ pageHtml: !pageData?.isMdx ? loading ? "<p>Loading...</p>" : pageData?.html || "<p>Page not found</p>" : void 0,
1444
+ pageComponent: pageData?.isMdx ? pageData.component : void 0,
1445
+ mdxComponents: MDX_COMPONENTS,
1446
+ pageTitle: pageData?.frontmatter.title || (loading ? "Loading..." : "Not Found"),
1447
+ pageDescription: pageData?.frontmatter.description,
1448
+ headings: pageData?.headings || [],
1449
+ onNavigate: navigateTo,
1450
+ allPages,
1451
+ docContext
1452
+ }
1453
+ )
1454
+ ] });
1455
+ }
1456
+ var container = document.getElementById("tome-root");
1457
+ if (container) {
1458
+ const root = createRoot(container);
1459
+ root.render(/* @__PURE__ */ jsx3(App, {}));
1460
+ }
1461
+ var entry_default = App;
1462
+
1463
+ export {
1464
+ THEME_PRESETS,
1465
+ AiChat,
1466
+ Shell,
1467
+ entry_default
1468
+ };