@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.
- package/LICENSE +21 -0
- package/README.md +52 -0
- package/dist/components/Button.d.ts +11 -0
- package/dist/components/Button.d.ts.map +1 -0
- package/dist/components/Button.js +8 -0
- package/dist/components/Button.js.map +1 -0
- package/dist/components/CopyButton.d.ts +10 -0
- package/dist/components/CopyButton.d.ts.map +1 -0
- package/dist/components/CopyButton.js +25 -0
- package/dist/components/CopyButton.js.map +1 -0
- package/dist/components/CopyButton.module.css +30 -0
- package/dist/components/CornerBracketCard.d.ts +7 -0
- package/dist/components/CornerBracketCard.d.ts.map +1 -0
- package/dist/components/CornerBracketCard.js +6 -0
- package/dist/components/CornerBracketCard.js.map +1 -0
- package/dist/components/CornerBracketCard.module.css +109 -0
- package/dist/components/CyberpunkCheckbox.d.ts +7 -0
- package/dist/components/CyberpunkCheckbox.d.ts.map +1 -0
- package/dist/components/CyberpunkCheckbox.js +8 -0
- package/dist/components/CyberpunkCheckbox.js.map +1 -0
- package/dist/components/CyberpunkCheckbox.module.css +57 -0
- package/dist/components/CyberpunkInput.d.ts +9 -0
- package/dist/components/CyberpunkInput.d.ts.map +1 -0
- package/dist/components/CyberpunkInput.js +9 -0
- package/dist/components/CyberpunkInput.js.map +1 -0
- package/dist/components/CyberpunkInput.module.css +115 -0
- package/dist/components/CyberpunkRadio.d.ts +17 -0
- package/dist/components/CyberpunkRadio.d.ts.map +1 -0
- package/dist/components/CyberpunkRadio.js +11 -0
- package/dist/components/CyberpunkRadio.js.map +1 -0
- package/dist/components/CyberpunkRadio.module.css +64 -0
- package/dist/components/CyberpunkSelect.d.ts +12 -0
- package/dist/components/CyberpunkSelect.d.ts.map +1 -0
- package/dist/components/CyberpunkSelect.js +8 -0
- package/dist/components/CyberpunkSelect.js.map +1 -0
- package/dist/components/CyberpunkSelect.module.css +108 -0
- package/dist/components/CyberpunkSlider.d.ts +10 -0
- package/dist/components/CyberpunkSlider.d.ts.map +1 -0
- package/dist/components/CyberpunkSlider.js +9 -0
- package/dist/components/CyberpunkSlider.js.map +1 -0
- package/dist/components/CyberpunkSlider.module.css +98 -0
- package/dist/components/CyberpunkTextarea.d.ts +7 -0
- package/dist/components/CyberpunkTextarea.d.ts.map +1 -0
- package/dist/components/CyberpunkTextarea.js +8 -0
- package/dist/components/CyberpunkTextarea.js.map +1 -0
- package/dist/components/CyberpunkTextarea.module.css +73 -0
- package/dist/components/CyberpunkTile.d.ts +10 -0
- package/dist/components/CyberpunkTile.d.ts.map +1 -0
- package/dist/components/CyberpunkTile.js +6 -0
- package/dist/components/CyberpunkTile.js.map +1 -0
- package/dist/components/CyberpunkTile.module.css +133 -0
- package/dist/components/ErrorMessage.d.ts +11 -0
- package/dist/components/ErrorMessage.d.ts.map +1 -0
- package/dist/components/ErrorMessage.js +16 -0
- package/dist/components/ErrorMessage.js.map +1 -0
- package/dist/components/ErrorMessage.module.css +75 -0
- package/dist/components/LoadingProgressBar.d.ts +8 -0
- package/dist/components/LoadingProgressBar.d.ts.map +1 -0
- package/dist/components/LoadingProgressBar.js +13 -0
- package/dist/components/LoadingProgressBar.js.map +1 -0
- package/dist/components/LoadingProgressBar.module.css +42 -0
- package/dist/components/PageHeader.d.ts +17 -0
- package/dist/components/PageHeader.d.ts.map +1 -0
- package/dist/components/PageHeader.js +7 -0
- package/dist/components/PageHeader.js.map +1 -0
- package/dist/components/PageHeader.module.css +118 -0
- package/dist/components/PageTemplate.d.ts +20 -0
- package/dist/components/PageTemplate.d.ts.map +1 -0
- package/dist/components/PageTemplate.js +8 -0
- package/dist/components/PageTemplate.js.map +1 -0
- package/dist/components/PageTemplate.module.css +55 -0
- package/dist/components/SegmentedControl.d.ts +16 -0
- package/dist/components/SegmentedControl.d.ts.map +1 -0
- package/dist/components/SegmentedControl.js +9 -0
- package/dist/components/SegmentedControl.js.map +1 -0
- package/dist/components/SegmentedControl.module.css +51 -0
- package/dist/components/StatusMessage.d.ts +6 -0
- package/dist/components/StatusMessage.d.ts.map +1 -0
- package/dist/components/StatusMessage.js +7 -0
- package/dist/components/StatusMessage.js.map +1 -0
- package/dist/components/StatusMessage.module.css +60 -0
- package/dist/components/TabbedPanel.d.ts +23 -0
- package/dist/components/TabbedPanel.d.ts.map +1 -0
- package/dist/components/TabbedPanel.js +36 -0
- package/dist/components/TabbedPanel.js.map +1 -0
- package/dist/components/TabbedPanel.module.css +61 -0
- package/dist/components/ThemeToggle.d.ts +8 -0
- package/dist/components/ThemeToggle.d.ts.map +1 -0
- package/dist/components/ThemeToggle.js +22 -0
- package/dist/components/ThemeToggle.js.map +1 -0
- package/dist/components/ThemeToggle.module.css +23 -0
- package/dist/components/ThemedCard.d.ts +7 -0
- package/dist/components/ThemedCard.d.ts.map +1 -0
- package/dist/components/ThemedCard.js +6 -0
- package/dist/components/ThemedCard.js.map +1 -0
- package/dist/components/ThemedCard.module.css +22 -0
- package/dist/hooks/useStableId.d.ts +2 -0
- package/dist/hooks/useStableId.d.ts.map +1 -0
- package/dist/hooks/useStableId.js +6 -0
- package/dist/hooks/useStableId.js.map +1 -0
- package/dist/index.d.ts +20 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +20 -0
- package/dist/index.js.map +1 -0
- package/dist/styles/Buttons.module.css +96 -0
- 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 @@
|
|
|
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"}
|