@camstack/ui-library 0.1.25

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 @@
1
+ {"version":3,"sources":["../../src/theme/index.ts","../../src/theme/defaults.ts","../../src/theme/create-theme.ts","../../src/theme/theme-to-css.ts","../../src/theme/theme-provider.tsx","../../src/theme/use-theme-mode.ts"],"sourcesContent":["export type {\n CamStackColorTokens,\n CamStackTheme,\n ThemeMode,\n DeepPartial,\n UseThemeModeReturn,\n} from './types'\n\nexport { darkColors, lightColors, defaultTheme } from './defaults'\nexport { createTheme } from './create-theme'\nexport { themeToCss } from './theme-to-css'\nexport { ThemeProvider } from './theme-provider'\nexport { useThemeMode } from './use-theme-mode'\n","import type { CamStackColorTokens, CamStackTheme } from './types'\n\nconst providerColors: CamStackColorTokens['provider'] = {\n frigate: '#3b82f6',\n scrypted: '#a855f7',\n reolink: '#06b6d4',\n homeAssistant: '#22d3ee',\n rtsp: '#78716c',\n}\n\nexport const darkColors: CamStackColorTokens = {\n primary: '#f59e42',\n primaryForeground: '#0c0a09',\n background: '#0c0a09',\n backgroundElevated: '#1c1917',\n surface: '#1c1917',\n surfaceHover: '#292524',\n border: '#292524',\n borderSubtle: '#1c1917',\n foreground: '#fafaf9',\n foregroundMuted: '#a8a29e',\n foregroundSubtle: '#78716c',\n foregroundDisabled: '#57534e',\n success: '#4ade80',\n warning: '#fbbf24',\n danger: '#f87171',\n info: '#60a5fa',\n provider: providerColors,\n}\n\nexport const lightColors: CamStackColorTokens = {\n primary: '#e67e22',\n primaryForeground: '#ffffff',\n background: '#fafaf9',\n backgroundElevated: '#ffffff',\n surface: '#f5f5f4',\n surfaceHover: '#e7e5e4',\n border: '#d6d3d1',\n borderSubtle: '#e7e5e4',\n foreground: '#1c1917',\n foregroundMuted: '#57534e',\n foregroundSubtle: '#78716c',\n foregroundDisabled: '#a8a29e',\n success: '#16a34a',\n warning: '#d97706',\n danger: '#dc2626',\n info: '#2563eb',\n provider: providerColors,\n}\n\nexport const defaultTheme: CamStackTheme = {\n colors: {\n dark: darkColors,\n light: lightColors,\n },\n spacing: {\n xs: 2,\n sm: 4,\n md: 8,\n lg: 12,\n xl: 16,\n '2xl': 24,\n '3xl': 32,\n },\n radius: {\n sm: 4,\n md: 6,\n lg: 8,\n xl: 12,\n },\n typography: {\n fontFamily: '-apple-system, BlinkMacSystemFont, \"Segoe UI\", sans-serif',\n sizes: {\n xs: { fontSize: 10, lineHeight: 14 },\n sm: { fontSize: 11, lineHeight: 16 },\n base: { fontSize: 12, lineHeight: 18 },\n lg: { fontSize: 13, lineHeight: 18 },\n xl: { fontSize: 14, lineHeight: 20 },\n '2xl': { fontSize: 16, lineHeight: 22 },\n '3xl': { fontSize: 20, lineHeight: 28 },\n },\n weights: {\n regular: 400,\n medium: 500,\n semibold: 600,\n bold: 700,\n },\n },\n table: {\n rowHeight: 28,\n headerHeight: 24,\n cellPaddingX: 8,\n cellPaddingY: 6,\n },\n sidebar: {\n width: 176,\n itemHeight: 28,\n iconSize: 14,\n },\n}\n","import type { CamStackTheme, DeepPartial } from './types'\nimport { defaultTheme } from './defaults'\n\nfunction deepMerge<T extends object>(target: T, source: DeepPartial<T>): T {\n const result = { ...target }\n for (const key in source) {\n const sourceVal = source[key]\n const targetVal = target[key]\n if (\n sourceVal !== undefined &&\n typeof sourceVal === 'object' &&\n sourceVal !== null &&\n !Array.isArray(sourceVal) &&\n typeof targetVal === 'object' &&\n targetVal !== null\n ) {\n result[key] = deepMerge(\n targetVal as Record<string, unknown>,\n sourceVal as Record<string, unknown>,\n ) as T[typeof key]\n } else if (sourceVal !== undefined) {\n result[key] = sourceVal as T[typeof key]\n }\n }\n return result\n}\n\nexport function createTheme(overrides?: DeepPartial<CamStackTheme>): CamStackTheme {\n if (!overrides) return structuredClone(defaultTheme)\n return deepMerge(structuredClone(defaultTheme), overrides)\n}\n","import type { CamStackColorTokens, CamStackTheme } from './types'\n\nfunction camelToKebab(str: string): string {\n return str.replace(/[A-Z]/g, (m) => `-${m.toLowerCase()}`)\n}\n\nfunction colorTokenToCssVar(key: string): string {\n return `--color-${camelToKebab(key)}`\n}\n\nfunction generateColorBlock(colors: CamStackColorTokens): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(colors)) {\n if (key === 'provider') continue\n lines.push(` ${colorTokenToCssVar(key)}: ${value};`)\n }\n return lines.join('\\n')\n}\n\nfunction generateProviderColors(provider: CamStackColorTokens['provider']): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(provider)) {\n lines.push(` --color-provider-${camelToKebab(key)}: ${value};`)\n }\n return lines.join('\\n')\n}\n\nfunction generateSpacingTokens(spacing: CamStackTheme['spacing']): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(spacing)) {\n lines.push(` --spacing-${key}: ${value}px;`)\n }\n return lines.join('\\n')\n}\n\nfunction generateRadiusTokens(radius: CamStackTheme['radius']): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(radius)) {\n lines.push(` --radius-${key}: ${value}px;`)\n }\n return lines.join('\\n')\n}\n\nexport function themeToCss(theme: CamStackTheme): string {\n const darkColorBlock = generateColorBlock(theme.colors.dark)\n const lightColorBlock = generateColorBlock(theme.colors.light)\n const providerBlock = generateProviderColors(theme.colors.dark.provider)\n const spacingBlock = generateSpacingTokens(theme.spacing)\n const radiusBlock = generateRadiusTokens(theme.radius)\n\n return `@theme {\n${providerBlock}\n${spacingBlock}\n${radiusBlock}\n}\n\n.dark {\n${darkColorBlock}\n}\n\n.light {\n${lightColorBlock}\n}\n\n@media (prefers-color-scheme: dark) {\n :root {\n${darkColorBlock.replace(/^ /gm, ' ')}\n }\n}\n\n@media (prefers-color-scheme: light) {\n :root {\n${lightColorBlock.replace(/^ /gm, ' ')}\n }\n}\n`\n}\n","import { createContext, useCallback, useEffect, useMemo, useState } from 'react'\nimport type { ReactNode } from 'react'\nimport type { ThemeMode, UseThemeModeReturn } from './types'\n\nexport const ThemeContext = createContext<UseThemeModeReturn | null>(null)\n\ninterface ThemeProviderProps {\n children: ReactNode\n defaultMode?: ThemeMode\n storageKey?: string\n}\n\nconst TOGGLE_ORDER: readonly ThemeMode[] = ['dark', 'light', 'system'] as const\n\nfunction getSystemPreference(): 'dark' | 'light' {\n if (typeof window === 'undefined') return 'dark'\n return window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light'\n}\n\nfunction getInitialMode(storageKey: string, defaultMode: ThemeMode): ThemeMode {\n if (typeof window === 'undefined') return defaultMode\n const stored = localStorage.getItem(storageKey)\n if (stored === 'dark' || stored === 'light' || stored === 'system') {\n return stored\n }\n return defaultMode\n}\n\nfunction resolveMode(mode: ThemeMode): 'dark' | 'light' {\n if (mode === 'system') return getSystemPreference()\n return mode\n}\n\nexport function ThemeProvider({\n children,\n defaultMode = 'system',\n storageKey = 'camstack-theme-mode',\n}: ThemeProviderProps) {\n const [mode, setModeState] = useState<ThemeMode>(() => getInitialMode(storageKey, defaultMode))\n const [resolvedMode, setResolvedMode] = useState<'dark' | 'light'>(() => resolveMode(mode))\n\n const setMode = useCallback(\n (newMode: ThemeMode) => {\n setModeState(newMode)\n setResolvedMode(resolveMode(newMode))\n if (typeof window !== 'undefined') {\n localStorage.setItem(storageKey, newMode)\n }\n },\n [storageKey],\n )\n\n const toggleMode = useCallback(() => {\n const currentIndex = TOGGLE_ORDER.indexOf(mode)\n const nextIndex = (currentIndex + 1) % TOGGLE_ORDER.length\n setMode(TOGGLE_ORDER[nextIndex] ?? 'dark')\n }, [mode, setMode])\n\n // Apply CSS class on document.documentElement\n useEffect(() => {\n if (typeof document === 'undefined') return\n\n const root = document.documentElement\n root.classList.remove('dark', 'light')\n // Always apply the resolved class — even in 'system' mode, the CSS\n // variables need the .dark or .light class to activate\n root.classList.add(resolvedMode)\n }, [mode, resolvedMode])\n\n // Listen for system theme changes when in 'system' mode\n useEffect(() => {\n if (typeof window === 'undefined' || mode !== 'system') return\n\n const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)')\n const handleChange = () => {\n setResolvedMode(getSystemPreference())\n }\n\n mediaQuery.addEventListener('change', handleChange)\n return () => mediaQuery.removeEventListener('change', handleChange)\n }, [mode])\n\n const value = useMemo<UseThemeModeReturn>(\n () => ({ mode, resolvedMode, setMode, toggleMode }),\n [mode, resolvedMode, setMode, toggleMode],\n )\n\n return <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>\n}\n","import { useContext } from 'react'\nimport { ThemeContext } from './theme-provider'\nimport type { UseThemeModeReturn } from './types'\n\nexport function useThemeMode(): UseThemeModeReturn {\n const context = useContext(ThemeContext)\n if (!context) {\n throw new Error('useThemeMode must be used within a ThemeProvider')\n }\n return context\n}\n"],"mappings":"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","names":["import_react"]}
@@ -0,0 +1,128 @@
1
+ import * as react_jsx_runtime from 'react/jsx-runtime';
2
+ import { ReactNode } from 'react';
3
+
4
+ interface CamStackColorTokens {
5
+ primary: string;
6
+ primaryForeground: string;
7
+ background: string;
8
+ backgroundElevated: string;
9
+ surface: string;
10
+ surfaceHover: string;
11
+ border: string;
12
+ borderSubtle: string;
13
+ foreground: string;
14
+ foregroundMuted: string;
15
+ foregroundSubtle: string;
16
+ foregroundDisabled: string;
17
+ success: string;
18
+ warning: string;
19
+ danger: string;
20
+ info: string;
21
+ provider: {
22
+ frigate: string;
23
+ scrypted: string;
24
+ reolink: string;
25
+ homeAssistant: string;
26
+ rtsp: string;
27
+ };
28
+ }
29
+ interface CamStackTheme {
30
+ colors: {
31
+ dark: CamStackColorTokens;
32
+ light: CamStackColorTokens;
33
+ };
34
+ spacing: {
35
+ xs: number;
36
+ sm: number;
37
+ md: number;
38
+ lg: number;
39
+ xl: number;
40
+ '2xl': number;
41
+ '3xl': number;
42
+ };
43
+ radius: {
44
+ sm: number;
45
+ md: number;
46
+ lg: number;
47
+ xl: number;
48
+ };
49
+ typography: {
50
+ fontFamily: string;
51
+ sizes: {
52
+ xs: {
53
+ fontSize: number;
54
+ lineHeight: number;
55
+ };
56
+ sm: {
57
+ fontSize: number;
58
+ lineHeight: number;
59
+ };
60
+ base: {
61
+ fontSize: number;
62
+ lineHeight: number;
63
+ };
64
+ lg: {
65
+ fontSize: number;
66
+ lineHeight: number;
67
+ };
68
+ xl: {
69
+ fontSize: number;
70
+ lineHeight: number;
71
+ };
72
+ '2xl': {
73
+ fontSize: number;
74
+ lineHeight: number;
75
+ };
76
+ '3xl': {
77
+ fontSize: number;
78
+ lineHeight: number;
79
+ };
80
+ };
81
+ weights: {
82
+ regular: number;
83
+ medium: number;
84
+ semibold: number;
85
+ bold: number;
86
+ };
87
+ };
88
+ table: {
89
+ rowHeight: number;
90
+ headerHeight: number;
91
+ cellPaddingX: number;
92
+ cellPaddingY: number;
93
+ };
94
+ sidebar: {
95
+ width: number;
96
+ itemHeight: number;
97
+ iconSize: number;
98
+ };
99
+ }
100
+ type ThemeMode = 'dark' | 'light' | 'system';
101
+ type DeepPartial<T> = {
102
+ [P in keyof T]?: T[P] extends object ? DeepPartial<T[P]> : T[P];
103
+ };
104
+ interface UseThemeModeReturn {
105
+ mode: ThemeMode;
106
+ resolvedMode: 'dark' | 'light';
107
+ setMode: (mode: ThemeMode) => void;
108
+ toggleMode: () => void;
109
+ }
110
+
111
+ declare const darkColors: CamStackColorTokens;
112
+ declare const lightColors: CamStackColorTokens;
113
+ declare const defaultTheme: CamStackTheme;
114
+
115
+ declare function createTheme(overrides?: DeepPartial<CamStackTheme>): CamStackTheme;
116
+
117
+ declare function themeToCss(theme: CamStackTheme): string;
118
+
119
+ interface ThemeProviderProps {
120
+ children: ReactNode;
121
+ defaultMode?: ThemeMode;
122
+ storageKey?: string;
123
+ }
124
+ declare function ThemeProvider({ children, defaultMode, storageKey, }: ThemeProviderProps): react_jsx_runtime.JSX.Element;
125
+
126
+ declare function useThemeMode(): UseThemeModeReturn;
127
+
128
+ export { type CamStackColorTokens, type CamStackTheme, type DeepPartial, type ThemeMode, ThemeProvider, type UseThemeModeReturn, createTheme, darkColors, defaultTheme, lightColors, themeToCss, useThemeMode };
@@ -0,0 +1,128 @@
1
+ import * as react_jsx_runtime from 'react/jsx-runtime';
2
+ import { ReactNode } from 'react';
3
+
4
+ interface CamStackColorTokens {
5
+ primary: string;
6
+ primaryForeground: string;
7
+ background: string;
8
+ backgroundElevated: string;
9
+ surface: string;
10
+ surfaceHover: string;
11
+ border: string;
12
+ borderSubtle: string;
13
+ foreground: string;
14
+ foregroundMuted: string;
15
+ foregroundSubtle: string;
16
+ foregroundDisabled: string;
17
+ success: string;
18
+ warning: string;
19
+ danger: string;
20
+ info: string;
21
+ provider: {
22
+ frigate: string;
23
+ scrypted: string;
24
+ reolink: string;
25
+ homeAssistant: string;
26
+ rtsp: string;
27
+ };
28
+ }
29
+ interface CamStackTheme {
30
+ colors: {
31
+ dark: CamStackColorTokens;
32
+ light: CamStackColorTokens;
33
+ };
34
+ spacing: {
35
+ xs: number;
36
+ sm: number;
37
+ md: number;
38
+ lg: number;
39
+ xl: number;
40
+ '2xl': number;
41
+ '3xl': number;
42
+ };
43
+ radius: {
44
+ sm: number;
45
+ md: number;
46
+ lg: number;
47
+ xl: number;
48
+ };
49
+ typography: {
50
+ fontFamily: string;
51
+ sizes: {
52
+ xs: {
53
+ fontSize: number;
54
+ lineHeight: number;
55
+ };
56
+ sm: {
57
+ fontSize: number;
58
+ lineHeight: number;
59
+ };
60
+ base: {
61
+ fontSize: number;
62
+ lineHeight: number;
63
+ };
64
+ lg: {
65
+ fontSize: number;
66
+ lineHeight: number;
67
+ };
68
+ xl: {
69
+ fontSize: number;
70
+ lineHeight: number;
71
+ };
72
+ '2xl': {
73
+ fontSize: number;
74
+ lineHeight: number;
75
+ };
76
+ '3xl': {
77
+ fontSize: number;
78
+ lineHeight: number;
79
+ };
80
+ };
81
+ weights: {
82
+ regular: number;
83
+ medium: number;
84
+ semibold: number;
85
+ bold: number;
86
+ };
87
+ };
88
+ table: {
89
+ rowHeight: number;
90
+ headerHeight: number;
91
+ cellPaddingX: number;
92
+ cellPaddingY: number;
93
+ };
94
+ sidebar: {
95
+ width: number;
96
+ itemHeight: number;
97
+ iconSize: number;
98
+ };
99
+ }
100
+ type ThemeMode = 'dark' | 'light' | 'system';
101
+ type DeepPartial<T> = {
102
+ [P in keyof T]?: T[P] extends object ? DeepPartial<T[P]> : T[P];
103
+ };
104
+ interface UseThemeModeReturn {
105
+ mode: ThemeMode;
106
+ resolvedMode: 'dark' | 'light';
107
+ setMode: (mode: ThemeMode) => void;
108
+ toggleMode: () => void;
109
+ }
110
+
111
+ declare const darkColors: CamStackColorTokens;
112
+ declare const lightColors: CamStackColorTokens;
113
+ declare const defaultTheme: CamStackTheme;
114
+
115
+ declare function createTheme(overrides?: DeepPartial<CamStackTheme>): CamStackTheme;
116
+
117
+ declare function themeToCss(theme: CamStackTheme): string;
118
+
119
+ interface ThemeProviderProps {
120
+ children: ReactNode;
121
+ defaultMode?: ThemeMode;
122
+ storageKey?: string;
123
+ }
124
+ declare function ThemeProvider({ children, defaultMode, storageKey, }: ThemeProviderProps): react_jsx_runtime.JSX.Element;
125
+
126
+ declare function useThemeMode(): UseThemeModeReturn;
127
+
128
+ export { type CamStackColorTokens, type CamStackTheme, type DeepPartial, type ThemeMode, ThemeProvider, type UseThemeModeReturn, createTheme, darkColors, defaultTheme, lightColors, themeToCss, useThemeMode };
@@ -0,0 +1,273 @@
1
+ // src/theme/defaults.ts
2
+ var providerColors = {
3
+ frigate: "#3b82f6",
4
+ scrypted: "#a855f7",
5
+ reolink: "#06b6d4",
6
+ homeAssistant: "#22d3ee",
7
+ rtsp: "#78716c"
8
+ };
9
+ var darkColors = {
10
+ primary: "#f59e42",
11
+ primaryForeground: "#0c0a09",
12
+ background: "#0c0a09",
13
+ backgroundElevated: "#1c1917",
14
+ surface: "#1c1917",
15
+ surfaceHover: "#292524",
16
+ border: "#292524",
17
+ borderSubtle: "#1c1917",
18
+ foreground: "#fafaf9",
19
+ foregroundMuted: "#a8a29e",
20
+ foregroundSubtle: "#78716c",
21
+ foregroundDisabled: "#57534e",
22
+ success: "#4ade80",
23
+ warning: "#fbbf24",
24
+ danger: "#f87171",
25
+ info: "#60a5fa",
26
+ provider: providerColors
27
+ };
28
+ var lightColors = {
29
+ primary: "#e67e22",
30
+ primaryForeground: "#ffffff",
31
+ background: "#fafaf9",
32
+ backgroundElevated: "#ffffff",
33
+ surface: "#f5f5f4",
34
+ surfaceHover: "#e7e5e4",
35
+ border: "#d6d3d1",
36
+ borderSubtle: "#e7e5e4",
37
+ foreground: "#1c1917",
38
+ foregroundMuted: "#57534e",
39
+ foregroundSubtle: "#78716c",
40
+ foregroundDisabled: "#a8a29e",
41
+ success: "#16a34a",
42
+ warning: "#d97706",
43
+ danger: "#dc2626",
44
+ info: "#2563eb",
45
+ provider: providerColors
46
+ };
47
+ var defaultTheme = {
48
+ colors: {
49
+ dark: darkColors,
50
+ light: lightColors
51
+ },
52
+ spacing: {
53
+ xs: 2,
54
+ sm: 4,
55
+ md: 8,
56
+ lg: 12,
57
+ xl: 16,
58
+ "2xl": 24,
59
+ "3xl": 32
60
+ },
61
+ radius: {
62
+ sm: 4,
63
+ md: 6,
64
+ lg: 8,
65
+ xl: 12
66
+ },
67
+ typography: {
68
+ fontFamily: '-apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif',
69
+ sizes: {
70
+ xs: { fontSize: 10, lineHeight: 14 },
71
+ sm: { fontSize: 11, lineHeight: 16 },
72
+ base: { fontSize: 12, lineHeight: 18 },
73
+ lg: { fontSize: 13, lineHeight: 18 },
74
+ xl: { fontSize: 14, lineHeight: 20 },
75
+ "2xl": { fontSize: 16, lineHeight: 22 },
76
+ "3xl": { fontSize: 20, lineHeight: 28 }
77
+ },
78
+ weights: {
79
+ regular: 400,
80
+ medium: 500,
81
+ semibold: 600,
82
+ bold: 700
83
+ }
84
+ },
85
+ table: {
86
+ rowHeight: 28,
87
+ headerHeight: 24,
88
+ cellPaddingX: 8,
89
+ cellPaddingY: 6
90
+ },
91
+ sidebar: {
92
+ width: 176,
93
+ itemHeight: 28,
94
+ iconSize: 14
95
+ }
96
+ };
97
+
98
+ // src/theme/create-theme.ts
99
+ function deepMerge(target, source) {
100
+ const result = { ...target };
101
+ for (const key in source) {
102
+ const sourceVal = source[key];
103
+ const targetVal = target[key];
104
+ if (sourceVal !== void 0 && typeof sourceVal === "object" && sourceVal !== null && !Array.isArray(sourceVal) && typeof targetVal === "object" && targetVal !== null) {
105
+ result[key] = deepMerge(
106
+ targetVal,
107
+ sourceVal
108
+ );
109
+ } else if (sourceVal !== void 0) {
110
+ result[key] = sourceVal;
111
+ }
112
+ }
113
+ return result;
114
+ }
115
+ function createTheme(overrides) {
116
+ if (!overrides) return structuredClone(defaultTheme);
117
+ return deepMerge(structuredClone(defaultTheme), overrides);
118
+ }
119
+
120
+ // src/theme/theme-to-css.ts
121
+ function camelToKebab(str) {
122
+ return str.replace(/[A-Z]/g, (m) => `-${m.toLowerCase()}`);
123
+ }
124
+ function colorTokenToCssVar(key) {
125
+ return `--color-${camelToKebab(key)}`;
126
+ }
127
+ function generateColorBlock(colors) {
128
+ const lines = [];
129
+ for (const [key, value] of Object.entries(colors)) {
130
+ if (key === "provider") continue;
131
+ lines.push(` ${colorTokenToCssVar(key)}: ${value};`);
132
+ }
133
+ return lines.join("\n");
134
+ }
135
+ function generateProviderColors(provider) {
136
+ const lines = [];
137
+ for (const [key, value] of Object.entries(provider)) {
138
+ lines.push(` --color-provider-${camelToKebab(key)}: ${value};`);
139
+ }
140
+ return lines.join("\n");
141
+ }
142
+ function generateSpacingTokens(spacing) {
143
+ const lines = [];
144
+ for (const [key, value] of Object.entries(spacing)) {
145
+ lines.push(` --spacing-${key}: ${value}px;`);
146
+ }
147
+ return lines.join("\n");
148
+ }
149
+ function generateRadiusTokens(radius) {
150
+ const lines = [];
151
+ for (const [key, value] of Object.entries(radius)) {
152
+ lines.push(` --radius-${key}: ${value}px;`);
153
+ }
154
+ return lines.join("\n");
155
+ }
156
+ function themeToCss(theme) {
157
+ const darkColorBlock = generateColorBlock(theme.colors.dark);
158
+ const lightColorBlock = generateColorBlock(theme.colors.light);
159
+ const providerBlock = generateProviderColors(theme.colors.dark.provider);
160
+ const spacingBlock = generateSpacingTokens(theme.spacing);
161
+ const radiusBlock = generateRadiusTokens(theme.radius);
162
+ return `@theme {
163
+ ${providerBlock}
164
+ ${spacingBlock}
165
+ ${radiusBlock}
166
+ }
167
+
168
+ .dark {
169
+ ${darkColorBlock}
170
+ }
171
+
172
+ .light {
173
+ ${lightColorBlock}
174
+ }
175
+
176
+ @media (prefers-color-scheme: dark) {
177
+ :root {
178
+ ${darkColorBlock.replace(/^ /gm, " ")}
179
+ }
180
+ }
181
+
182
+ @media (prefers-color-scheme: light) {
183
+ :root {
184
+ ${lightColorBlock.replace(/^ /gm, " ")}
185
+ }
186
+ }
187
+ `;
188
+ }
189
+
190
+ // src/theme/theme-provider.tsx
191
+ import { createContext, useCallback, useEffect, useMemo, useState } from "react";
192
+ import { jsx } from "react/jsx-runtime";
193
+ var ThemeContext = createContext(null);
194
+ var TOGGLE_ORDER = ["dark", "light", "system"];
195
+ function getSystemPreference() {
196
+ if (typeof window === "undefined") return "dark";
197
+ return window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light";
198
+ }
199
+ function getInitialMode(storageKey, defaultMode) {
200
+ if (typeof window === "undefined") return defaultMode;
201
+ const stored = localStorage.getItem(storageKey);
202
+ if (stored === "dark" || stored === "light" || stored === "system") {
203
+ return stored;
204
+ }
205
+ return defaultMode;
206
+ }
207
+ function resolveMode(mode) {
208
+ if (mode === "system") return getSystemPreference();
209
+ return mode;
210
+ }
211
+ function ThemeProvider({
212
+ children,
213
+ defaultMode = "system",
214
+ storageKey = "camstack-theme-mode"
215
+ }) {
216
+ const [mode, setModeState] = useState(() => getInitialMode(storageKey, defaultMode));
217
+ const [resolvedMode, setResolvedMode] = useState(() => resolveMode(mode));
218
+ const setMode = useCallback(
219
+ (newMode) => {
220
+ setModeState(newMode);
221
+ setResolvedMode(resolveMode(newMode));
222
+ if (typeof window !== "undefined") {
223
+ localStorage.setItem(storageKey, newMode);
224
+ }
225
+ },
226
+ [storageKey]
227
+ );
228
+ const toggleMode = useCallback(() => {
229
+ const currentIndex = TOGGLE_ORDER.indexOf(mode);
230
+ const nextIndex = (currentIndex + 1) % TOGGLE_ORDER.length;
231
+ setMode(TOGGLE_ORDER[nextIndex] ?? "dark");
232
+ }, [mode, setMode]);
233
+ useEffect(() => {
234
+ if (typeof document === "undefined") return;
235
+ const root = document.documentElement;
236
+ root.classList.remove("dark", "light");
237
+ root.classList.add(resolvedMode);
238
+ }, [mode, resolvedMode]);
239
+ useEffect(() => {
240
+ if (typeof window === "undefined" || mode !== "system") return;
241
+ const mediaQuery = window.matchMedia("(prefers-color-scheme: dark)");
242
+ const handleChange = () => {
243
+ setResolvedMode(getSystemPreference());
244
+ };
245
+ mediaQuery.addEventListener("change", handleChange);
246
+ return () => mediaQuery.removeEventListener("change", handleChange);
247
+ }, [mode]);
248
+ const value = useMemo(
249
+ () => ({ mode, resolvedMode, setMode, toggleMode }),
250
+ [mode, resolvedMode, setMode, toggleMode]
251
+ );
252
+ return /* @__PURE__ */ jsx(ThemeContext.Provider, { value, children });
253
+ }
254
+
255
+ // src/theme/use-theme-mode.ts
256
+ import { useContext } from "react";
257
+ function useThemeMode() {
258
+ const context = useContext(ThemeContext);
259
+ if (!context) {
260
+ throw new Error("useThemeMode must be used within a ThemeProvider");
261
+ }
262
+ return context;
263
+ }
264
+ export {
265
+ ThemeProvider,
266
+ createTheme,
267
+ darkColors,
268
+ defaultTheme,
269
+ lightColors,
270
+ themeToCss,
271
+ useThemeMode
272
+ };
273
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/theme/defaults.ts","../../src/theme/create-theme.ts","../../src/theme/theme-to-css.ts","../../src/theme/theme-provider.tsx","../../src/theme/use-theme-mode.ts"],"sourcesContent":["import type { CamStackColorTokens, CamStackTheme } from './types'\n\nconst providerColors: CamStackColorTokens['provider'] = {\n frigate: '#3b82f6',\n scrypted: '#a855f7',\n reolink: '#06b6d4',\n homeAssistant: '#22d3ee',\n rtsp: '#78716c',\n}\n\nexport const darkColors: CamStackColorTokens = {\n primary: '#f59e42',\n primaryForeground: '#0c0a09',\n background: '#0c0a09',\n backgroundElevated: '#1c1917',\n surface: '#1c1917',\n surfaceHover: '#292524',\n border: '#292524',\n borderSubtle: '#1c1917',\n foreground: '#fafaf9',\n foregroundMuted: '#a8a29e',\n foregroundSubtle: '#78716c',\n foregroundDisabled: '#57534e',\n success: '#4ade80',\n warning: '#fbbf24',\n danger: '#f87171',\n info: '#60a5fa',\n provider: providerColors,\n}\n\nexport const lightColors: CamStackColorTokens = {\n primary: '#e67e22',\n primaryForeground: '#ffffff',\n background: '#fafaf9',\n backgroundElevated: '#ffffff',\n surface: '#f5f5f4',\n surfaceHover: '#e7e5e4',\n border: '#d6d3d1',\n borderSubtle: '#e7e5e4',\n foreground: '#1c1917',\n foregroundMuted: '#57534e',\n foregroundSubtle: '#78716c',\n foregroundDisabled: '#a8a29e',\n success: '#16a34a',\n warning: '#d97706',\n danger: '#dc2626',\n info: '#2563eb',\n provider: providerColors,\n}\n\nexport const defaultTheme: CamStackTheme = {\n colors: {\n dark: darkColors,\n light: lightColors,\n },\n spacing: {\n xs: 2,\n sm: 4,\n md: 8,\n lg: 12,\n xl: 16,\n '2xl': 24,\n '3xl': 32,\n },\n radius: {\n sm: 4,\n md: 6,\n lg: 8,\n xl: 12,\n },\n typography: {\n fontFamily: '-apple-system, BlinkMacSystemFont, \"Segoe UI\", sans-serif',\n sizes: {\n xs: { fontSize: 10, lineHeight: 14 },\n sm: { fontSize: 11, lineHeight: 16 },\n base: { fontSize: 12, lineHeight: 18 },\n lg: { fontSize: 13, lineHeight: 18 },\n xl: { fontSize: 14, lineHeight: 20 },\n '2xl': { fontSize: 16, lineHeight: 22 },\n '3xl': { fontSize: 20, lineHeight: 28 },\n },\n weights: {\n regular: 400,\n medium: 500,\n semibold: 600,\n bold: 700,\n },\n },\n table: {\n rowHeight: 28,\n headerHeight: 24,\n cellPaddingX: 8,\n cellPaddingY: 6,\n },\n sidebar: {\n width: 176,\n itemHeight: 28,\n iconSize: 14,\n },\n}\n","import type { CamStackTheme, DeepPartial } from './types'\nimport { defaultTheme } from './defaults'\n\nfunction deepMerge<T extends object>(target: T, source: DeepPartial<T>): T {\n const result = { ...target }\n for (const key in source) {\n const sourceVal = source[key]\n const targetVal = target[key]\n if (\n sourceVal !== undefined &&\n typeof sourceVal === 'object' &&\n sourceVal !== null &&\n !Array.isArray(sourceVal) &&\n typeof targetVal === 'object' &&\n targetVal !== null\n ) {\n result[key] = deepMerge(\n targetVal as Record<string, unknown>,\n sourceVal as Record<string, unknown>,\n ) as T[typeof key]\n } else if (sourceVal !== undefined) {\n result[key] = sourceVal as T[typeof key]\n }\n }\n return result\n}\n\nexport function createTheme(overrides?: DeepPartial<CamStackTheme>): CamStackTheme {\n if (!overrides) return structuredClone(defaultTheme)\n return deepMerge(structuredClone(defaultTheme), overrides)\n}\n","import type { CamStackColorTokens, CamStackTheme } from './types'\n\nfunction camelToKebab(str: string): string {\n return str.replace(/[A-Z]/g, (m) => `-${m.toLowerCase()}`)\n}\n\nfunction colorTokenToCssVar(key: string): string {\n return `--color-${camelToKebab(key)}`\n}\n\nfunction generateColorBlock(colors: CamStackColorTokens): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(colors)) {\n if (key === 'provider') continue\n lines.push(` ${colorTokenToCssVar(key)}: ${value};`)\n }\n return lines.join('\\n')\n}\n\nfunction generateProviderColors(provider: CamStackColorTokens['provider']): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(provider)) {\n lines.push(` --color-provider-${camelToKebab(key)}: ${value};`)\n }\n return lines.join('\\n')\n}\n\nfunction generateSpacingTokens(spacing: CamStackTheme['spacing']): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(spacing)) {\n lines.push(` --spacing-${key}: ${value}px;`)\n }\n return lines.join('\\n')\n}\n\nfunction generateRadiusTokens(radius: CamStackTheme['radius']): string {\n const lines: string[] = []\n for (const [key, value] of Object.entries(radius)) {\n lines.push(` --radius-${key}: ${value}px;`)\n }\n return lines.join('\\n')\n}\n\nexport function themeToCss(theme: CamStackTheme): string {\n const darkColorBlock = generateColorBlock(theme.colors.dark)\n const lightColorBlock = generateColorBlock(theme.colors.light)\n const providerBlock = generateProviderColors(theme.colors.dark.provider)\n const spacingBlock = generateSpacingTokens(theme.spacing)\n const radiusBlock = generateRadiusTokens(theme.radius)\n\n return `@theme {\n${providerBlock}\n${spacingBlock}\n${radiusBlock}\n}\n\n.dark {\n${darkColorBlock}\n}\n\n.light {\n${lightColorBlock}\n}\n\n@media (prefers-color-scheme: dark) {\n :root {\n${darkColorBlock.replace(/^ /gm, ' ')}\n }\n}\n\n@media (prefers-color-scheme: light) {\n :root {\n${lightColorBlock.replace(/^ /gm, ' ')}\n }\n}\n`\n}\n","import { createContext, useCallback, useEffect, useMemo, useState } from 'react'\nimport type { ReactNode } from 'react'\nimport type { ThemeMode, UseThemeModeReturn } from './types'\n\nexport const ThemeContext = createContext<UseThemeModeReturn | null>(null)\n\ninterface ThemeProviderProps {\n children: ReactNode\n defaultMode?: ThemeMode\n storageKey?: string\n}\n\nconst TOGGLE_ORDER: readonly ThemeMode[] = ['dark', 'light', 'system'] as const\n\nfunction getSystemPreference(): 'dark' | 'light' {\n if (typeof window === 'undefined') return 'dark'\n return window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light'\n}\n\nfunction getInitialMode(storageKey: string, defaultMode: ThemeMode): ThemeMode {\n if (typeof window === 'undefined') return defaultMode\n const stored = localStorage.getItem(storageKey)\n if (stored === 'dark' || stored === 'light' || stored === 'system') {\n return stored\n }\n return defaultMode\n}\n\nfunction resolveMode(mode: ThemeMode): 'dark' | 'light' {\n if (mode === 'system') return getSystemPreference()\n return mode\n}\n\nexport function ThemeProvider({\n children,\n defaultMode = 'system',\n storageKey = 'camstack-theme-mode',\n}: ThemeProviderProps) {\n const [mode, setModeState] = useState<ThemeMode>(() => getInitialMode(storageKey, defaultMode))\n const [resolvedMode, setResolvedMode] = useState<'dark' | 'light'>(() => resolveMode(mode))\n\n const setMode = useCallback(\n (newMode: ThemeMode) => {\n setModeState(newMode)\n setResolvedMode(resolveMode(newMode))\n if (typeof window !== 'undefined') {\n localStorage.setItem(storageKey, newMode)\n }\n },\n [storageKey],\n )\n\n const toggleMode = useCallback(() => {\n const currentIndex = TOGGLE_ORDER.indexOf(mode)\n const nextIndex = (currentIndex + 1) % TOGGLE_ORDER.length\n setMode(TOGGLE_ORDER[nextIndex] ?? 'dark')\n }, [mode, setMode])\n\n // Apply CSS class on document.documentElement\n useEffect(() => {\n if (typeof document === 'undefined') return\n\n const root = document.documentElement\n root.classList.remove('dark', 'light')\n // Always apply the resolved class — even in 'system' mode, the CSS\n // variables need the .dark or .light class to activate\n root.classList.add(resolvedMode)\n }, [mode, resolvedMode])\n\n // Listen for system theme changes when in 'system' mode\n useEffect(() => {\n if (typeof window === 'undefined' || mode !== 'system') return\n\n const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)')\n const handleChange = () => {\n setResolvedMode(getSystemPreference())\n }\n\n mediaQuery.addEventListener('change', handleChange)\n return () => mediaQuery.removeEventListener('change', handleChange)\n }, [mode])\n\n const value = useMemo<UseThemeModeReturn>(\n () => ({ mode, resolvedMode, setMode, toggleMode }),\n [mode, resolvedMode, setMode, toggleMode],\n )\n\n return <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>\n}\n","import { useContext } from 'react'\nimport { ThemeContext } from './theme-provider'\nimport type { UseThemeModeReturn } from './types'\n\nexport function useThemeMode(): UseThemeModeReturn {\n const context = useContext(ThemeContext)\n if (!context) {\n throw new Error('useThemeMode must be used within a ThemeProvider')\n }\n return context\n}\n"],"mappings":"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","names":[]}
package/package.json ADDED
@@ -0,0 +1,58 @@
1
+ {
2
+ "name": "@camstack/ui-library",
3
+ "version": "0.1.25",
4
+ "type": "module",
5
+ "main": "dist/index.cjs",
6
+ "module": "dist/index.js",
7
+ "types": "dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "types": "./dist/index.d.ts",
11
+ "import": "./dist/index.js",
12
+ "require": "./dist/index.cjs"
13
+ },
14
+ "./theme": {
15
+ "types": "./dist/theme/index.d.ts",
16
+ "import": "./dist/theme/index.js",
17
+ "require": "./dist/theme/index.cjs"
18
+ },
19
+ "./tailwind/*": "./src/tailwind/*"
20
+ },
21
+ "files": [
22
+ "dist",
23
+ "src/tailwind",
24
+ "assets"
25
+ ],
26
+ "scripts": {
27
+ "build": "tsup",
28
+ "typecheck": "tsc --noEmit"
29
+ },
30
+ "peerDependencies": {
31
+ "@tanstack/react-table": "^8.21.3",
32
+ "react": ">=18",
33
+ "react-dom": ">=18",
34
+ "tailwindcss": ">=4.0"
35
+ },
36
+ "peerDependenciesMeta": {
37
+ "react-dom": {
38
+ "optional": true
39
+ },
40
+ "@tanstack/react-table": {
41
+ "optional": true
42
+ }
43
+ },
44
+ "dependencies": {
45
+ "class-variance-authority": "^0.7",
46
+ "clsx": "^2.1",
47
+ "lucide-react": "^0.576",
48
+ "tailwind-merge": "^3.0"
49
+ },
50
+ "devDependencies": {
51
+ "@testing-library/dom": "^10.4.1",
52
+ "@testing-library/jest-dom": "^6.9.1",
53
+ "@testing-library/react": "^16.3.2",
54
+ "jsdom": "^29.0.1",
55
+ "tsup": "^8.5.1",
56
+ "typescript": "^5.7.0"
57
+ }
58
+ }