@shiftbloom-studio/circadian-ui 0.2.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 +22 -0
- package/README.md +275 -0
- package/dist/index.cjs +890 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +201 -0
- package/dist/index.d.ts +201 -0
- package/dist/index.js +850 -0
- package/dist/index.js.map +1 -0
- package/dist/server.cjs +216 -0
- package/dist/server.cjs.map +1 -0
- package/dist/server.d.cts +73 -0
- package/dist/server.d.ts +73 -0
- package/dist/server.js +212 -0
- package/dist/server.js.map +1 -0
- package/package.json +87 -0
package/dist/server.js
ADDED
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
// src/core/schedule.ts
|
|
2
|
+
var defaultSchedule = {
|
|
3
|
+
dawn: { start: "05:30", end: "08:30" },
|
|
4
|
+
day: { start: "08:30", end: "17:30" },
|
|
5
|
+
dusk: { start: "17:30", end: "21:30" },
|
|
6
|
+
night: { start: "21:30", end: "05:30" }
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
// src/core/storage.ts
|
|
10
|
+
var defaultStorageKey = "cui:preferences";
|
|
11
|
+
|
|
12
|
+
// src/core/tokens.ts
|
|
13
|
+
var defaultTokens = {
|
|
14
|
+
dawn: {
|
|
15
|
+
bg: "27 60% 96%",
|
|
16
|
+
fg: "24 18% 18%",
|
|
17
|
+
muted: "27 40% 90%",
|
|
18
|
+
mutedFg: "24 14% 35%",
|
|
19
|
+
card: "0 0% 100%",
|
|
20
|
+
cardFg: "24 18% 18%",
|
|
21
|
+
border: "24 22% 84%",
|
|
22
|
+
ring: "20 65% 45%",
|
|
23
|
+
accent: "20 80% 92%",
|
|
24
|
+
accentFg: "20 40% 30%",
|
|
25
|
+
destructive: "0 74% 55%",
|
|
26
|
+
destructiveFg: "0 0% 100%"
|
|
27
|
+
},
|
|
28
|
+
day: {
|
|
29
|
+
bg: "0 0% 100%",
|
|
30
|
+
fg: "222 28% 14%",
|
|
31
|
+
muted: "210 20% 96%",
|
|
32
|
+
mutedFg: "215 16% 35%",
|
|
33
|
+
card: "0 0% 100%",
|
|
34
|
+
cardFg: "222 28% 14%",
|
|
35
|
+
border: "214 20% 90%",
|
|
36
|
+
ring: "220 65% 45%",
|
|
37
|
+
accent: "220 90% 95%",
|
|
38
|
+
accentFg: "220 45% 30%",
|
|
39
|
+
destructive: "0 72% 55%",
|
|
40
|
+
destructiveFg: "0 0% 100%"
|
|
41
|
+
},
|
|
42
|
+
dusk: {
|
|
43
|
+
bg: "240 24% 14%",
|
|
44
|
+
fg: "30 40% 95%",
|
|
45
|
+
muted: "245 20% 22%",
|
|
46
|
+
mutedFg: "30 20% 80%",
|
|
47
|
+
card: "240 22% 16%",
|
|
48
|
+
cardFg: "30 40% 95%",
|
|
49
|
+
border: "245 16% 30%",
|
|
50
|
+
ring: "32 70% 60%",
|
|
51
|
+
accent: "32 55% 25%",
|
|
52
|
+
accentFg: "32 70% 85%",
|
|
53
|
+
destructive: "0 70% 55%",
|
|
54
|
+
destructiveFg: "0 0% 100%"
|
|
55
|
+
},
|
|
56
|
+
night: {
|
|
57
|
+
bg: "230 22% 10%",
|
|
58
|
+
fg: "210 40% 96%",
|
|
59
|
+
muted: "230 18% 16%",
|
|
60
|
+
mutedFg: "210 20% 80%",
|
|
61
|
+
card: "230 20% 12%",
|
|
62
|
+
cardFg: "210 40% 96%",
|
|
63
|
+
border: "230 16% 24%",
|
|
64
|
+
ring: "210 80% 60%",
|
|
65
|
+
accent: "210 35% 20%",
|
|
66
|
+
accentFg: "210 50% 90%",
|
|
67
|
+
destructive: "0 65% 55%",
|
|
68
|
+
destructiveFg: "0 0% 100%"
|
|
69
|
+
}
|
|
70
|
+
};
|
|
71
|
+
var cssVarMap = {
|
|
72
|
+
bg: "--cui-bg",
|
|
73
|
+
fg: "--cui-fg",
|
|
74
|
+
muted: "--cui-muted",
|
|
75
|
+
mutedFg: "--cui-muted-fg",
|
|
76
|
+
card: "--cui-card",
|
|
77
|
+
cardFg: "--cui-card-fg",
|
|
78
|
+
border: "--cui-border",
|
|
79
|
+
ring: "--cui-ring",
|
|
80
|
+
accent: "--cui-accent",
|
|
81
|
+
accentFg: "--cui-accent-fg",
|
|
82
|
+
destructive: "--cui-destructive",
|
|
83
|
+
destructiveFg: "--cui-destructive-fg"
|
|
84
|
+
};
|
|
85
|
+
var tokensToCssVars = (tokens) => {
|
|
86
|
+
const vars = {};
|
|
87
|
+
for (const [key, value] of Object.entries(tokens)) {
|
|
88
|
+
const cssVar = cssVarMap[key];
|
|
89
|
+
vars[cssVar] = value;
|
|
90
|
+
}
|
|
91
|
+
return vars;
|
|
92
|
+
};
|
|
93
|
+
|
|
94
|
+
// src/core/script.ts
|
|
95
|
+
var serialize = (value) => JSON.stringify(value);
|
|
96
|
+
var getMergedSchedule = (schedule) => ({
|
|
97
|
+
...defaultSchedule,
|
|
98
|
+
...schedule,
|
|
99
|
+
dawn: { ...defaultSchedule.dawn, ...schedule?.dawn },
|
|
100
|
+
day: { ...defaultSchedule.day, ...schedule?.day },
|
|
101
|
+
dusk: { ...defaultSchedule.dusk, ...schedule?.dusk },
|
|
102
|
+
night: { ...defaultSchedule.night, ...schedule?.night }
|
|
103
|
+
});
|
|
104
|
+
var getMode = (mode) => mode ?? "time";
|
|
105
|
+
var createInlineScript = (config) => {
|
|
106
|
+
const schedule = getMergedSchedule(config?.schedule);
|
|
107
|
+
const storageKey = config?.storageKey ?? defaultStorageKey;
|
|
108
|
+
const persist = config?.persist !== false;
|
|
109
|
+
const tokens = {
|
|
110
|
+
dawn: tokensToCssVars({
|
|
111
|
+
...defaultTokens.dawn,
|
|
112
|
+
...config?.tokens?.dawn
|
|
113
|
+
}),
|
|
114
|
+
day: tokensToCssVars({
|
|
115
|
+
...defaultTokens.day,
|
|
116
|
+
...config?.tokens?.day
|
|
117
|
+
}),
|
|
118
|
+
dusk: tokensToCssVars({
|
|
119
|
+
...defaultTokens.dusk,
|
|
120
|
+
...config?.tokens?.dusk
|
|
121
|
+
}),
|
|
122
|
+
night: tokensToCssVars({
|
|
123
|
+
...defaultTokens.night,
|
|
124
|
+
...config?.tokens?.night
|
|
125
|
+
})
|
|
126
|
+
};
|
|
127
|
+
return `(() => {
|
|
128
|
+
try {
|
|
129
|
+
const schedule = ${serialize(schedule)};
|
|
130
|
+
const tokens = ${serialize(tokens)};
|
|
131
|
+
const storageKey = ${serialize(storageKey)};
|
|
132
|
+
const persist = ${serialize(persist)};
|
|
133
|
+
const fallbackMode = ${serialize(getMode(config?.mode))};
|
|
134
|
+
const now = new Date();
|
|
135
|
+
const minutes = now.getHours() * 60 + now.getMinutes();
|
|
136
|
+
|
|
137
|
+
const isWithin = (value, start, end) => {
|
|
138
|
+
if (start === end) return true;
|
|
139
|
+
if (start < end) return value >= start && value < end;
|
|
140
|
+
return value >= start || value < end;
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
const parse = (time) => {
|
|
144
|
+
const [h, m] = time.split(":").map(Number);
|
|
145
|
+
return ((h % 24) * 60 + m) % 1440;
|
|
146
|
+
};
|
|
147
|
+
|
|
148
|
+
const normalized = {
|
|
149
|
+
dawn: { start: parse(schedule.dawn.start), end: parse(schedule.dawn.end) },
|
|
150
|
+
day: { start: parse(schedule.day.start), end: parse(schedule.day.end) },
|
|
151
|
+
dusk: { start: parse(schedule.dusk.start), end: parse(schedule.dusk.end) },
|
|
152
|
+
night: { start: parse(schedule.night.start), end: parse(schedule.night.end) }
|
|
153
|
+
};
|
|
154
|
+
|
|
155
|
+
const order = ["dawn", "day", "dusk", "night"];
|
|
156
|
+
let phase = "night";
|
|
157
|
+
for (const key of order) {
|
|
158
|
+
const window = normalized[key];
|
|
159
|
+
if (isWithin(minutes, window.start, window.end)) {
|
|
160
|
+
phase = key;
|
|
161
|
+
break;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
let mode = fallbackMode;
|
|
166
|
+
const persisted = persist && window.localStorage ? window.localStorage.getItem(storageKey) : null;
|
|
167
|
+
if (persisted) {
|
|
168
|
+
try {
|
|
169
|
+
const parsed = JSON.parse(persisted);
|
|
170
|
+
if (parsed.mode) mode = parsed.mode;
|
|
171
|
+
if (parsed.phase) phase = parsed.phase;
|
|
172
|
+
} catch {
|
|
173
|
+
// ignore
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
const root = document.documentElement;
|
|
178
|
+
root.setAttribute("data-cui-phase", phase);
|
|
179
|
+
const vars = tokens[phase] || tokens.night;
|
|
180
|
+
for (const key in vars) {
|
|
181
|
+
root.style.setProperty(key, vars[key]);
|
|
182
|
+
}
|
|
183
|
+
} catch {
|
|
184
|
+
// ignore
|
|
185
|
+
}
|
|
186
|
+
})();`;
|
|
187
|
+
};
|
|
188
|
+
var resolveInitialPhase = (date, schedule) => {
|
|
189
|
+
const merged = getMergedSchedule(schedule);
|
|
190
|
+
const minutes = date.getHours() * 60 + date.getMinutes();
|
|
191
|
+
const isWithin = (value, start, end) => {
|
|
192
|
+
if (start === end) return true;
|
|
193
|
+
if (start < end) return value >= start && value < end;
|
|
194
|
+
return value >= start || value < end;
|
|
195
|
+
};
|
|
196
|
+
const parse = (time) => {
|
|
197
|
+
const [h, m] = time.split(":").map(Number);
|
|
198
|
+
return (h % 24 * 60 + m) % 1440;
|
|
199
|
+
};
|
|
200
|
+
const order = ["dawn", "day", "dusk", "night"];
|
|
201
|
+
for (const phase of order) {
|
|
202
|
+
const window2 = merged[phase];
|
|
203
|
+
if (isWithin(minutes, parse(window2.start), parse(window2.end))) {
|
|
204
|
+
return phase;
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
return "night";
|
|
208
|
+
};
|
|
209
|
+
|
|
210
|
+
export { createInlineScript, defaultSchedule, resolveInitialPhase };
|
|
211
|
+
//# sourceMappingURL=server.js.map
|
|
212
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/core/schedule.ts","../src/core/storage.ts","../src/core/tokens.ts","../src/core/script.ts"],"names":["window"],"mappings":";AAEO,IAAM,eAAA,GAAqC;AAAA,EAChD,IAAA,EAAM,EAAE,KAAA,EAAO,OAAA,EAAS,KAAK,OAAA,EAAQ;AAAA,EACrC,GAAA,EAAK,EAAE,KAAA,EAAO,OAAA,EAAS,KAAK,OAAA,EAAQ;AAAA,EACpC,IAAA,EAAM,EAAE,KAAA,EAAO,OAAA,EAAS,KAAK,OAAA,EAAQ;AAAA,EACrC,KAAA,EAAO,EAAE,KAAA,EAAO,OAAA,EAAS,KAAK,OAAA;AAChC;;;ACLO,IAAM,iBAAA,GAAoB,iBAAA;;;ACA1B,IAAM,aAAA,GAAgD;AAAA,EAC3D,IAAA,EAAM;AAAA,IACJ,EAAA,EAAI,YAAA;AAAA,IACJ,EAAA,EAAI,YAAA;AAAA,IACJ,KAAA,EAAO,YAAA;AAAA,IACP,OAAA,EAAS,YAAA;AAAA,IACT,IAAA,EAAM,WAAA;AAAA,IACN,MAAA,EAAQ,YAAA;AAAA,IACR,MAAA,EAAQ,YAAA;AAAA,IACR,IAAA,EAAM,YAAA;AAAA,IACN,MAAA,EAAQ,YAAA;AAAA,IACR,QAAA,EAAU,YAAA;AAAA,IACV,WAAA,EAAa,WAAA;AAAA,IACb,aAAA,EAAe;AAAA,GACjB;AAAA,EACA,GAAA,EAAK;AAAA,IACH,EAAA,EAAI,WAAA;AAAA,IACJ,EAAA,EAAI,aAAA;AAAA,IACJ,KAAA,EAAO,aAAA;AAAA,IACP,OAAA,EAAS,aAAA;AAAA,IACT,IAAA,EAAM,WAAA;AAAA,IACN,MAAA,EAAQ,aAAA;AAAA,IACR,MAAA,EAAQ,aAAA;AAAA,IACR,IAAA,EAAM,aAAA;AAAA,IACN,MAAA,EAAQ,aAAA;AAAA,IACR,QAAA,EAAU,aAAA;AAAA,IACV,WAAA,EAAa,WAAA;AAAA,IACb,aAAA,EAAe;AAAA,GACjB;AAAA,EACA,IAAA,EAAM;AAAA,IACJ,EAAA,EAAI,aAAA;AAAA,IACJ,EAAA,EAAI,YAAA;AAAA,IACJ,KAAA,EAAO,aAAA;AAAA,IACP,OAAA,EAAS,YAAA;AAAA,IACT,IAAA,EAAM,aAAA;AAAA,IACN,MAAA,EAAQ,YAAA;AAAA,IACR,MAAA,EAAQ,aAAA;AAAA,IACR,IAAA,EAAM,YAAA;AAAA,IACN,MAAA,EAAQ,YAAA;AAAA,IACR,QAAA,EAAU,YAAA;AAAA,IACV,WAAA,EAAa,WAAA;AAAA,IACb,aAAA,EAAe;AAAA,GACjB;AAAA,EACA,KAAA,EAAO;AAAA,IACL,EAAA,EAAI,aAAA;AAAA,IACJ,EAAA,EAAI,aAAA;AAAA,IACJ,KAAA,EAAO,aAAA;AAAA,IACP,OAAA,EAAS,aAAA;AAAA,IACT,IAAA,EAAM,aAAA;AAAA,IACN,MAAA,EAAQ,aAAA;AAAA,IACR,MAAA,EAAQ,aAAA;AAAA,IACR,IAAA,EAAM,aAAA;AAAA,IACN,MAAA,EAAQ,aAAA;AAAA,IACR,QAAA,EAAU,aAAA;AAAA,IACV,WAAA,EAAa,WAAA;AAAA,IACb,aAAA,EAAe;AAAA;AAEnB,CAAA;AAEO,IAAM,SAAA,GAAmD;AAAA,EAC9D,EAAA,EAAI,UAAA;AAAA,EACJ,EAAA,EAAI,UAAA;AAAA,EACJ,KAAA,EAAO,aAAA;AAAA,EACP,OAAA,EAAS,gBAAA;AAAA,EACT,IAAA,EAAM,YAAA;AAAA,EACN,MAAA,EAAQ,eAAA;AAAA,EACR,MAAA,EAAQ,cAAA;AAAA,EACR,IAAA,EAAM,YAAA;AAAA,EACN,MAAA,EAAQ,cAAA;AAAA,EACR,QAAA,EAAU,iBAAA;AAAA,EACV,WAAA,EAAa,mBAAA;AAAA,EACb,aAAA,EAAe;AACjB,CAAA;AAYO,IAAM,eAAA,GAAkB,CAAC,MAAA,KAAoD;AAClF,EAAA,MAAM,OAA+B,EAAC;AACtC,EAAA,KAAA,MAAW,CAAC,GAAA,EAAK,KAAK,KAAK,MAAA,CAAO,OAAA,CAAQ,MAAM,CAAA,EAAG;AACjD,IAAA,MAAM,MAAA,GAAS,UAAU,GAA4B,CAAA;AACrD,IAAA,IAAA,CAAK,MAAM,CAAA,GAAI,KAAA;AAAA,EACjB;AACA,EAAA,OAAO,IAAA;AACT,CAAA;;;ACxFA,IAAM,SAAA,GAAY,CAAC,KAAA,KAA2B,IAAA,CAAK,UAAU,KAAK,CAAA;AAElE,IAAM,iBAAA,GAAoB,CAAC,QAAA,MAA8D;AAAA,EACvF,GAAG,eAAA;AAAA,EACH,GAAG,QAAA;AAAA,EACH,MAAM,EAAE,GAAG,gBAAgB,IAAA,EAAM,GAAG,UAAU,IAAA,EAAK;AAAA,EACnD,KAAK,EAAE,GAAG,gBAAgB,GAAA,EAAK,GAAG,UAAU,GAAA,EAAI;AAAA,EAChD,MAAM,EAAE,GAAG,gBAAgB,IAAA,EAAM,GAAG,UAAU,IAAA,EAAK;AAAA,EACnD,OAAO,EAAE,GAAG,gBAAgB,KAAA,EAAO,GAAG,UAAU,KAAA;AAClD,CAAA,CAAA;AAEA,IAAM,OAAA,GAAU,CAAC,IAAA,KAAsC,IAAA,IAAQ,MAAA;AAExD,IAAM,kBAAA,GAAqB,CAAC,MAAA,KAAqC;AACtE,EAAA,MAAM,QAAA,GAAW,iBAAA,CAAkB,MAAA,EAAQ,QAAQ,CAAA;AACnD,EAAA,MAAM,UAAA,GAAa,QAAQ,UAAA,IAAc,iBAAA;AACzC,EAAA,MAAM,OAAA,GAAU,QAAQ,OAAA,KAAY,KAAA;AACpC,EAAA,MAAM,MAAA,GAAS;AAAA,IACb,MAAM,eAAA,CAAgB;AAAA,MACpB,GAAG,aAAA,CAAc,IAAA;AAAA,MACjB,GAAG,QAAQ,MAAA,EAAQ;AAAA,KACpB,CAAA;AAAA,IACD,KAAK,eAAA,CAAgB;AAAA,MACnB,GAAG,aAAA,CAAc,GAAA;AAAA,MACjB,GAAG,QAAQ,MAAA,EAAQ;AAAA,KACpB,CAAA;AAAA,IACD,MAAM,eAAA,CAAgB;AAAA,MACpB,GAAG,aAAA,CAAc,IAAA;AAAA,MACjB,GAAG,QAAQ,MAAA,EAAQ;AAAA,KACpB,CAAA;AAAA,IACD,OAAO,eAAA,CAAgB;AAAA,MACrB,GAAG,aAAA,CAAc,KAAA;AAAA,MACjB,GAAG,QAAQ,MAAA,EAAQ;AAAA,KACpB;AAAA,GACH;AAEA,EAAA,OAAO,CAAA;AAAA;AAAA,qBAAA,EAEc,SAAA,CAAU,QAAQ,CAAC,CAAA;AAAA,mBAAA,EACrB,SAAA,CAAU,MAAM,CAAC,CAAA;AAAA,uBAAA,EACb,SAAA,CAAU,UAAU,CAAC,CAAA;AAAA,oBAAA,EACxB,SAAA,CAAU,OAAO,CAAC,CAAA;AAAA,yBAAA,EACb,SAAA,CAAU,OAAA,CAAQ,MAAA,EAAQ,IAAI,CAAC,CAAC,CAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,KAAA,CAAA;AAsD3D;AAEO,IAAM,mBAAA,GAAsB,CAAC,IAAA,EAAY,QAAA,KAAiD;AAC/F,EAAA,MAAM,MAAA,GAAS,kBAAkB,QAAQ,CAAA;AACzC,EAAA,MAAM,UAAU,IAAA,CAAK,QAAA,EAAS,GAAI,EAAA,GAAK,KAAK,UAAA,EAAW;AACvD,EAAA,MAAM,QAAA,GAAW,CAAC,KAAA,EAAe,KAAA,EAAe,GAAA,KAAgB;AAC9D,IAAA,IAAI,KAAA,KAAU,KAAK,OAAO,IAAA;AAC1B,IAAA,IAAI,KAAA,GAAQ,GAAA,EAAK,OAAO,KAAA,IAAS,SAAS,KAAA,GAAQ,GAAA;AAClD,IAAA,OAAO,KAAA,IAAS,SAAS,KAAA,GAAQ,GAAA;AAAA,EACnC,CAAA;AACA,EAAA,MAAM,KAAA,GAAQ,CAAC,IAAA,KAAiB;AAC9B,IAAA,MAAM,CAAC,GAAG,CAAC,CAAA,GAAI,KAAK,KAAA,CAAM,GAAG,CAAA,CAAE,GAAA,CAAI,MAAM,CAAA;AACzC,IAAA,OAAA,CAAS,CAAA,GAAI,EAAA,GAAM,EAAA,GAAK,CAAA,IAAK,IAAA;AAAA,EAC/B,CAAA;AACA,EAAA,MAAM,KAAA,GAAiB,CAAC,MAAA,EAAQ,KAAA,EAAO,QAAQ,OAAO,CAAA;AACtD,EAAA,KAAA,MAAW,SAAS,KAAA,EAAO;AACzB,IAAA,MAAMA,OAAAA,GAAS,OAAO,KAAK,CAAA;AAC3B,IAAA,IAAI,QAAA,CAAS,OAAA,EAAS,KAAA,CAAMA,OAAAA,CAAO,KAAK,GAAG,KAAA,CAAMA,OAAAA,CAAO,GAAG,CAAC,CAAA,EAAG;AAC7D,MAAA,OAAO,KAAA;AAAA,IACT;AAAA,EACF;AACA,EAAA,OAAO,OAAA;AACT","file":"server.js","sourcesContent":["import { CircadianSchedule, Phase } from \"./types\";\n\nexport const defaultSchedule: CircadianSchedule = {\n dawn: { start: \"05:30\", end: \"08:30\" },\n day: { start: \"08:30\", end: \"17:30\" },\n dusk: { start: \"17:30\", end: \"21:30\" },\n night: { start: \"21:30\", end: \"05:30\" }\n};\n\nexport interface PhaseWindowMinutes {\n start: number;\n end: number;\n}\n\nexport type CircadianScheduleMinutes = Record<Phase, PhaseWindowMinutes>;\n\nconst minutesInDay = 24 * 60;\n\nexport const parseTimeToMinutes = (value: string): number => {\n const [hours, minutes] = value.split(\":\").map(Number);\n if (Number.isNaN(hours) || Number.isNaN(minutes)) {\n throw new Error(`Invalid time format: ${value}`);\n }\n return ((hours % 24) * 60 + minutes) % minutesInDay;\n};\n\nexport const normalizeSchedule = (\n schedule?: Partial<CircadianSchedule>\n): CircadianScheduleMinutes => {\n const merged: CircadianSchedule = {\n ...defaultSchedule,\n ...schedule,\n dawn: { ...defaultSchedule.dawn, ...schedule?.dawn },\n day: { ...defaultSchedule.day, ...schedule?.day },\n dusk: { ...defaultSchedule.dusk, ...schedule?.dusk },\n night: { ...defaultSchedule.night, ...schedule?.night }\n };\n\n return {\n dawn: {\n start: parseTimeToMinutes(merged.dawn.start),\n end: parseTimeToMinutes(merged.dawn.end)\n },\n day: {\n start: parseTimeToMinutes(merged.day.start),\n end: parseTimeToMinutes(merged.day.end)\n },\n dusk: {\n start: parseTimeToMinutes(merged.dusk.start),\n end: parseTimeToMinutes(merged.dusk.end)\n },\n night: {\n start: parseTimeToMinutes(merged.night.start),\n end: parseTimeToMinutes(merged.night.end)\n }\n };\n};\n\nexport const getMinutesFromDate = (date: Date): number => date.getHours() * 60 + date.getMinutes();\n\nconst isWithinRange = (minutes: number, start: number, end: number): boolean => {\n if (start === end) {\n return true;\n }\n if (start < end) {\n return minutes >= start && minutes < end;\n }\n return minutes >= start || minutes < end;\n};\n\nexport const getPhaseFromTime = (date: Date, schedule?: Partial<CircadianSchedule>): Phase => {\n const minutes = getMinutesFromDate(date);\n const normalized = normalizeSchedule(schedule);\n const phases: Phase[] = [\"dawn\", \"day\", \"dusk\", \"night\"];\n for (const phase of phases) {\n const window = normalized[phase];\n if (isWithinRange(minutes, window.start, window.end)) {\n return phase;\n }\n }\n return \"night\";\n};\n\nexport const computeNextTransition = (date: Date, schedule?: Partial<CircadianSchedule>): Date => {\n const normalized = normalizeSchedule(schedule);\n const currentPhase = getPhaseFromTime(date, schedule);\n const minutes = getMinutesFromDate(date);\n const endMinutes = normalized[currentPhase].end;\n let delta = endMinutes - minutes;\n if (delta <= 0) {\n delta += minutesInDay;\n }\n return new Date(date.getTime() + delta * 60 * 1000);\n};\n","import { PersistedState } from \"./types\";\n\nexport const defaultStorageKey = \"cui:preferences\";\n\nconst hasStorage = (): boolean =>\n typeof window !== \"undefined\" && typeof window.localStorage !== \"undefined\";\n\nexport const loadPersistedState = (key: string = defaultStorageKey): PersistedState | null => {\n if (!hasStorage()) {\n return null;\n }\n try {\n const raw = window.localStorage.getItem(key);\n if (!raw) {\n return null;\n }\n return JSON.parse(raw) as PersistedState;\n } catch {\n return null;\n }\n};\n\nexport const persistState = (state: PersistedState, key: string = defaultStorageKey): void => {\n if (!hasStorage()) {\n return;\n }\n try {\n window.localStorage.setItem(key, JSON.stringify(state));\n } catch {\n // Ignore write errors\n }\n};\n\nexport const clearPersistedState = (key: string = defaultStorageKey): void => {\n if (!hasStorage()) {\n return;\n }\n try {\n window.localStorage.removeItem(key);\n } catch {\n // Ignore cleanup errors\n }\n};\n","import { CircadianTokens, ColorSchemeBias, Phase } from \"./types\";\n\nexport const defaultTokens: Record<Phase, CircadianTokens> = {\n dawn: {\n bg: \"27 60% 96%\",\n fg: \"24 18% 18%\",\n muted: \"27 40% 90%\",\n mutedFg: \"24 14% 35%\",\n card: \"0 0% 100%\",\n cardFg: \"24 18% 18%\",\n border: \"24 22% 84%\",\n ring: \"20 65% 45%\",\n accent: \"20 80% 92%\",\n accentFg: \"20 40% 30%\",\n destructive: \"0 74% 55%\",\n destructiveFg: \"0 0% 100%\"\n },\n day: {\n bg: \"0 0% 100%\",\n fg: \"222 28% 14%\",\n muted: \"210 20% 96%\",\n mutedFg: \"215 16% 35%\",\n card: \"0 0% 100%\",\n cardFg: \"222 28% 14%\",\n border: \"214 20% 90%\",\n ring: \"220 65% 45%\",\n accent: \"220 90% 95%\",\n accentFg: \"220 45% 30%\",\n destructive: \"0 72% 55%\",\n destructiveFg: \"0 0% 100%\"\n },\n dusk: {\n bg: \"240 24% 14%\",\n fg: \"30 40% 95%\",\n muted: \"245 20% 22%\",\n mutedFg: \"30 20% 80%\",\n card: \"240 22% 16%\",\n cardFg: \"30 40% 95%\",\n border: \"245 16% 30%\",\n ring: \"32 70% 60%\",\n accent: \"32 55% 25%\",\n accentFg: \"32 70% 85%\",\n destructive: \"0 70% 55%\",\n destructiveFg: \"0 0% 100%\"\n },\n night: {\n bg: \"230 22% 10%\",\n fg: \"210 40% 96%\",\n muted: \"230 18% 16%\",\n mutedFg: \"210 20% 80%\",\n card: \"230 20% 12%\",\n cardFg: \"210 40% 96%\",\n border: \"230 16% 24%\",\n ring: \"210 80% 60%\",\n accent: \"210 35% 20%\",\n accentFg: \"210 50% 90%\",\n destructive: \"0 65% 55%\",\n destructiveFg: \"0 0% 100%\"\n }\n};\n\nexport const cssVarMap: Record<keyof CircadianTokens, string> = {\n bg: \"--cui-bg\",\n fg: \"--cui-fg\",\n muted: \"--cui-muted\",\n mutedFg: \"--cui-muted-fg\",\n card: \"--cui-card\",\n cardFg: \"--cui-card-fg\",\n border: \"--cui-border\",\n ring: \"--cui-ring\",\n accent: \"--cui-accent\",\n accentFg: \"--cui-accent-fg\",\n destructive: \"--cui-destructive\",\n destructiveFg: \"--cui-destructive-fg\"\n};\n\nexport const resolveTokens = (\n phase: Phase,\n overrides?: Partial<Record<Phase, Partial<CircadianTokens>>>\n): CircadianTokens => {\n return {\n ...defaultTokens[phase],\n ...overrides?.[phase]\n };\n};\n\nexport const tokensToCssVars = (tokens: CircadianTokens): Record<string, string> => {\n const vars: Record<string, string> = {};\n for (const [key, value] of Object.entries(tokens)) {\n const cssVar = cssVarMap[key as keyof CircadianTokens];\n vars[cssVar] = value;\n }\n return vars;\n};\n\nexport const applyTokensToElement = (element: HTMLElement, tokens: CircadianTokens) => {\n const vars = tokensToCssVars(tokens);\n for (const [key, value] of Object.entries(vars)) {\n element.style.setProperty(key, value);\n }\n};\n\nexport const applyColorSchemeBias = (\n tokens: CircadianTokens,\n prefers: \"dark\" | \"light\" | \"no-preference\",\n bias: ColorSchemeBias\n): CircadianTokens => {\n if (prefers === \"no-preference\") {\n return tokens;\n }\n const delta = prefers === \"dark\" ? bias.dark : bias.light;\n const adjust = (value: string): string => {\n const [h, s, l] = value.split(\" \");\n const lightness = Math.max(0, Math.min(100, Number(l.replace(\"%\", \"\")) + delta));\n return `${h} ${s} ${lightness}%`;\n };\n\n return {\n ...tokens,\n bg: adjust(tokens.bg),\n fg: adjust(tokens.fg),\n muted: adjust(tokens.muted),\n mutedFg: adjust(tokens.mutedFg),\n card: adjust(tokens.card),\n cardFg: adjust(tokens.cardFg),\n border: adjust(tokens.border),\n ring: adjust(tokens.ring),\n accent: adjust(tokens.accent),\n accentFg: adjust(tokens.accentFg),\n destructive: adjust(tokens.destructive),\n destructiveFg: adjust(tokens.destructiveFg)\n };\n};\n","import { CircadianConfig, CircadianSchedule, Phase, ScheduleMode } from \"./types\";\nimport { defaultSchedule } from \"./schedule\";\nimport { defaultStorageKey } from \"./storage\";\nimport { defaultTokens, tokensToCssVars } from \"./tokens\";\n\nconst serialize = (value: unknown): string => JSON.stringify(value);\n\nconst getMergedSchedule = (schedule?: Partial<CircadianSchedule>): CircadianSchedule => ({\n ...defaultSchedule,\n ...schedule,\n dawn: { ...defaultSchedule.dawn, ...schedule?.dawn },\n day: { ...defaultSchedule.day, ...schedule?.day },\n dusk: { ...defaultSchedule.dusk, ...schedule?.dusk },\n night: { ...defaultSchedule.night, ...schedule?.night }\n});\n\nconst getMode = (mode?: ScheduleMode): ScheduleMode => mode ?? \"time\";\n\nexport const createInlineScript = (config?: CircadianConfig): string => {\n const schedule = getMergedSchedule(config?.schedule);\n const storageKey = config?.storageKey ?? defaultStorageKey;\n const persist = config?.persist !== false;\n const tokens = {\n dawn: tokensToCssVars({\n ...defaultTokens.dawn,\n ...config?.tokens?.dawn\n }),\n day: tokensToCssVars({\n ...defaultTokens.day,\n ...config?.tokens?.day\n }),\n dusk: tokensToCssVars({\n ...defaultTokens.dusk,\n ...config?.tokens?.dusk\n }),\n night: tokensToCssVars({\n ...defaultTokens.night,\n ...config?.tokens?.night\n })\n };\n\n return `(() => {\n try {\n const schedule = ${serialize(schedule)};\n const tokens = ${serialize(tokens)};\n const storageKey = ${serialize(storageKey)};\n const persist = ${serialize(persist)};\n const fallbackMode = ${serialize(getMode(config?.mode))};\n const now = new Date();\n const minutes = now.getHours() * 60 + now.getMinutes();\n\n const isWithin = (value, start, end) => {\n if (start === end) return true;\n if (start < end) return value >= start && value < end;\n return value >= start || value < end;\n };\n\n const parse = (time) => {\n const [h, m] = time.split(\":\").map(Number);\n return ((h % 24) * 60 + m) % 1440;\n };\n\n const normalized = {\n dawn: { start: parse(schedule.dawn.start), end: parse(schedule.dawn.end) },\n day: { start: parse(schedule.day.start), end: parse(schedule.day.end) },\n dusk: { start: parse(schedule.dusk.start), end: parse(schedule.dusk.end) },\n night: { start: parse(schedule.night.start), end: parse(schedule.night.end) }\n };\n\n const order = [\"dawn\", \"day\", \"dusk\", \"night\"];\n let phase = \"night\";\n for (const key of order) {\n const window = normalized[key];\n if (isWithin(minutes, window.start, window.end)) {\n phase = key;\n break;\n }\n }\n\n let mode = fallbackMode;\n const persisted = persist && window.localStorage ? window.localStorage.getItem(storageKey) : null;\n if (persisted) {\n try {\n const parsed = JSON.parse(persisted);\n if (parsed.mode) mode = parsed.mode;\n if (parsed.phase) phase = parsed.phase;\n } catch {\n // ignore\n }\n }\n\n const root = document.documentElement;\n root.setAttribute(\"data-cui-phase\", phase);\n const vars = tokens[phase] || tokens.night;\n for (const key in vars) {\n root.style.setProperty(key, vars[key]);\n }\n } catch {\n // ignore\n }\n})();`;\n};\n\nexport const resolveInitialPhase = (date: Date, schedule?: Partial<CircadianSchedule>): Phase => {\n const merged = getMergedSchedule(schedule);\n const minutes = date.getHours() * 60 + date.getMinutes();\n const isWithin = (value: number, start: number, end: number) => {\n if (start === end) return true;\n if (start < end) return value >= start && value < end;\n return value >= start || value < end;\n };\n const parse = (time: string) => {\n const [h, m] = time.split(\":\").map(Number);\n return ((h % 24) * 60 + m) % 1440;\n };\n const order: Phase[] = [\"dawn\", \"day\", \"dusk\", \"night\"];\n for (const phase of order) {\n const window = merged[phase];\n if (isWithin(minutes, parse(window.start), parse(window.end))) {\n return phase;\n }\n }\n return \"night\";\n};\n"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@shiftbloom-studio/circadian-ui",
|
|
3
|
+
"version": "0.2.0",
|
|
4
|
+
"description": "Accessible, time-aware theming for React and Next.js.",
|
|
5
|
+
"keywords": [
|
|
6
|
+
"react",
|
|
7
|
+
"nextjs",
|
|
8
|
+
"theme",
|
|
9
|
+
"accessibility",
|
|
10
|
+
"tailwind",
|
|
11
|
+
"wcag",
|
|
12
|
+
"design-tokens"
|
|
13
|
+
],
|
|
14
|
+
"homepage": "https://github.com/shiftbloom-studio/circadian-ui#readme",
|
|
15
|
+
"bugs": {
|
|
16
|
+
"url": "https://github.com/shiftbloom-studio/circadian-ui/issues"
|
|
17
|
+
},
|
|
18
|
+
"repository": {
|
|
19
|
+
"type": "git",
|
|
20
|
+
"url": "git+https://github.com/shiftbloom-studio/circadian-ui.git"
|
|
21
|
+
},
|
|
22
|
+
"license": "MIT",
|
|
23
|
+
"author": "",
|
|
24
|
+
"publishConfig": {
|
|
25
|
+
"access": "public"
|
|
26
|
+
},
|
|
27
|
+
"engines": {
|
|
28
|
+
"node": ">=18"
|
|
29
|
+
},
|
|
30
|
+
"type": "module",
|
|
31
|
+
"main": "dist/index.cjs",
|
|
32
|
+
"module": "dist/index.js",
|
|
33
|
+
"types": "dist/index.d.ts",
|
|
34
|
+
"exports": {
|
|
35
|
+
".": {
|
|
36
|
+
"types": "./dist/index.d.ts",
|
|
37
|
+
"import": "./dist/index.js",
|
|
38
|
+
"require": "./dist/index.cjs"
|
|
39
|
+
},
|
|
40
|
+
"./server": {
|
|
41
|
+
"types": "./dist/server.d.ts",
|
|
42
|
+
"import": "./dist/server.js",
|
|
43
|
+
"require": "./dist/server.cjs"
|
|
44
|
+
}
|
|
45
|
+
},
|
|
46
|
+
"files": [
|
|
47
|
+
"dist",
|
|
48
|
+
"README.md",
|
|
49
|
+
"LICENSE"
|
|
50
|
+
],
|
|
51
|
+
"sideEffects": false,
|
|
52
|
+
"peerDependencies": {
|
|
53
|
+
"react": ">=18",
|
|
54
|
+
"react-dom": ">=18"
|
|
55
|
+
},
|
|
56
|
+
"scripts": {
|
|
57
|
+
"build": "tsup",
|
|
58
|
+
"dev": "tsup --watch",
|
|
59
|
+
"clean": "node scripts/clean.mjs",
|
|
60
|
+
"changeset": "changeset",
|
|
61
|
+
"version": "changeset version",
|
|
62
|
+
"release": "changeset publish",
|
|
63
|
+
"test": "jest",
|
|
64
|
+
"test:ci": "jest --ci",
|
|
65
|
+
"test:watch": "jest --watch",
|
|
66
|
+
"lint": "eslint . --ext .ts,.tsx --max-warnings=0",
|
|
67
|
+
"typecheck": "tsc -p tsconfig.json --noEmit"
|
|
68
|
+
},
|
|
69
|
+
"devDependencies": {
|
|
70
|
+
"@eslint/js": "^9.17.0",
|
|
71
|
+
"@changesets/changelog-git": "^0.2.1",
|
|
72
|
+
"@changesets/cli": "^2.28.1",
|
|
73
|
+
"@testing-library/jest-dom": "^6.8.0",
|
|
74
|
+
"@types/jest": "^30.0.0",
|
|
75
|
+
"@typescript-eslint/eslint-plugin": "^8.18.2",
|
|
76
|
+
"@typescript-eslint/parser": "^8.18.2",
|
|
77
|
+
"eslint": "^9.17.0",
|
|
78
|
+
"eslint-config-prettier": "^10.1.8",
|
|
79
|
+
"globals": "^17.0.0",
|
|
80
|
+
"jest": "^30.2.0",
|
|
81
|
+
"jest-environment-jsdom": "^30.2.0",
|
|
82
|
+
"prettier": "^3.4.2",
|
|
83
|
+
"ts-jest": "^29.2.5",
|
|
84
|
+
"tsup": "^8.3.6",
|
|
85
|
+
"typescript": "^5.7.3"
|
|
86
|
+
}
|
|
87
|
+
}
|