@steez-ui/ui 0.1.0

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.
Files changed (106) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +52 -0
  3. package/dist/components/Button.d.ts +11 -0
  4. package/dist/components/Button.d.ts.map +1 -0
  5. package/dist/components/Button.js +8 -0
  6. package/dist/components/Button.js.map +1 -0
  7. package/dist/components/CopyButton.d.ts +10 -0
  8. package/dist/components/CopyButton.d.ts.map +1 -0
  9. package/dist/components/CopyButton.js +25 -0
  10. package/dist/components/CopyButton.js.map +1 -0
  11. package/dist/components/CopyButton.module.css +30 -0
  12. package/dist/components/CornerBracketCard.d.ts +7 -0
  13. package/dist/components/CornerBracketCard.d.ts.map +1 -0
  14. package/dist/components/CornerBracketCard.js +6 -0
  15. package/dist/components/CornerBracketCard.js.map +1 -0
  16. package/dist/components/CornerBracketCard.module.css +109 -0
  17. package/dist/components/CyberpunkCheckbox.d.ts +7 -0
  18. package/dist/components/CyberpunkCheckbox.d.ts.map +1 -0
  19. package/dist/components/CyberpunkCheckbox.js +8 -0
  20. package/dist/components/CyberpunkCheckbox.js.map +1 -0
  21. package/dist/components/CyberpunkCheckbox.module.css +57 -0
  22. package/dist/components/CyberpunkInput.d.ts +9 -0
  23. package/dist/components/CyberpunkInput.d.ts.map +1 -0
  24. package/dist/components/CyberpunkInput.js +9 -0
  25. package/dist/components/CyberpunkInput.js.map +1 -0
  26. package/dist/components/CyberpunkInput.module.css +115 -0
  27. package/dist/components/CyberpunkRadio.d.ts +17 -0
  28. package/dist/components/CyberpunkRadio.d.ts.map +1 -0
  29. package/dist/components/CyberpunkRadio.js +11 -0
  30. package/dist/components/CyberpunkRadio.js.map +1 -0
  31. package/dist/components/CyberpunkRadio.module.css +64 -0
  32. package/dist/components/CyberpunkSelect.d.ts +12 -0
  33. package/dist/components/CyberpunkSelect.d.ts.map +1 -0
  34. package/dist/components/CyberpunkSelect.js +8 -0
  35. package/dist/components/CyberpunkSelect.js.map +1 -0
  36. package/dist/components/CyberpunkSelect.module.css +108 -0
  37. package/dist/components/CyberpunkSlider.d.ts +10 -0
  38. package/dist/components/CyberpunkSlider.d.ts.map +1 -0
  39. package/dist/components/CyberpunkSlider.js +9 -0
  40. package/dist/components/CyberpunkSlider.js.map +1 -0
  41. package/dist/components/CyberpunkSlider.module.css +98 -0
  42. package/dist/components/CyberpunkTextarea.d.ts +7 -0
  43. package/dist/components/CyberpunkTextarea.d.ts.map +1 -0
  44. package/dist/components/CyberpunkTextarea.js +8 -0
  45. package/dist/components/CyberpunkTextarea.js.map +1 -0
  46. package/dist/components/CyberpunkTextarea.module.css +73 -0
  47. package/dist/components/CyberpunkTile.d.ts +10 -0
  48. package/dist/components/CyberpunkTile.d.ts.map +1 -0
  49. package/dist/components/CyberpunkTile.js +6 -0
  50. package/dist/components/CyberpunkTile.js.map +1 -0
  51. package/dist/components/CyberpunkTile.module.css +133 -0
  52. package/dist/components/ErrorMessage.d.ts +11 -0
  53. package/dist/components/ErrorMessage.d.ts.map +1 -0
  54. package/dist/components/ErrorMessage.js +16 -0
  55. package/dist/components/ErrorMessage.js.map +1 -0
  56. package/dist/components/ErrorMessage.module.css +75 -0
  57. package/dist/components/LoadingProgressBar.d.ts +8 -0
  58. package/dist/components/LoadingProgressBar.d.ts.map +1 -0
  59. package/dist/components/LoadingProgressBar.js +13 -0
  60. package/dist/components/LoadingProgressBar.js.map +1 -0
  61. package/dist/components/LoadingProgressBar.module.css +42 -0
  62. package/dist/components/PageHeader.d.ts +17 -0
  63. package/dist/components/PageHeader.d.ts.map +1 -0
  64. package/dist/components/PageHeader.js +7 -0
  65. package/dist/components/PageHeader.js.map +1 -0
  66. package/dist/components/PageHeader.module.css +118 -0
  67. package/dist/components/PageTemplate.d.ts +20 -0
  68. package/dist/components/PageTemplate.d.ts.map +1 -0
  69. package/dist/components/PageTemplate.js +8 -0
  70. package/dist/components/PageTemplate.js.map +1 -0
  71. package/dist/components/PageTemplate.module.css +55 -0
  72. package/dist/components/SegmentedControl.d.ts +16 -0
  73. package/dist/components/SegmentedControl.d.ts.map +1 -0
  74. package/dist/components/SegmentedControl.js +9 -0
  75. package/dist/components/SegmentedControl.js.map +1 -0
  76. package/dist/components/SegmentedControl.module.css +51 -0
  77. package/dist/components/StatusMessage.d.ts +6 -0
  78. package/dist/components/StatusMessage.d.ts.map +1 -0
  79. package/dist/components/StatusMessage.js +7 -0
  80. package/dist/components/StatusMessage.js.map +1 -0
  81. package/dist/components/StatusMessage.module.css +60 -0
  82. package/dist/components/TabbedPanel.d.ts +23 -0
  83. package/dist/components/TabbedPanel.d.ts.map +1 -0
  84. package/dist/components/TabbedPanel.js +36 -0
  85. package/dist/components/TabbedPanel.js.map +1 -0
  86. package/dist/components/TabbedPanel.module.css +61 -0
  87. package/dist/components/ThemeToggle.d.ts +8 -0
  88. package/dist/components/ThemeToggle.d.ts.map +1 -0
  89. package/dist/components/ThemeToggle.js +22 -0
  90. package/dist/components/ThemeToggle.js.map +1 -0
  91. package/dist/components/ThemeToggle.module.css +23 -0
  92. package/dist/components/ThemedCard.d.ts +7 -0
  93. package/dist/components/ThemedCard.d.ts.map +1 -0
  94. package/dist/components/ThemedCard.js +6 -0
  95. package/dist/components/ThemedCard.js.map +1 -0
  96. package/dist/components/ThemedCard.module.css +22 -0
  97. package/dist/hooks/useStableId.d.ts +2 -0
  98. package/dist/hooks/useStableId.d.ts.map +1 -0
  99. package/dist/hooks/useStableId.js +6 -0
  100. package/dist/hooks/useStableId.js.map +1 -0
  101. package/dist/index.d.ts +20 -0
  102. package/dist/index.d.ts.map +1 -0
  103. package/dist/index.js +20 -0
  104. package/dist/index.js.map +1 -0
  105. package/dist/styles/Buttons.module.css +96 -0
  106. package/package.json +35 -0
@@ -0,0 +1,13 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import styles from "./LoadingProgressBar.module.css";
3
+ export const LOADING_PROGRESS_SEGMENT_COUNT = 20;
4
+ const BAR_INDEXES = Array.from({ length: LOADING_PROGRESS_SEGMENT_COUNT }, (_, index) => index);
5
+ export function LoadingProgressBar({ progress, className, valueLabel, }) {
6
+ const clampedProgress = Math.max(0, Math.min(100, progress));
7
+ const filledBars = Math.round((clampedProgress / 100) * LOADING_PROGRESS_SEGMENT_COUNT);
8
+ return (_jsxs("div", { className: className ? `${styles.root} ${className}` : styles.root, children: [_jsx("div", { className: styles.barGroup, children: BAR_INDEXES.map((index) => {
9
+ const isFilled = index < filledBars;
10
+ return (_jsx("div", { className: `${styles.bar} ${isFilled ? styles.barFilled : ""}`.trim(), style: { ["--bar-delay"]: `${index * 20}ms` } }, index));
11
+ }) }), _jsx("div", { className: styles.value, children: valueLabel ?? `${Math.round(clampedProgress)}%` })] }));
12
+ }
13
+ //# sourceMappingURL=LoadingProgressBar.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadingProgressBar.js","sourceRoot":"","sources":["../../src/components/LoadingProgressBar.tsx"],"names":[],"mappings":";AAEA,OAAO,MAAM,MAAM,iCAAiC,CAAC;AAErD,MAAM,CAAC,MAAM,8BAA8B,GAAG,EAAE,CAAC;AAEjD,MAAM,WAAW,GAAG,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,8BAA8B,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;AAQhG,MAAM,UAAU,kBAAkB,CAAC,EACjC,QAAQ,EACR,SAAS,EACT,UAAU,GACc;IACxB,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC,CAAC;IAC7D,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,eAAe,GAAG,GAAG,CAAC,GAAG,8BAA8B,CAAC,CAAC;IAExF,OAAO,CACL,eAAK,SAAS,EAAE,SAAS,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,IAAI,IAAI,SAAS,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,aACrE,cAAK,SAAS,EAAE,MAAM,CAAC,QAAQ,YAC5B,WAAW,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;oBACzB,MAAM,QAAQ,GAAG,KAAK,GAAG,UAAU,CAAC;oBACpC,OAAO,CACL,cAEE,SAAS,EAAE,GAAG,MAAM,CAAC,GAAG,IAAI,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,EAAE,EACrE,KAAK,EAAE,EAAE,CAAC,aAAuB,CAAC,EAAE,GAAG,KAAK,GAAG,EAAE,IAAI,EAAE,IAFlD,KAAK,CAGV,CACH,CAAC;gBACJ,CAAC,CAAC,GACE,EACN,cAAK,SAAS,EAAE,MAAM,CAAC,KAAK,YAAG,UAAU,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,GAAG,GAAO,IACjF,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,42 @@
1
+ .root {
2
+ display: flex;
3
+ align-items: center;
4
+ gap: 4px;
5
+ }
6
+
7
+ .barGroup {
8
+ display: flex;
9
+ width: 100%;
10
+ max-width: 650px;
11
+ flex-wrap: wrap;
12
+ justify-content: center;
13
+ gap: 3px;
14
+ }
15
+
16
+ .bar {
17
+ width: clamp(10px, 2.5vw, 28px);
18
+ height: clamp(10px, 2.5vw, 28px);
19
+ box-sizing: border-box;
20
+ flex-shrink: 0;
21
+ border: none;
22
+ border-radius: 4px;
23
+ background: rgba(203, 203, 204, 0.15);
24
+ transform: skewX(-15deg);
25
+ transition: all 0.1s ease-out;
26
+ transition-delay: var(--bar-delay, 0ms);
27
+ }
28
+
29
+ .barFilled {
30
+ background: currentColor;
31
+ }
32
+
33
+ .value {
34
+ width: 4ch;
35
+ margin-left: 16px;
36
+ text-align: right;
37
+ font-family: var(--font-mono);
38
+ font-size: clamp(18px, 5vw, 32px);
39
+ font-weight: 900;
40
+ letter-spacing: 2px;
41
+ }
42
+
@@ -0,0 +1,17 @@
1
+ import { ReactNode } from "react";
2
+ import type { SteezIconName } from "@steez-ui/icons";
3
+ export interface PageHeaderProps {
4
+ title: string;
5
+ description?: string;
6
+ icon?: SteezIconName;
7
+ extra?: ReactNode;
8
+ className?: string;
9
+ brand?: ReactNode;
10
+ onBrandClick?: () => void;
11
+ onBack?: () => void;
12
+ onSettings?: () => void;
13
+ onViewerToggle?: () => void;
14
+ viewerVisible?: boolean;
15
+ }
16
+ export declare function PageHeader({ title, description, icon, extra, className, brand, onBrandClick, onBack, onSettings, onViewerToggle, viewerVisible, }: PageHeaderProps): import("react/jsx-runtime").JSX.Element;
17
+ //# sourceMappingURL=PageHeader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PageHeader.d.ts","sourceRoot":"","sources":["../../src/components/PageHeader.tsx"],"names":[],"mappings":"AAAA,OAAc,EAAE,SAAS,EAAE,MAAM,OAAO,CAAC;AAGzC,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAIrD,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,IAAI,CAAC,EAAE,aAAa,CAAC;IACrB,KAAK,CAAC,EAAE,SAAS,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,KAAK,CAAC,EAAE,SAAS,CAAC;IAClB,YAAY,CAAC,EAAE,MAAM,IAAI,CAAC;IAC1B,MAAM,CAAC,EAAE,MAAM,IAAI,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,IAAI,CAAC;IACxB,cAAc,CAAC,EAAE,MAAM,IAAI,CAAC;IAC5B,aAAa,CAAC,EAAE,OAAO,CAAC;CACzB;AAED,wBAAgB,UAAU,CAAC,EACzB,KAAK,EACL,WAAW,EACX,IAAI,EACJ,KAAK,EACL,SAAc,EACd,KAAK,EACL,YAAY,EACZ,MAAM,EACN,UAAU,EACV,cAAc,EACd,aAAa,GACd,EAAE,eAAe,2CAgDjB"}
@@ -0,0 +1,7 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { EyeIcon, Icon, SlidersIcon } from "@steez-ui/icons";
3
+ import styles from "./PageHeader.module.css";
4
+ export function PageHeader({ title, description, icon, extra, className = "", brand, onBrandClick, onBack, onSettings, onViewerToggle, viewerVisible, }) {
5
+ return (_jsx("div", { className: `${styles.header} ${className}`.trim(), children: _jsxs("div", { className: styles.headerRow, children: [_jsxs("div", { className: styles.brandCluster, children: [brand || onBrandClick ? (_jsx("button", { type: "button", className: styles.brandButton, onClick: onBrandClick, children: _jsx("span", { className: styles.brandVisual, children: brand ?? title.slice(0, 1) }) })) : null, _jsxs("div", { className: styles.headerContent, children: [_jsxs("h2", { children: [icon ? (_jsx("span", { className: styles.titleIcon, children: _jsx(Icon, { icon: icon, width: 18, height: 18 }) })) : null, title] }), description ? _jsx("p", { children: description }) : null] })] }), _jsxs("div", { className: styles.headerExtra, children: [extra, onBack ? (_jsx("button", { type: "button", className: styles.iconButton, onClick: onBack, "aria-label": "Go back", children: _jsx(Icon, { icon: "chevronLeft", width: 18, height: 18 }) })) : null, onSettings ? (_jsx("button", { type: "button", className: styles.iconButton, onClick: onSettings, "aria-label": "Open settings", children: _jsx(SlidersIcon, { width: 18, height: 18 }) })) : null, onViewerToggle ? (_jsx("button", { type: "button", className: styles.iconButton, onClick: onViewerToggle, "aria-label": viewerVisible ? "Hide viewer" : "Show viewer", children: _jsx(EyeIcon, { width: 18, height: 18 }) })) : null] })] }) }));
6
+ }
7
+ //# sourceMappingURL=PageHeader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PageHeader.js","sourceRoot":"","sources":["../../src/components/PageHeader.tsx"],"names":[],"mappings":";AAEA,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAG7D,OAAO,MAAM,MAAM,yBAAyB,CAAC;AAgB7C,MAAM,UAAU,UAAU,CAAC,EACzB,KAAK,EACL,WAAW,EACX,IAAI,EACJ,KAAK,EACL,SAAS,GAAG,EAAE,EACd,KAAK,EACL,YAAY,EACZ,MAAM,EACN,UAAU,EACV,cAAc,EACd,aAAa,GACG;IAChB,OAAO,CACL,cAAK,SAAS,EAAE,GAAG,MAAM,CAAC,MAAM,IAAI,SAAS,EAAE,CAAC,IAAI,EAAE,YACpD,eAAK,SAAS,EAAE,MAAM,CAAC,SAAS,aAC9B,eAAK,SAAS,EAAE,MAAM,CAAC,YAAY,aAChC,KAAK,IAAI,YAAY,CAAC,CAAC,CAAC,CACvB,iBAAQ,IAAI,EAAC,QAAQ,EAAC,SAAS,EAAE,MAAM,CAAC,WAAW,EAAE,OAAO,EAAE,YAAY,YACxE,eAAM,SAAS,EAAE,MAAM,CAAC,WAAW,YAAG,KAAK,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,GAAQ,GACjE,CACV,CAAC,CAAC,CAAC,IAAI,EACR,eAAK,SAAS,EAAE,MAAM,CAAC,aAAa,aAClC,yBACG,IAAI,CAAC,CAAC,CAAC,CACN,eAAM,SAAS,EAAE,MAAM,CAAC,SAAS,YAC/B,KAAC,IAAI,IAAC,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,GACtC,CACR,CAAC,CAAC,CAAC,IAAI,EACP,KAAK,IACH,EACJ,WAAW,CAAC,CAAC,CAAC,sBAAI,WAAW,GAAK,CAAC,CAAC,CAAC,IAAI,IACtC,IACF,EACN,eAAK,SAAS,EAAE,MAAM,CAAC,WAAW,aAC/B,KAAK,EACL,MAAM,CAAC,CAAC,CAAC,CACR,iBAAQ,IAAI,EAAC,QAAQ,EAAC,SAAS,EAAE,MAAM,CAAC,UAAU,EAAE,OAAO,EAAE,MAAM,gBAAa,SAAS,YACvF,KAAC,IAAI,IAAC,IAAI,EAAC,aAAa,EAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,GAC3C,CACV,CAAC,CAAC,CAAC,IAAI,EACP,UAAU,CAAC,CAAC,CAAC,CACZ,iBAAQ,IAAI,EAAC,QAAQ,EAAC,SAAS,EAAE,MAAM,CAAC,UAAU,EAAE,OAAO,EAAE,UAAU,gBAAa,eAAe,YACjG,KAAC,WAAW,IAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,GAC/B,CACV,CAAC,CAAC,CAAC,IAAI,EACP,cAAc,CAAC,CAAC,CAAC,CAChB,iBACE,IAAI,EAAC,QAAQ,EACb,SAAS,EAAE,MAAM,CAAC,UAAU,EAC5B,OAAO,EAAE,cAAc,gBACX,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,YAEzD,KAAC,OAAO,IAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,GAC3B,CACV,CAAC,CAAC,CAAC,IAAI,IACJ,IACF,GACF,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,118 @@
1
+ .header {
2
+ padding: 16px 24px;
3
+ background: var(--bg-secondary);
4
+ }
5
+
6
+ .headerRow {
7
+ display: flex;
8
+ align-items: center;
9
+ gap: 12px;
10
+ }
11
+
12
+ .brandCluster {
13
+ display: flex;
14
+ align-items: center;
15
+ gap: 12px;
16
+ }
17
+
18
+ .brandButton {
19
+ border: none;
20
+ background: transparent;
21
+ padding: 0;
22
+ margin: 0;
23
+ cursor: pointer;
24
+ display: inline-flex;
25
+ align-items: center;
26
+ }
27
+
28
+ .brandButton:focus-visible {
29
+ outline: 2px solid var(--color-border-hover);
30
+ outline-offset: 2px;
31
+ border-radius: 6px;
32
+ }
33
+
34
+ .brandVisual {
35
+ height: 48px;
36
+ width: 48px;
37
+ display: inline-flex;
38
+ align-items: center;
39
+ justify-content: center;
40
+ border: 1px solid var(--color-border-default);
41
+ border-radius: 6px;
42
+ color: var(--text-primary);
43
+ background: color-mix(in srgb, var(--bg-primary) 90%, transparent);
44
+ }
45
+
46
+ .headerContent {
47
+ line-height: normal;
48
+ }
49
+
50
+ .headerContent h2 {
51
+ margin: 0;
52
+ font-size: 32px;
53
+ font-weight: 400;
54
+ font-family: var(--font-display);
55
+ letter-spacing: 0.03em;
56
+ color: var(--color-text-primary);
57
+ }
58
+
59
+ .titleIcon {
60
+ display: inline-flex;
61
+ margin-right: 8px;
62
+ vertical-align: middle;
63
+ }
64
+
65
+ .headerContent p {
66
+ margin: 0;
67
+ font-size: 14px;
68
+ color: var(--color-text-secondary);
69
+ }
70
+
71
+ .headerExtra {
72
+ margin-left: auto;
73
+ display: flex;
74
+ align-items: center;
75
+ gap: 8px;
76
+ }
77
+
78
+ .iconButton {
79
+ border: none;
80
+ background: transparent;
81
+ padding: 8px;
82
+ margin: 0;
83
+ cursor: pointer;
84
+ display: inline-flex;
85
+ align-items: center;
86
+ color: var(--color-text-secondary);
87
+ transition: color var(--transition-fast);
88
+ }
89
+
90
+ .iconButton:hover {
91
+ color: var(--color-text-primary);
92
+ }
93
+
94
+ .iconButton:focus-visible {
95
+ outline: 2px solid var(--color-border-hover);
96
+ outline-offset: 2px;
97
+ border-radius: 4px;
98
+ }
99
+
100
+ @media (max-width: 767px) {
101
+ .header {
102
+ padding: 12px 16px;
103
+ }
104
+
105
+ .brandVisual {
106
+ height: 32px;
107
+ width: 32px;
108
+ }
109
+
110
+ .headerContent h2 {
111
+ font-size: 22px;
112
+ }
113
+
114
+ .headerContent p {
115
+ display: none;
116
+ }
117
+ }
118
+
@@ -0,0 +1,20 @@
1
+ import React from "react";
2
+ import { type PageHeaderProps } from "./PageHeader";
3
+ interface SubTab {
4
+ id: string;
5
+ label: string;
6
+ }
7
+ export interface PageTemplateProps extends Pick<PageHeaderProps, "title" | "description" | "icon" | "onBack" | "onSettings" | "brand" | "onBrandClick" | "onViewerToggle" | "viewerVisible"> {
8
+ actions?: React.ReactNode;
9
+ extra?: React.ReactNode;
10
+ className?: string;
11
+ subTabs?: SubTab[];
12
+ activeSubTab?: string;
13
+ onSubTabChange?: (tabId: string) => void;
14
+ children?: React.ReactNode;
15
+ loading?: boolean;
16
+ showTitle?: boolean;
17
+ }
18
+ export declare function PageTemplate({ title, actions, extra, description, icon, subTabs, activeSubTab, onSubTabChange, onBack, onSettings, brand, onBrandClick, onViewerToggle, viewerVisible, children, loading, showTitle, className, }: PageTemplateProps): import("react/jsx-runtime").JSX.Element;
19
+ export {};
20
+ //# sourceMappingURL=PageTemplate.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PageTemplate.d.ts","sourceRoot":"","sources":["../../src/components/PageTemplate.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,OAAO,EAAc,KAAK,eAAe,EAAE,MAAM,cAAc,CAAC;AAGhE,UAAU,MAAM;IACd,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,iBACf,SAAQ,IAAI,CACV,eAAe,EACb,OAAO,GACP,aAAa,GACb,MAAM,GACN,QAAQ,GACR,YAAY,GACZ,OAAO,GACP,cAAc,GACd,gBAAgB,GAChB,eAAe,CAClB;IACD,OAAO,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IAC1B,KAAK,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IACxB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;IACnB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,cAAc,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IACzC,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IAC3B,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,wBAAgB,YAAY,CAAC,EAC3B,KAAK,EACL,OAAO,EACP,KAAK,EACL,WAAW,EACX,IAAI,EACJ,OAAO,EACP,YAAY,EACZ,cAAc,EACd,MAAM,EACN,UAAU,EACV,KAAK,EACL,YAAY,EACZ,cAAc,EACd,aAAa,EACb,QAAQ,EACR,OAAO,EACP,SAAgB,EAChB,SAAc,GACf,EAAE,iBAAiB,2CA4CnB"}
@@ -0,0 +1,8 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { PageHeader } from "./PageHeader";
3
+ import styles from "./PageTemplate.module.css";
4
+ export function PageTemplate({ title, actions, extra, description, icon, subTabs, activeSubTab, onSubTabChange, onBack, onSettings, brand, onBrandClick, onViewerToggle, viewerVisible, children, loading, showTitle = true, className = "", }) {
5
+ const headerExtra = extra ?? actions;
6
+ return (_jsxs("div", { className: `${styles.root} ${className}`.trim(), children: [showTitle ? (_jsx("div", { className: styles.header, children: _jsx(PageHeader, { title: title, description: description, icon: icon, extra: headerExtra, onBack: onBack, onSettings: onSettings, brand: brand, onBrandClick: onBrandClick, onViewerToggle: onViewerToggle, viewerVisible: viewerVisible }) })) : null, subTabs?.length ? (_jsx("div", { className: styles.subTabs, role: "tablist", "aria-label": `${title} sections`, children: subTabs.map((tab) => (_jsx("button", { onClick: () => onSubTabChange?.(tab.id), className: `${styles.subTabButton} ${activeSubTab === tab.id ? styles.subTabButtonActive : ""}`.trim(), type: "button", role: "tab", "aria-selected": activeSubTab === tab.id, children: tab.label }, tab.id))) })) : null, _jsx("div", { className: styles.content, children: loading ? _jsx("div", { className: styles.loading, children: "Loading..." }) : children ?? null })] }));
7
+ }
8
+ //# sourceMappingURL=PageTemplate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PageTemplate.js","sourceRoot":"","sources":["../../src/components/PageTemplate.tsx"],"names":[],"mappings":";AAEA,OAAO,EAAE,UAAU,EAAwB,MAAM,cAAc,CAAC;AAChE,OAAO,MAAM,MAAM,2BAA2B,CAAC;AA+B/C,MAAM,UAAU,YAAY,CAAC,EAC3B,KAAK,EACL,OAAO,EACP,KAAK,EACL,WAAW,EACX,IAAI,EACJ,OAAO,EACP,YAAY,EACZ,cAAc,EACd,MAAM,EACN,UAAU,EACV,KAAK,EACL,YAAY,EACZ,cAAc,EACd,aAAa,EACb,QAAQ,EACR,OAAO,EACP,SAAS,GAAG,IAAI,EAChB,SAAS,GAAG,EAAE,GACI;IAClB,MAAM,WAAW,GAAG,KAAK,IAAI,OAAO,CAAC;IAErC,OAAO,CACL,eAAK,SAAS,EAAE,GAAG,MAAM,CAAC,IAAI,IAAI,SAAS,EAAE,CAAC,IAAI,EAAE,aACjD,SAAS,CAAC,CAAC,CAAC,CACX,cAAK,SAAS,EAAE,MAAM,CAAC,MAAM,YAC3B,KAAC,UAAU,IACT,KAAK,EAAE,KAAK,EACZ,WAAW,EAAE,WAAW,EACxB,IAAI,EAAE,IAAI,EACV,KAAK,EAAE,WAAW,EAClB,MAAM,EAAE,MAAM,EACd,UAAU,EAAE,UAAU,EACtB,KAAK,EAAE,KAAK,EACZ,YAAY,EAAE,YAAY,EAC1B,cAAc,EAAE,cAAc,EAC9B,aAAa,EAAE,aAAa,GAC5B,GACE,CACP,CAAC,CAAC,CAAC,IAAI,EAEP,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,CACjB,cAAK,SAAS,EAAE,MAAM,CAAC,OAAO,EAAE,IAAI,EAAC,SAAS,gBAAa,GAAG,KAAK,WAAW,YAC3E,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CACpB,iBAEE,OAAO,EAAE,GAAG,EAAE,CAAC,cAAc,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EACvC,SAAS,EAAE,GAAG,MAAM,CAAC,YAAY,IAAI,YAAY,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,EAAE,EACtG,IAAI,EAAC,QAAQ,EACb,IAAI,EAAC,KAAK,mBACK,YAAY,KAAK,GAAG,CAAC,EAAE,YAErC,GAAG,CAAC,KAAK,IAPL,GAAG,CAAC,EAAE,CAQJ,CACV,CAAC,GACE,CACP,CAAC,CAAC,CAAC,IAAI,EAER,cAAK,SAAS,EAAE,MAAM,CAAC,OAAO,YAC3B,OAAO,CAAC,CAAC,CAAC,cAAK,SAAS,EAAE,MAAM,CAAC,OAAO,2BAAkB,CAAC,CAAC,CAAC,QAAQ,IAAI,IAAI,GAC1E,IACF,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,55 @@
1
+ .root {
2
+ display: flex;
3
+ flex-direction: column;
4
+ }
5
+
6
+ .header {
7
+ flex-shrink: 0;
8
+ position: sticky;
9
+ top: 0;
10
+ z-index: 10;
11
+ }
12
+
13
+ .subTabs {
14
+ display: flex;
15
+ gap: 4px;
16
+ padding: 0 24px 16px;
17
+ border-bottom: 1px solid var(--border-color);
18
+ flex-shrink: 0;
19
+ }
20
+
21
+ .subTabButton {
22
+ padding: 10px 16px;
23
+ background: transparent;
24
+ border: none;
25
+ border-bottom: 2px solid transparent;
26
+ color: var(--text-secondary);
27
+ cursor: pointer;
28
+ font-size: 13px;
29
+ font-family: var(--font-mono);
30
+ font-weight: 600;
31
+ text-transform: uppercase;
32
+ letter-spacing: 0.08em;
33
+ transition: all var(--transition-fast);
34
+ }
35
+
36
+ .subTabButtonActive {
37
+ border-bottom-color: var(--accent-primary);
38
+ color: var(--text-primary);
39
+ font-weight: 800;
40
+ }
41
+
42
+ .content {
43
+ display: flex;
44
+ flex-direction: column;
45
+ padding: 1.5rem;
46
+ }
47
+
48
+ .loading {
49
+ display: flex;
50
+ align-items: center;
51
+ justify-content: center;
52
+ padding: 60px;
53
+ color: var(--text-secondary);
54
+ }
55
+
@@ -0,0 +1,16 @@
1
+ export interface SegmentedControlOption {
2
+ value: string;
3
+ label: string;
4
+ }
5
+ export interface SegmentedControlProps {
6
+ value: string;
7
+ options: SegmentedControlOption[];
8
+ onChange: (value: string) => void;
9
+ label?: string;
10
+ hint?: string;
11
+ ariaLabel?: string;
12
+ className?: string;
13
+ compact?: boolean;
14
+ }
15
+ export declare function SegmentedControl({ value, options, onChange, label, hint, ariaLabel, className, compact, }: SegmentedControlProps): import("react/jsx-runtime").JSX.Element;
16
+ //# sourceMappingURL=SegmentedControl.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SegmentedControl.d.ts","sourceRoot":"","sources":["../../src/components/SegmentedControl.tsx"],"names":[],"mappings":"AAIA,MAAM,WAAW,sBAAsB;IACrC,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,qBAAqB;IACpC,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,sBAAsB,EAAE,CAAC;IAClC,QAAQ,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IAClC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AAED,wBAAgB,gBAAgB,CAAC,EAC/B,KAAK,EACL,OAAO,EACP,QAAQ,EACR,KAAK,EACL,IAAI,EACJ,SAAS,EACT,SAAc,EACd,OAAe,GAChB,EAAE,qBAAqB,2CA4BvB"}
@@ -0,0 +1,9 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import styles from "./SegmentedControl.module.css";
3
+ export function SegmentedControl({ value, options, onChange, label, hint, ariaLabel, className = "", compact = false, }) {
4
+ return (_jsxs("div", { className: `${styles.wrapper} ${className}`.trim(), children: [label ? _jsx("div", { className: styles.label, children: label }) : null, _jsx("div", { className: `${styles.track} ${compact ? styles.trackCompact : ""}`.trim(), role: "tablist", "aria-label": ariaLabel || label, children: options.map((option) => {
5
+ const isActive = option.value === value;
6
+ return (_jsx("button", { type: "button", role: "tab", "aria-selected": isActive, className: `${styles.option} ${isActive ? styles.optionActive : ""} ${compact ? styles.optionCompact : ""}`.trim(), onClick: () => onChange(option.value), children: option.label }, option.value));
7
+ }) }), hint ? _jsx("div", { className: styles.hint, children: hint }) : null] }));
8
+ }
9
+ //# sourceMappingURL=SegmentedControl.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SegmentedControl.js","sourceRoot":"","sources":["../../src/components/SegmentedControl.tsx"],"names":[],"mappings":";AAEA,OAAO,MAAM,MAAM,+BAA+B,CAAC;AAkBnD,MAAM,UAAU,gBAAgB,CAAC,EAC/B,KAAK,EACL,OAAO,EACP,QAAQ,EACR,KAAK,EACL,IAAI,EACJ,SAAS,EACT,SAAS,GAAG,EAAE,EACd,OAAO,GAAG,KAAK,GACO;IACtB,OAAO,CACL,eAAK,SAAS,EAAE,GAAG,MAAM,CAAC,OAAO,IAAI,SAAS,EAAE,CAAC,IAAI,EAAE,aACpD,KAAK,CAAC,CAAC,CAAC,cAAK,SAAS,EAAE,MAAM,CAAC,KAAK,YAAG,KAAK,GAAO,CAAC,CAAC,CAAC,IAAI,EAC3D,cACE,SAAS,EAAE,GAAG,MAAM,CAAC,KAAK,IAAI,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,EAAE,EACzE,IAAI,EAAC,SAAS,gBACF,SAAS,IAAI,KAAK,YAE7B,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE;oBACtB,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,KAAK,KAAK,CAAC;oBACxC,OAAO,CACL,iBAEE,IAAI,EAAC,QAAQ,EACb,IAAI,EAAC,KAAK,mBACK,QAAQ,EACvB,SAAS,EAAE,GAAG,MAAM,CAAC,MAAM,IAAI,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,IAAI,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,EAAE,EAClH,OAAO,EAAE,GAAG,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,YAEpC,MAAM,CAAC,KAAK,IAPR,MAAM,CAAC,KAAK,CAQV,CACV,CAAC;gBACJ,CAAC,CAAC,GACE,EACL,IAAI,CAAC,CAAC,CAAC,cAAK,SAAS,EAAE,MAAM,CAAC,IAAI,YAAG,IAAI,GAAO,CAAC,CAAC,CAAC,IAAI,IACpD,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,51 @@
1
+ .wrapper {
2
+ display: flex;
3
+ flex-direction: column;
4
+ gap: 0.5rem;
5
+ }
6
+
7
+ .label {
8
+ color: var(--text-primary);
9
+ font-size: 0.875rem;
10
+ font-weight: 500;
11
+ }
12
+
13
+ .hint {
14
+ color: var(--text-secondary);
15
+ font-size: 0.75rem;
16
+ }
17
+
18
+ .track {
19
+ display: inline-flex;
20
+ gap: 4px;
21
+ padding: 4px;
22
+ border: 1px solid var(--border-color);
23
+ background: color-mix(in srgb, var(--bg-secondary) 85%, transparent);
24
+ border-radius: 999px;
25
+ }
26
+
27
+ .trackCompact {
28
+ padding: 2px;
29
+ }
30
+
31
+ .option {
32
+ border: none;
33
+ background: transparent;
34
+ color: var(--text-secondary);
35
+ padding: 0.5rem 0.875rem;
36
+ border-radius: 999px;
37
+ font-family: var(--font-mono);
38
+ font-size: 0.8125rem;
39
+ cursor: pointer;
40
+ transition: color var(--transition-fast), background-color var(--transition-fast);
41
+ }
42
+
43
+ .optionCompact {
44
+ padding: 0.35rem 0.7rem;
45
+ }
46
+
47
+ .optionActive {
48
+ background: var(--interactive-primary-bg);
49
+ color: var(--interactive-primary-fg);
50
+ }
51
+
@@ -0,0 +1,6 @@
1
+ export interface StatusMessageProps {
2
+ message: string;
3
+ type: "success" | "error" | "info";
4
+ }
5
+ export declare function StatusMessage({ message, type }: StatusMessageProps): import("react/jsx-runtime").JSX.Element;
6
+ //# sourceMappingURL=StatusMessage.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"StatusMessage.d.ts","sourceRoot":"","sources":["../../src/components/StatusMessage.tsx"],"names":[],"mappings":"AAMA,MAAM,WAAW,kBAAkB;IACjC,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,SAAS,GAAG,OAAO,GAAG,MAAM,CAAC;CACpC;AAED,wBAAgB,aAAa,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,EAAE,kBAAkB,2CAelE"}
@@ -0,0 +1,7 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { CheckIcon, ErrorIcon, InfoIcon } from "@steez-ui/icons";
3
+ import styles from "./StatusMessage.module.css";
4
+ export function StatusMessage({ message, type }) {
5
+ return (_jsxs("div", { className: `${styles.message} ${styles[type]}`.trim(), role: type === "error" ? "alert" : "status", "aria-live": type === "error" ? "assertive" : "polite", children: [_jsxs("span", { className: styles.icon, children: [type === "success" ? _jsx(CheckIcon, { width: 16, height: 16 }) : null, type === "error" ? _jsx(ErrorIcon, { width: 16, height: 16 }) : null, type === "info" ? _jsx(InfoIcon, { width: 16, height: 16 }) : null] }), _jsx("span", { className: styles.text, children: message })] }));
6
+ }
7
+ //# sourceMappingURL=StatusMessage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"StatusMessage.js","sourceRoot":"","sources":["../../src/components/StatusMessage.tsx"],"names":[],"mappings":";AAEA,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAEjE,OAAO,MAAM,MAAM,4BAA4B,CAAC;AAOhD,MAAM,UAAU,aAAa,CAAC,EAAE,OAAO,EAAE,IAAI,EAAsB;IACjE,OAAO,CACL,eACE,SAAS,EAAE,GAAG,MAAM,CAAC,OAAO,IAAI,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,EAAE,EACrD,IAAI,EAAE,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,eAChC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,QAAQ,aAEpD,gBAAM,SAAS,EAAE,MAAM,CAAC,IAAI,aACzB,IAAI,KAAK,SAAS,CAAC,CAAC,CAAC,KAAC,SAAS,IAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,CAAC,CAAC,CAAC,IAAI,EAChE,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,KAAC,SAAS,IAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,CAAC,CAAC,CAAC,IAAI,EAC9D,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,KAAC,QAAQ,IAAC,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,GAAI,CAAC,CAAC,CAAC,IAAI,IACxD,EACP,eAAM,SAAS,EAAE,MAAM,CAAC,IAAI,YAAG,OAAO,GAAQ,IAC1C,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,60 @@
1
+ .message {
2
+ position: fixed;
3
+ bottom: 2rem;
4
+ right: 2rem;
5
+ display: flex;
6
+ align-items: center;
7
+ gap: 0.75rem;
8
+ padding: 1rem 1.5rem;
9
+ background: var(--bg-secondary);
10
+ border: 1px solid var(--border-color);
11
+ border-radius: 6px;
12
+ box-shadow: 0 4px 12px rgba(1, 6, 7, 0.15);
13
+ font-size: 0.9rem;
14
+ animation: slideIn 0.3s ease-out;
15
+ z-index: 1000;
16
+ }
17
+
18
+ @keyframes slideIn {
19
+ from {
20
+ transform: translateX(100%);
21
+ opacity: 0;
22
+ }
23
+ to {
24
+ transform: translateX(0);
25
+ opacity: 1;
26
+ }
27
+ }
28
+
29
+ .icon {
30
+ display: inline-flex;
31
+ }
32
+
33
+ .text {
34
+ color: var(--text-primary);
35
+ }
36
+
37
+ .success {
38
+ border-color: var(--success);
39
+ }
40
+
41
+ .success .icon {
42
+ color: var(--success);
43
+ }
44
+
45
+ .error {
46
+ border-color: var(--danger);
47
+ }
48
+
49
+ .error .icon {
50
+ color: var(--danger);
51
+ }
52
+
53
+ .info {
54
+ border-color: var(--info);
55
+ }
56
+
57
+ .info .icon {
58
+ color: var(--info);
59
+ }
60
+
@@ -0,0 +1,23 @@
1
+ import React from "react";
2
+ export interface TabbedPanelTab {
3
+ id: string;
4
+ label: string;
5
+ content?: React.ReactNode;
6
+ panel?: React.ReactNode;
7
+ disabled?: boolean;
8
+ }
9
+ export interface TabbedPanelProps {
10
+ tabs: TabbedPanelTab[];
11
+ activeTab?: string;
12
+ defaultTab?: string;
13
+ onTabChange?: (tabId: string) => void;
14
+ onChange?: (tabId: string) => void;
15
+ label?: string;
16
+ hint?: string;
17
+ ariaLabel?: string;
18
+ className?: string;
19
+ navClassName?: string;
20
+ panelClassName?: string;
21
+ }
22
+ export declare function TabbedPanel({ tabs, activeTab, defaultTab, onTabChange, onChange, label, hint, ariaLabel, className, navClassName, panelClassName, }: TabbedPanelProps): import("react/jsx-runtime").JSX.Element;
23
+ //# sourceMappingURL=TabbedPanel.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabbedPanel.d.ts","sourceRoot":"","sources":["../../src/components/TabbedPanel.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAI1B,MAAM,WAAW,cAAc;IAC7B,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IAC1B,KAAK,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IACxB,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB;AAED,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,cAAc,EAAE,CAAC;IACvB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IACtC,QAAQ,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IACnC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB;AAED,wBAAgB,WAAW,CAAC,EAC1B,IAAI,EACJ,SAAS,EACT,UAAU,EACV,WAAW,EACX,QAAQ,EACR,KAAK,EACL,IAAI,EACJ,SAAS,EACT,SAAc,EACd,YAAiB,EACjB,cAAmB,GACpB,EAAE,gBAAgB,2CAkElB"}
@@ -0,0 +1,36 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import React from "react";
3
+ import styles from "./TabbedPanel.module.css";
4
+ export function TabbedPanel({ tabs, activeTab, defaultTab, onTabChange, onChange, label, hint, ariaLabel, className = "", navClassName = "", panelClassName = "", }) {
5
+ const isControlled = typeof activeTab === "string";
6
+ const initialTabId = defaultTab || activeTab || tabs[0]?.id || "";
7
+ const [internalTabId, setInternalTabId] = React.useState(initialTabId);
8
+ const currentTabId = isControlled ? activeTab || tabs[0]?.id || "" : internalTabId;
9
+ const currentTab = tabs.find((tab) => tab.id === currentTabId) ?? tabs[0];
10
+ React.useEffect(() => {
11
+ if (!tabs.length) {
12
+ return;
13
+ }
14
+ if (tabs.some((tab) => tab.id === currentTabId)) {
15
+ return;
16
+ }
17
+ const fallbackTabId = defaultTab || activeTab || tabs[0]?.id || "";
18
+ if (!isControlled) {
19
+ setInternalTabId(fallbackTabId);
20
+ }
21
+ onTabChange?.(fallbackTabId);
22
+ onChange?.(fallbackTabId);
23
+ }, [activeTab, currentTabId, defaultTab, isControlled, onChange, onTabChange, tabs]);
24
+ const handleSelect = React.useCallback((tabId) => {
25
+ if (!isControlled) {
26
+ setInternalTabId(tabId);
27
+ }
28
+ onTabChange?.(tabId);
29
+ onChange?.(tabId);
30
+ }, [isControlled, onChange, onTabChange]);
31
+ return (_jsxs("div", { className: `${styles.root} ${className}`.trim(), children: [label || hint ? (_jsxs("div", { className: styles.header, children: [label ? _jsx("div", { className: styles.label, children: label }) : null, hint ? _jsx("div", { className: styles.hint, children: hint }) : null] })) : null, _jsx("div", { className: `${styles.tabs} ${navClassName}`.trim(), role: "tablist", "aria-label": ariaLabel || label, children: tabs.map((tab) => {
32
+ const isActive = tab.id === currentTab?.id;
33
+ return (_jsx("button", { type: "button", role: "tab", "aria-selected": isActive, disabled: tab.disabled, className: `${styles.tab} ${isActive ? styles.tabActive : ""}`.trim(), onClick: () => handleSelect(tab.id), children: tab.label }, tab.id));
34
+ }) }), _jsx("div", { className: `${styles.panel} ${panelClassName}`.trim(), role: "tabpanel", children: _jsx("div", { className: styles.panelBody, children: currentTab?.content ?? currentTab?.panel ?? null }) })] }));
35
+ }
36
+ //# sourceMappingURL=TabbedPanel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabbedPanel.js","sourceRoot":"","sources":["../../src/components/TabbedPanel.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,61 @@
1
+ .root {
2
+ display: flex;
3
+ flex-direction: column;
4
+ gap: 1rem;
5
+ }
6
+
7
+ .header {
8
+ display: flex;
9
+ justify-content: space-between;
10
+ align-items: baseline;
11
+ gap: 1rem;
12
+ }
13
+
14
+ .label {
15
+ color: var(--text-primary);
16
+ font-size: 0.875rem;
17
+ font-family: var(--font-mono);
18
+ text-transform: uppercase;
19
+ letter-spacing: 0.08em;
20
+ }
21
+
22
+ .hint {
23
+ color: var(--text-secondary);
24
+ font-size: 0.8125rem;
25
+ }
26
+
27
+ .tabs {
28
+ display: flex;
29
+ flex-wrap: wrap;
30
+ gap: 0.5rem;
31
+ position: relative;
32
+ }
33
+
34
+ .tab {
35
+ border: 1px solid var(--border-color);
36
+ background: transparent;
37
+ color: var(--text-secondary);
38
+ padding: 0.5rem 0.875rem;
39
+ font-family: var(--font-mono);
40
+ font-size: 0.8125rem;
41
+ cursor: pointer;
42
+ transition: all var(--transition-fast);
43
+ }
44
+
45
+ .tabActive {
46
+ background: var(--interactive-primary-bg);
47
+ color: var(--interactive-primary-fg);
48
+ border-color: var(--interactive-primary-border);
49
+ }
50
+
51
+ .panel {
52
+ border: 1px solid var(--border-color);
53
+ background: color-mix(in srgb, var(--bg-secondary) 88%, transparent);
54
+ padding: 1rem;
55
+ }
56
+
57
+ .panelBody {
58
+ display: flex;
59
+ flex-direction: column;
60
+ gap: 1rem;
61
+ }
@@ -0,0 +1,8 @@
1
+ export interface ThemeToggleProps {
2
+ storageKey?: string;
3
+ defaultTheme?: "dark" | "light";
4
+ className?: string;
5
+ onThemeChange?: (theme: "dark" | "light") => void;
6
+ }
7
+ export declare function ThemeToggle({ storageKey, defaultTheme, className, onThemeChange, }: ThemeToggleProps): import("react/jsx-runtime").JSX.Element;
8
+ //# sourceMappingURL=ThemeToggle.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ThemeToggle.d.ts","sourceRoot":"","sources":["../../src/components/ThemeToggle.tsx"],"names":[],"mappings":"AAMA,MAAM,WAAW,gBAAgB;IAC/B,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC;IAChC,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,aAAa,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,KAAK,IAAI,CAAC;CACnD;AAED,wBAAgB,WAAW,CAAC,EAC1B,UAAoB,EACpB,YAAqB,EACrB,SAAc,EACd,aAAa,GACd,EAAE,gBAAgB,2CA6BlB"}