@rusticarcade/palette 0.1.0-rc.5
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/README.md +19 -0
- package/dist/dev/index.js +1861 -0
- package/dist/prod/index.d.ts +922 -0
- package/dist/prod/index.js +3 -0
- package/package.json +40 -0
|
@@ -0,0 +1,1861 @@
|
|
|
1
|
+
// src/errors.ts
|
|
2
|
+
function getDevError(code) {
|
|
3
|
+
const DEV_ERROR_INFO = {
|
|
4
|
+
[0 /* INVARIANT */]: `
|
|
5
|
+
INVARIANT
|
|
6
|
+
A condition has occurred which should never happen, such as a code path that
|
|
7
|
+
shouldn't be reachable when using correct types, or a disallowed state.
|
|
8
|
+
`,
|
|
9
|
+
[1 /* INVALID_ENV */]: `
|
|
10
|
+
INVALID_ENV
|
|
11
|
+
Failed to find a custom elements registry in this environment. Is a window
|
|
12
|
+
global available with a customElements property?
|
|
13
|
+
`,
|
|
14
|
+
[100 /* DUPE_TAGNAME */]: `
|
|
15
|
+
DUPE_TAGNAME
|
|
16
|
+
Failed to register a custom element with the tagname %s because another
|
|
17
|
+
custom element is already defined with that name.
|
|
18
|
+
`,
|
|
19
|
+
[101 /* INVALID_TAGNAME */]: `
|
|
20
|
+
INVALID_TAGNAME
|
|
21
|
+
Failed to determine a valid HTML Tag to use when registering a component as a
|
|
22
|
+
custom element. Specify a static readonly tagName property on your component
|
|
23
|
+
class or provide a tagName argument to Component.register.
|
|
24
|
+
|
|
25
|
+
Alternatively, you can directly register your element as you would any other
|
|
26
|
+
HTML custom element: customElements.define("my-tag", MyComponentClass);
|
|
27
|
+
`,
|
|
28
|
+
[300 /* MISSING_ELEMENT */]: `
|
|
29
|
+
MISSING_ELEMENT
|
|
30
|
+
An element was expected to exist but was not found when the DOM was queried.
|
|
31
|
+
`,
|
|
32
|
+
[301 /* MISSING_STATE */]: `
|
|
33
|
+
MISSING_STATE
|
|
34
|
+
A Component's reactive state was accessed, but no initialState is defined for
|
|
35
|
+
the Component's class. Initial state must be defined as an object or function.
|
|
36
|
+
`,
|
|
37
|
+
[302 /* INVALID_CONTENT */]: `
|
|
38
|
+
INVALID_CONTENT
|
|
39
|
+
Tried to set a value as content in a template, but the value was not a supported
|
|
40
|
+
type. Values must be a string, HTMLElement.
|
|
41
|
+
|
|
42
|
+
Instead, got %s: %s.
|
|
43
|
+
`,
|
|
44
|
+
[200 /* TEMPLATE_INVALID */]: `
|
|
45
|
+
TEMPLATE_INVALID
|
|
46
|
+
The value %s cannot be interpolated via the Template html string helper.
|
|
47
|
+
Values may only be HTMLTemplateElements (template content is adopted) or
|
|
48
|
+
strings (shorthand for a ::swap directive on an element)
|
|
49
|
+
`,
|
|
50
|
+
[201 /* TEMPLATE_INVALID_NOTATION */]: `
|
|
51
|
+
TEMPLATE_INVALID_NOTATION
|
|
52
|
+
A template value notation failed to parse. Notations must begin with one of
|
|
53
|
+
@, $, *, or #, followed by a dot-separated path of accessor names. Optionally,
|
|
54
|
+
notations may have modifiers around them, such as the NOT() modifier.
|
|
55
|
+
|
|
56
|
+
The notation that failed to parse was "%s"
|
|
57
|
+
`,
|
|
58
|
+
[202 /* TEMPLATE_INVALID_COMPONENT */]: `
|
|
59
|
+
TEMPLATE_INVALID_COMPONENT
|
|
60
|
+
Failed to interpolate a Component in an html template string due to the
|
|
61
|
+
Component having an invalid tagName property.
|
|
62
|
+
|
|
63
|
+
Component "%s" has an invalid tagName property "%s"
|
|
64
|
+
`,
|
|
65
|
+
[205 /* TEMPLATE_MISSING_LIST_PARENT */]: `
|
|
66
|
+
TEMPLATE_MISSING_LIST_PARENT
|
|
67
|
+
A template failed to prepare an ::each directive because no valid parent node
|
|
68
|
+
was found for the list to render into.
|
|
69
|
+
`,
|
|
70
|
+
[203 /* TEMPLATE_INVALID_LIST */]: `
|
|
71
|
+
TEMPLATE_INVALID_LIST
|
|
72
|
+
A template failed to render because an ::each directive fetched a value which
|
|
73
|
+
was not an array of actionable data.
|
|
74
|
+
`,
|
|
75
|
+
[204 /* TEMPLATE_INVALID_NODE_REF */]: `
|
|
76
|
+
TEMPLATE_INVALID_NODE_REF
|
|
77
|
+
A template failed to prepare a node from a compiled source. Typically, this
|
|
78
|
+
means something is broken within the compiled template rather than an error with
|
|
79
|
+
the template content itself.
|
|
80
|
+
`,
|
|
81
|
+
[206 /* TEMPLATE_MISSING_LIST_KEY */]: `
|
|
82
|
+
TEMPLATE_MISSING_LIST_KEY
|
|
83
|
+
A template failed to compile because an ::each directive was found on an element
|
|
84
|
+
without a corresponding ::key directive. ::each directives must have a ::key
|
|
85
|
+
directive specified on the same element, whose notation evaluates to a unique
|
|
86
|
+
value for each item in the list.
|
|
87
|
+
|
|
88
|
+
Example: <li ::each="$items" ::key="#id"> ... </li>
|
|
89
|
+
`,
|
|
90
|
+
[207 /* TEMPLATE_DUPLICATE_LIST_KEY */]: `
|
|
91
|
+
TEMPLATE_DUPLICATE_LIST_KEY
|
|
92
|
+
A list failed to render because item key %s appeared multiple times when
|
|
93
|
+
resolving list item keys. Each ::key directive must reference a notation whose
|
|
94
|
+
value is unique for each list item.
|
|
95
|
+
`,
|
|
96
|
+
[303 /* INVALID_STATE_KEY */]: `
|
|
97
|
+
INVALID_STATE_KEY
|
|
98
|
+
Tried to access a state property which does not exist
|
|
99
|
+
|
|
100
|
+
Key path: %s
|
|
101
|
+
`,
|
|
102
|
+
[304 /* INVALID_STATE_UPDATE */]: `
|
|
103
|
+
INVALID_STATE_UPDATE
|
|
104
|
+
A Component's setState() was called with a value which could not be patched in
|
|
105
|
+
to the existing component state.
|
|
106
|
+
|
|
107
|
+
setState() takes a partial object describing state keys to update.
|
|
108
|
+
|
|
109
|
+
Instead, received: %s
|
|
110
|
+
`,
|
|
111
|
+
[305 /* STATE_LOCKED */]: `
|
|
112
|
+
STATE_LOCKED
|
|
113
|
+
A State received a request to update or lock while already locked.
|
|
114
|
+
States may be locked using the .lock() method and unlocked using the .unlock()
|
|
115
|
+
method. States lock during async mutations done through the .mutateAsync()
|
|
116
|
+
method if "true" is passed as the second parameter.
|
|
117
|
+
|
|
118
|
+
You can check the lock status of a State instance with .isLocked
|
|
119
|
+
`,
|
|
120
|
+
[306 /* MAX_SEQUENTIAL_RENDERS */]: `
|
|
121
|
+
MAX_SEQUENTIAL_RENDERS
|
|
122
|
+
A component has errored after re-rendering more than %s times in the same frame.
|
|
123
|
+
|
|
124
|
+
This typically happens because state or live attributes are being modified
|
|
125
|
+
during a render lifecycle method such as beforeUpdate() or afterUpdate(). If
|
|
126
|
+
state changes unconditionally during a render, the render will infinitely loop.
|
|
127
|
+
Updating render source data during a render cycle is supported, but infinite
|
|
128
|
+
update loops may occur if unchecked.
|
|
129
|
+
`
|
|
130
|
+
};
|
|
131
|
+
return DEV_ERROR_INFO[code];
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
class PaletteError extends Error {
|
|
135
|
+
name = "PaletteError";
|
|
136
|
+
code;
|
|
137
|
+
constructor(code, ...values) {
|
|
138
|
+
let message = `Code: ${code}`;
|
|
139
|
+
if (true) {
|
|
140
|
+
message += getDevError(code);
|
|
141
|
+
for (const val of values) {
|
|
142
|
+
message = message.replace(`%s`, String(val));
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
super(message);
|
|
146
|
+
this.code = code;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// src/helper/css.ts
|
|
151
|
+
function css(strings, ...values) {
|
|
152
|
+
let result = strings[0] ?? "";
|
|
153
|
+
for (let i = 0;i < values.length; i++) {
|
|
154
|
+
const nextVal = values[i] ?? "";
|
|
155
|
+
const nextStr = strings[i + 1] ?? "";
|
|
156
|
+
result += `${nextVal}${nextStr}`;
|
|
157
|
+
}
|
|
158
|
+
const sheet = new CSSStyleSheet;
|
|
159
|
+
sheet.replaceSync(result);
|
|
160
|
+
return sheet;
|
|
161
|
+
}
|
|
162
|
+
function classify(...args) {
|
|
163
|
+
let resultString = "";
|
|
164
|
+
for (let index = 0;index < args.length; index++) {
|
|
165
|
+
const argument = args[index];
|
|
166
|
+
if (!argument) {
|
|
167
|
+
continue;
|
|
168
|
+
}
|
|
169
|
+
if (typeof argument === "string" || typeof argument === "number") {
|
|
170
|
+
if (resultString) {
|
|
171
|
+
resultString += " ";
|
|
172
|
+
}
|
|
173
|
+
resultString += argument;
|
|
174
|
+
continue;
|
|
175
|
+
}
|
|
176
|
+
if (Array.isArray(argument)) {
|
|
177
|
+
const nestedResult = classify(...argument);
|
|
178
|
+
if (nestedResult) {
|
|
179
|
+
if (resultString) {
|
|
180
|
+
resultString += " ";
|
|
181
|
+
}
|
|
182
|
+
resultString += nestedResult;
|
|
183
|
+
}
|
|
184
|
+
continue;
|
|
185
|
+
}
|
|
186
|
+
if (typeof argument === "object") {
|
|
187
|
+
for (const className in argument) {
|
|
188
|
+
if (argument[className]) {
|
|
189
|
+
if (resultString) {
|
|
190
|
+
resultString += " ";
|
|
191
|
+
}
|
|
192
|
+
resultString += className;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
return resultString;
|
|
198
|
+
}
|
|
199
|
+
// src/util/attributes.ts
|
|
200
|
+
function serializeAttribute(value) {
|
|
201
|
+
if (typeof value === "string" || typeof value === "number") {
|
|
202
|
+
return `${value}`;
|
|
203
|
+
}
|
|
204
|
+
if (value === true) {
|
|
205
|
+
return "";
|
|
206
|
+
}
|
|
207
|
+
if (value === null || value === false || value === undefined) {
|
|
208
|
+
return null;
|
|
209
|
+
}
|
|
210
|
+
return value.toString();
|
|
211
|
+
}
|
|
212
|
+
function createAttributeMap(source) {
|
|
213
|
+
const map = new Map;
|
|
214
|
+
if (source instanceof HTMLElement) {
|
|
215
|
+
for (const name of source.getAttributeNames()) {
|
|
216
|
+
map.set(name, serializeAttribute(source.getAttribute(name)));
|
|
217
|
+
}
|
|
218
|
+
return map;
|
|
219
|
+
}
|
|
220
|
+
if (source instanceof Map) {
|
|
221
|
+
for (const [key, val] of source) {
|
|
222
|
+
map.set(key, serializeAttribute(val));
|
|
223
|
+
}
|
|
224
|
+
return map;
|
|
225
|
+
}
|
|
226
|
+
if (typeof source === "object" && source !== null) {
|
|
227
|
+
for (const [key, val] of Object.entries(source)) {
|
|
228
|
+
map.set(key, serializeAttribute(val));
|
|
229
|
+
}
|
|
230
|
+
return map;
|
|
231
|
+
}
|
|
232
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
233
|
+
}
|
|
234
|
+
function applyAttributeMap(target, attrs) {
|
|
235
|
+
const currentNames = new Set(target.getAttributeNames());
|
|
236
|
+
const incomingNames = new Set(attrs.keys());
|
|
237
|
+
const attributesToRemove = currentNames.difference(incomingNames);
|
|
238
|
+
for (const attr of attributesToRemove) {
|
|
239
|
+
target.removeAttribute(attr);
|
|
240
|
+
}
|
|
241
|
+
for (const [name, val] of attrs) {
|
|
242
|
+
if (val === null) {
|
|
243
|
+
target.removeAttribute(name);
|
|
244
|
+
} else {
|
|
245
|
+
target.setAttribute(name, val);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
// src/util/change-tag.ts
|
|
250
|
+
function changeTag($target, newTag) {
|
|
251
|
+
const $newEl = document.createElement(newTag);
|
|
252
|
+
while ($target.firstChild !== null) {
|
|
253
|
+
$newEl.appendChild($target.firstChild);
|
|
254
|
+
}
|
|
255
|
+
for (const attrName of $target.getAttributeNames()) {
|
|
256
|
+
const val = $target.getAttribute(attrName);
|
|
257
|
+
if (val !== null) {
|
|
258
|
+
$newEl.setAttribute(attrName, val);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
$target.replaceWith($newEl);
|
|
262
|
+
return $newEl;
|
|
263
|
+
}
|
|
264
|
+
// src/util/element-match.ts
|
|
265
|
+
function isTagName(str) {
|
|
266
|
+
for (let i = 0;i < str.length; i++) {
|
|
267
|
+
const char = str.charCodeAt(i);
|
|
268
|
+
if (!(char === 45 /* Hyphen */ || char >= 65 /* A */ && char <= 90 /* Z */ || char >= 97 /* a */ && char <= 122 /* z */)) {
|
|
269
|
+
return false;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
return true;
|
|
273
|
+
}
|
|
274
|
+
function elementMatches(element, query) {
|
|
275
|
+
if (query[0] === "#" && query.indexOf(" ") === -1 && query.indexOf(".") === -1 && query.indexOf("[") === -1 && query.indexOf(":") === -1 && query.indexOf(">") === -1) {
|
|
276
|
+
return element.id === query.slice(1);
|
|
277
|
+
}
|
|
278
|
+
if (query[0] === "." && query.indexOf(" ") === -1 && query.indexOf("#") === -1 && query.indexOf("[") === -1 && query.indexOf(":") === -1 && query.indexOf(">") === -1) {
|
|
279
|
+
return element.classList.contains(query.slice(1));
|
|
280
|
+
}
|
|
281
|
+
if (isTagName(query)) {
|
|
282
|
+
return element.tagName.toLowerCase() === query.toLowerCase();
|
|
283
|
+
}
|
|
284
|
+
if (query[0] === "[" && query[query.length - 1] === "]" && query.indexOf(" ") === -1 && query.indexOf(":") === -1) {
|
|
285
|
+
const attr = query.slice(1, -1);
|
|
286
|
+
const eqIndex = attr.indexOf("=");
|
|
287
|
+
if (eqIndex === -1) {
|
|
288
|
+
return element.hasAttribute(attr);
|
|
289
|
+
}
|
|
290
|
+
const attrName = attr.slice(0, eqIndex);
|
|
291
|
+
const attrValue = attr.slice(eqIndex + 1).replace(/^["']|["']$/g, "");
|
|
292
|
+
return element.getAttribute(attrName) === attrValue;
|
|
293
|
+
}
|
|
294
|
+
return element.matches(query);
|
|
295
|
+
}
|
|
296
|
+
// src/util/lru-cache.ts
|
|
297
|
+
class LRUCache {
|
|
298
|
+
_maxSize;
|
|
299
|
+
_data;
|
|
300
|
+
_metrics = { hits: 0, misses: 0 };
|
|
301
|
+
constructor(maxSize = 500) {
|
|
302
|
+
if (maxSize <= 0) {
|
|
303
|
+
throw new Error("LRU Cache capacity must be >= 1");
|
|
304
|
+
}
|
|
305
|
+
if (true) {
|
|
306
|
+
this._metrics = { hits: 0, misses: 0 };
|
|
307
|
+
}
|
|
308
|
+
this._maxSize = maxSize;
|
|
309
|
+
this._data = new Map;
|
|
310
|
+
}
|
|
311
|
+
_trim = () => {
|
|
312
|
+
while (this._data.size > this._maxSize) {
|
|
313
|
+
const firstKey = this._data.keys().next().value;
|
|
314
|
+
if (firstKey === undefined) {
|
|
315
|
+
throw new Error("Absurd");
|
|
316
|
+
}
|
|
317
|
+
this._data.delete(firstKey);
|
|
318
|
+
}
|
|
319
|
+
};
|
|
320
|
+
get metrics() {
|
|
321
|
+
if (true) {
|
|
322
|
+
const { hits = 0, misses = 0 } = this._metrics ?? {};
|
|
323
|
+
const lookups = hits + misses;
|
|
324
|
+
const hitRate = lookups === 0 ? 0 : hits / lookups;
|
|
325
|
+
return {
|
|
326
|
+
lookups,
|
|
327
|
+
hits,
|
|
328
|
+
misses,
|
|
329
|
+
capacity: this._maxSize,
|
|
330
|
+
entries: this._data.size,
|
|
331
|
+
hitRate
|
|
332
|
+
};
|
|
333
|
+
}
|
|
334
|
+
return {
|
|
335
|
+
lookups: 0,
|
|
336
|
+
hits: 0,
|
|
337
|
+
misses: 0,
|
|
338
|
+
capacity: 0,
|
|
339
|
+
entries: 0,
|
|
340
|
+
hitRate: 0
|
|
341
|
+
};
|
|
342
|
+
}
|
|
343
|
+
setCapacity = (maxSize) => {
|
|
344
|
+
if (maxSize <= 0 && true) {
|
|
345
|
+
console.warn("[Palette LRU Cache] Cache size is <= 0. Cache is disabled.");
|
|
346
|
+
}
|
|
347
|
+
this._maxSize = maxSize;
|
|
348
|
+
this._trim();
|
|
349
|
+
};
|
|
350
|
+
get(key) {
|
|
351
|
+
const value = this._data.get(key);
|
|
352
|
+
if (value === undefined) {
|
|
353
|
+
if (true) {
|
|
354
|
+
this._metrics.misses += 1;
|
|
355
|
+
}
|
|
356
|
+
return;
|
|
357
|
+
}
|
|
358
|
+
this._data.delete(key);
|
|
359
|
+
this._data.set(key, value);
|
|
360
|
+
if (true) {
|
|
361
|
+
this._metrics.hits += 1;
|
|
362
|
+
}
|
|
363
|
+
return value;
|
|
364
|
+
}
|
|
365
|
+
set(key, value) {
|
|
366
|
+
if (this._data.has(key)) {
|
|
367
|
+
this._data.delete(key);
|
|
368
|
+
}
|
|
369
|
+
this._data.set(key, value);
|
|
370
|
+
if (this._data.size > this._maxSize) {
|
|
371
|
+
this._trim();
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
clear() {
|
|
375
|
+
this._data.clear();
|
|
376
|
+
}
|
|
377
|
+
get size() {
|
|
378
|
+
return this._data.size;
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
var lru_cache_default = LRUCache;
|
|
382
|
+
|
|
383
|
+
// src/util/fragments.ts
|
|
384
|
+
var fragmentCache = new lru_cache_default;
|
|
385
|
+
function htmlToFragment(html) {
|
|
386
|
+
const lookup = fragmentCache.get(html);
|
|
387
|
+
if (lookup !== undefined) {
|
|
388
|
+
return lookup.cloneNode(true);
|
|
389
|
+
}
|
|
390
|
+
const temp = document.createElement("template");
|
|
391
|
+
temp.innerHTML = html;
|
|
392
|
+
fragmentCache.set(html, temp.content);
|
|
393
|
+
return temp.content.cloneNode(true);
|
|
394
|
+
}
|
|
395
|
+
// src/helper/html.ts
|
|
396
|
+
function parseTemplateStringValue(val) {
|
|
397
|
+
if (val instanceof HTMLTemplateElement) {
|
|
398
|
+
return val.innerHTML;
|
|
399
|
+
}
|
|
400
|
+
if (typeof val === "function" && (val.prototype instanceof Component || val === Component)) {
|
|
401
|
+
const tagName = val.tagName;
|
|
402
|
+
if (!isTagName(tagName)) {
|
|
403
|
+
throw new PaletteError(202 /* TEMPLATE_INVALID_COMPONENT */, val.name, val.tagName);
|
|
404
|
+
}
|
|
405
|
+
return tagName;
|
|
406
|
+
}
|
|
407
|
+
if (typeof val === "string") {
|
|
408
|
+
return `<span ::swap="${val}"></span>`;
|
|
409
|
+
}
|
|
410
|
+
throw new PaletteError(200 /* TEMPLATE_INVALID */, String(val));
|
|
411
|
+
}
|
|
412
|
+
function html(strings, ...values) {
|
|
413
|
+
const fullString = strings.reduce((acc, str, i) => {
|
|
414
|
+
const parsedValue = values[i] ? `${parseTemplateStringValue(values[i])}` : "";
|
|
415
|
+
return acc + str + parsedValue;
|
|
416
|
+
}, "");
|
|
417
|
+
const templateElement = document.createElement("template");
|
|
418
|
+
templateElement.innerHTML = fullString;
|
|
419
|
+
return templateElement;
|
|
420
|
+
}
|
|
421
|
+
// src/state/state.ts
|
|
422
|
+
var PROXY_DEPTH_WARNING_LIMIT = 4;
|
|
423
|
+
var ArrayMutatorFunctions = new Set([
|
|
424
|
+
"push",
|
|
425
|
+
"pop",
|
|
426
|
+
"shift",
|
|
427
|
+
"unshift",
|
|
428
|
+
"splice",
|
|
429
|
+
"sort",
|
|
430
|
+
"reverse",
|
|
431
|
+
"fill",
|
|
432
|
+
"copyWithin"
|
|
433
|
+
]);
|
|
434
|
+
|
|
435
|
+
class State {
|
|
436
|
+
_data;
|
|
437
|
+
_listeners = new Set;
|
|
438
|
+
_proxy;
|
|
439
|
+
_proxyCache = new WeakMap;
|
|
440
|
+
_reverseProxyCache = new WeakMap;
|
|
441
|
+
_isLocked = false;
|
|
442
|
+
_hasWarned = false;
|
|
443
|
+
constructor(initialData, onChange) {
|
|
444
|
+
this._data = initialData;
|
|
445
|
+
this.get = this.get.bind(this);
|
|
446
|
+
if (onChange) {
|
|
447
|
+
this._listeners.add(onChange);
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
_createProxy(target, path = []) {
|
|
451
|
+
if (typeof target !== "object" || target === null) {
|
|
452
|
+
return target;
|
|
453
|
+
}
|
|
454
|
+
const cached = this._proxyCache.get(target);
|
|
455
|
+
if (cached) {
|
|
456
|
+
return cached;
|
|
457
|
+
}
|
|
458
|
+
if (true) {
|
|
459
|
+
if (!this._hasWarned && path.length > PROXY_DEPTH_WARNING_LIMIT) {
|
|
460
|
+
this._hasWarned = true;
|
|
461
|
+
console.warn(`
|
|
462
|
+
[State] Deeply nested live state access may have performance impacts. For deeply
|
|
463
|
+
nested or complex state updates, use .mutate(), .patch(), or .transaction() for
|
|
464
|
+
optimal performance (Warning while evaluating live access to ${path.join(".")})
|
|
465
|
+
`);
|
|
466
|
+
}
|
|
467
|
+
}
|
|
468
|
+
const proxy = new Proxy(target, {
|
|
469
|
+
get: (obj, key) => {
|
|
470
|
+
if (typeof key === "symbol") {
|
|
471
|
+
return obj[key];
|
|
472
|
+
}
|
|
473
|
+
const value = obj[key];
|
|
474
|
+
if (Array.isArray(obj) && typeof value === "function") {
|
|
475
|
+
const isMutation = ArrayMutatorFunctions.has(key);
|
|
476
|
+
return (...args) => {
|
|
477
|
+
const snapshot = this.snapshot();
|
|
478
|
+
let current = snapshot;
|
|
479
|
+
for (const p of path) {
|
|
480
|
+
current = current[p];
|
|
481
|
+
}
|
|
482
|
+
const res = current[key](...args);
|
|
483
|
+
if (isMutation) {
|
|
484
|
+
if (this._isLocked) {
|
|
485
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
486
|
+
}
|
|
487
|
+
this._processIncomingState(snapshot);
|
|
488
|
+
}
|
|
489
|
+
return res;
|
|
490
|
+
};
|
|
491
|
+
}
|
|
492
|
+
if (typeof value === "object" && value !== null) {
|
|
493
|
+
return this._createProxy(value, [...path, key]);
|
|
494
|
+
}
|
|
495
|
+
return value;
|
|
496
|
+
},
|
|
497
|
+
set: (_obj, key, value) => {
|
|
498
|
+
if (this._isLocked) {
|
|
499
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
500
|
+
}
|
|
501
|
+
if (typeof key === "symbol") {
|
|
502
|
+
return false;
|
|
503
|
+
}
|
|
504
|
+
const newPath = [...path, key];
|
|
505
|
+
this._setNested(newPath, this._unwrapProxy(value));
|
|
506
|
+
return true;
|
|
507
|
+
},
|
|
508
|
+
has: (obj, key) => {
|
|
509
|
+
return key in obj;
|
|
510
|
+
},
|
|
511
|
+
ownKeys: (obj) => {
|
|
512
|
+
return Reflect.ownKeys(obj);
|
|
513
|
+
},
|
|
514
|
+
getOwnPropertyDescriptor: (obj, key) => {
|
|
515
|
+
const desc = Object.getOwnPropertyDescriptor(obj, key);
|
|
516
|
+
if (desc) {
|
|
517
|
+
desc.configurable = true;
|
|
518
|
+
}
|
|
519
|
+
return desc;
|
|
520
|
+
},
|
|
521
|
+
deleteProperty: (obj, key) => {
|
|
522
|
+
if (typeof key === "symbol" || !(key in obj)) {
|
|
523
|
+
return false;
|
|
524
|
+
}
|
|
525
|
+
const newPath = [...path, key];
|
|
526
|
+
this._setNested(newPath, undefined);
|
|
527
|
+
return true;
|
|
528
|
+
}
|
|
529
|
+
});
|
|
530
|
+
this._proxyCache.set(target, proxy);
|
|
531
|
+
this._reverseProxyCache.set(proxy, target);
|
|
532
|
+
return proxy;
|
|
533
|
+
}
|
|
534
|
+
_unwrapProxy(value) {
|
|
535
|
+
if (typeof value !== "object" || value === null) {
|
|
536
|
+
return value;
|
|
537
|
+
}
|
|
538
|
+
const rawTarget = this._reverseProxyCache.get(value);
|
|
539
|
+
if (rawTarget) {
|
|
540
|
+
return rawTarget;
|
|
541
|
+
}
|
|
542
|
+
if (Array.isArray(value)) {
|
|
543
|
+
return value.map((item) => this._unwrapProxy(item));
|
|
544
|
+
}
|
|
545
|
+
if (Object.getPrototypeOf(value) === Object.prototype) {
|
|
546
|
+
const unwrapped = {};
|
|
547
|
+
for (const key in value) {
|
|
548
|
+
if (Object.hasOwn(value, key)) {
|
|
549
|
+
unwrapped[key] = this._unwrapProxy(value[key]);
|
|
550
|
+
}
|
|
551
|
+
}
|
|
552
|
+
return unwrapped;
|
|
553
|
+
}
|
|
554
|
+
return value;
|
|
555
|
+
}
|
|
556
|
+
_setNested(path, value) {
|
|
557
|
+
if (!path.length) {
|
|
558
|
+
return;
|
|
559
|
+
}
|
|
560
|
+
let current = this._data;
|
|
561
|
+
for (let i = 0;i < path.length - 1; i++) {
|
|
562
|
+
const key = path[i] ?? "";
|
|
563
|
+
if (!(key in current)) {
|
|
564
|
+
throw new PaletteError(303 /* INVALID_STATE_KEY */, path.join("."));
|
|
565
|
+
}
|
|
566
|
+
current = current[key];
|
|
567
|
+
}
|
|
568
|
+
const finalKey = path[path.length - 1] ?? "";
|
|
569
|
+
if (!(finalKey in current)) {
|
|
570
|
+
throw new PaletteError(303 /* INVALID_STATE_KEY */, path.join("."));
|
|
571
|
+
}
|
|
572
|
+
if (current[finalKey] !== value && !Object.is(current[finalKey], value)) {
|
|
573
|
+
current[finalKey] = value;
|
|
574
|
+
this._emit(this.current);
|
|
575
|
+
}
|
|
576
|
+
}
|
|
577
|
+
_processIncomingState = (incomingState) => {
|
|
578
|
+
if (typeof incomingState !== "object" || incomingState === null || Array.isArray(incomingState)) {
|
|
579
|
+
throw new PaletteError(304 /* INVALID_STATE_UPDATE */, JSON.stringify(incomingState));
|
|
580
|
+
}
|
|
581
|
+
let hasChanges = false;
|
|
582
|
+
for (const [key, newValue] of Object.entries(incomingState)) {
|
|
583
|
+
const currentValue = this._data[key];
|
|
584
|
+
if (!Object.is(currentValue, newValue)) {
|
|
585
|
+
hasChanges = true;
|
|
586
|
+
this._data[key] = newValue;
|
|
587
|
+
}
|
|
588
|
+
}
|
|
589
|
+
if (hasChanges) {
|
|
590
|
+
this._emit(this._data);
|
|
591
|
+
}
|
|
592
|
+
};
|
|
593
|
+
_emit = (data) => {
|
|
594
|
+
for (const listener of this._listeners) {
|
|
595
|
+
listener(data);
|
|
596
|
+
}
|
|
597
|
+
};
|
|
598
|
+
get current() {
|
|
599
|
+
return this._data;
|
|
600
|
+
}
|
|
601
|
+
get live() {
|
|
602
|
+
if (!this._proxy) {
|
|
603
|
+
this._proxy = this._createProxy(this._data);
|
|
604
|
+
}
|
|
605
|
+
return this._proxy;
|
|
606
|
+
}
|
|
607
|
+
get isLocked() {
|
|
608
|
+
return this._isLocked;
|
|
609
|
+
}
|
|
610
|
+
addListener = (listener) => {
|
|
611
|
+
this._listeners.add(listener);
|
|
612
|
+
};
|
|
613
|
+
removeListener = (listener) => {
|
|
614
|
+
this._listeners.delete(listener);
|
|
615
|
+
};
|
|
616
|
+
get = (key) => {
|
|
617
|
+
return this._data[key];
|
|
618
|
+
};
|
|
619
|
+
snapshot = () => {
|
|
620
|
+
return structuredClone(this._data);
|
|
621
|
+
};
|
|
622
|
+
set = (key, value) => {
|
|
623
|
+
if (this._isLocked) {
|
|
624
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
625
|
+
}
|
|
626
|
+
const partial = {};
|
|
627
|
+
partial[key] = value;
|
|
628
|
+
this._processIncomingState(partial);
|
|
629
|
+
};
|
|
630
|
+
patch = (patch) => {
|
|
631
|
+
if (this._isLocked) {
|
|
632
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
633
|
+
}
|
|
634
|
+
this._processIncomingState(patch);
|
|
635
|
+
return this;
|
|
636
|
+
};
|
|
637
|
+
replace = (state) => {
|
|
638
|
+
if (this._isLocked) {
|
|
639
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
640
|
+
}
|
|
641
|
+
this._data = state;
|
|
642
|
+
this._emit(this._data);
|
|
643
|
+
return this;
|
|
644
|
+
};
|
|
645
|
+
lock = () => {
|
|
646
|
+
if (this._isLocked) {
|
|
647
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
648
|
+
}
|
|
649
|
+
this._isLocked = true;
|
|
650
|
+
return this;
|
|
651
|
+
};
|
|
652
|
+
unlock = () => {
|
|
653
|
+
this._isLocked = false;
|
|
654
|
+
return this;
|
|
655
|
+
};
|
|
656
|
+
mutate = (mutator) => {
|
|
657
|
+
if (this._isLocked) {
|
|
658
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
659
|
+
}
|
|
660
|
+
this._processIncomingState(mutator(this.snapshot()));
|
|
661
|
+
return this;
|
|
662
|
+
};
|
|
663
|
+
mutateAsync = async (mutator, lock = false) => {
|
|
664
|
+
if (this._isLocked) {
|
|
665
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
666
|
+
}
|
|
667
|
+
if (lock) {
|
|
668
|
+
this._isLocked = true;
|
|
669
|
+
}
|
|
670
|
+
const mutationResult = await mutator(this.snapshot());
|
|
671
|
+
this._processIncomingState(mutationResult);
|
|
672
|
+
if (lock) {
|
|
673
|
+
this._isLocked = false;
|
|
674
|
+
}
|
|
675
|
+
return this;
|
|
676
|
+
};
|
|
677
|
+
transaction = (fn) => {
|
|
678
|
+
if (this._isLocked) {
|
|
679
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
680
|
+
}
|
|
681
|
+
this._isLocked = true;
|
|
682
|
+
let success = true;
|
|
683
|
+
try {
|
|
684
|
+
const transactionState = new State(this.snapshot());
|
|
685
|
+
fn(transactionState);
|
|
686
|
+
this._processIncomingState(transactionState.current);
|
|
687
|
+
} catch (error) {
|
|
688
|
+
if (true) {
|
|
689
|
+
console.warn("[State] Transaction failed due to caught error", error);
|
|
690
|
+
}
|
|
691
|
+
success = false;
|
|
692
|
+
} finally {
|
|
693
|
+
this._isLocked = false;
|
|
694
|
+
}
|
|
695
|
+
return success;
|
|
696
|
+
};
|
|
697
|
+
transactionAsync = async (fn) => {
|
|
698
|
+
if (this._isLocked) {
|
|
699
|
+
throw new PaletteError(305 /* STATE_LOCKED */);
|
|
700
|
+
}
|
|
701
|
+
this._isLocked = true;
|
|
702
|
+
let success = true;
|
|
703
|
+
try {
|
|
704
|
+
const transactionState = new State(this.snapshot());
|
|
705
|
+
await fn(transactionState);
|
|
706
|
+
this._processIncomingState(transactionState.current);
|
|
707
|
+
} catch (error) {
|
|
708
|
+
if (true) {
|
|
709
|
+
console.warn("[State] Transaction failed due to caught error", error);
|
|
710
|
+
}
|
|
711
|
+
success = false;
|
|
712
|
+
} finally {
|
|
713
|
+
this._isLocked = false;
|
|
714
|
+
}
|
|
715
|
+
return success;
|
|
716
|
+
};
|
|
717
|
+
}
|
|
718
|
+
// src/template/syntax.ts
|
|
719
|
+
function isBoundAttribute(attrName) {
|
|
720
|
+
if (attrName[0] !== ":" || attrName.length < 2) {
|
|
721
|
+
return false;
|
|
722
|
+
}
|
|
723
|
+
for (let i = 1;i < attrName.length; i++) {
|
|
724
|
+
const char = attrName.charCodeAt(i);
|
|
725
|
+
if (!(char === 45 /* Hyphen */ || char >= 65 /* A */ && char <= 90 /* Z */ || char >= 97 /* a */ && char <= 122 /* z */)) {
|
|
726
|
+
return false;
|
|
727
|
+
}
|
|
728
|
+
}
|
|
729
|
+
return true;
|
|
730
|
+
}
|
|
731
|
+
function parseNotation(notation) {
|
|
732
|
+
let workingNotation = notation;
|
|
733
|
+
let notModifier = false;
|
|
734
|
+
if (notation[0] === "N" && notation[1] === "O" && notation[2] === "T" && notation[3] === ":") {
|
|
735
|
+
workingNotation = notation.slice(4);
|
|
736
|
+
notModifier = true;
|
|
737
|
+
}
|
|
738
|
+
const firstChar = workingNotation[0];
|
|
739
|
+
if (firstChar === "@" /* Attributes */) {
|
|
740
|
+
workingNotation = `attr.${workingNotation.slice(1)}`;
|
|
741
|
+
} else if (firstChar === "$" /* State */) {
|
|
742
|
+
workingNotation = `state.${workingNotation.slice(1)}`;
|
|
743
|
+
} else if (firstChar === "*" /* Data */) {
|
|
744
|
+
workingNotation = `data.${workingNotation.slice(1)}`;
|
|
745
|
+
} else if (firstChar === "#" /* Item */) {
|
|
746
|
+
workingNotation = `item.${workingNotation.slice(1)}`;
|
|
747
|
+
} else {
|
|
748
|
+
throw new PaletteError(201 /* TEMPLATE_INVALID_NOTATION */, workingNotation);
|
|
749
|
+
}
|
|
750
|
+
const path = workingNotation.split(".");
|
|
751
|
+
if (path.length < 2) {
|
|
752
|
+
throw new PaletteError(201 /* TEMPLATE_INVALID_NOTATION */, workingNotation);
|
|
753
|
+
}
|
|
754
|
+
return {
|
|
755
|
+
base: notation,
|
|
756
|
+
path,
|
|
757
|
+
modifiers: notModifier ? { not: true } : undefined
|
|
758
|
+
};
|
|
759
|
+
}
|
|
760
|
+
function buildValueAccessor(notation) {
|
|
761
|
+
const { path, modifiers } = notation;
|
|
762
|
+
if (path.length < 2) {
|
|
763
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
764
|
+
}
|
|
765
|
+
if (path.length === 2) {
|
|
766
|
+
const [firstKey, secondKey] = path;
|
|
767
|
+
if (firstKey === undefined || secondKey === undefined) {
|
|
768
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
769
|
+
}
|
|
770
|
+
return (source) => {
|
|
771
|
+
if (modifiers?.not) {
|
|
772
|
+
const value = source?.[firstKey]?.[secondKey] ?? null;
|
|
773
|
+
if (firstKey === "attr") {
|
|
774
|
+
return value !== "" && !value;
|
|
775
|
+
}
|
|
776
|
+
return !value;
|
|
777
|
+
}
|
|
778
|
+
return source?.[firstKey]?.[secondKey] ?? null;
|
|
779
|
+
};
|
|
780
|
+
}
|
|
781
|
+
if (path.length === 3) {
|
|
782
|
+
const [firstKey, secondKey, thirdKey] = path;
|
|
783
|
+
if (firstKey === undefined || secondKey === undefined || thirdKey === undefined) {
|
|
784
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
785
|
+
}
|
|
786
|
+
return (source) => {
|
|
787
|
+
if (modifiers?.not) {
|
|
788
|
+
const value = source?.[firstKey]?.[secondKey]?.[thirdKey] ?? null;
|
|
789
|
+
if (firstKey === "attr") {
|
|
790
|
+
return value !== "" && !value;
|
|
791
|
+
}
|
|
792
|
+
return !value;
|
|
793
|
+
}
|
|
794
|
+
return source?.[firstKey]?.[secondKey]?.[thirdKey] ?? null;
|
|
795
|
+
};
|
|
796
|
+
}
|
|
797
|
+
return (source) => {
|
|
798
|
+
let current = source;
|
|
799
|
+
for (const part of path) {
|
|
800
|
+
if (!current || typeof current !== "object") {
|
|
801
|
+
return null;
|
|
802
|
+
}
|
|
803
|
+
current = current[part];
|
|
804
|
+
}
|
|
805
|
+
if (modifiers?.not) {
|
|
806
|
+
if (path[0] === "attr") {
|
|
807
|
+
return current !== "" && !current;
|
|
808
|
+
}
|
|
809
|
+
return !current;
|
|
810
|
+
}
|
|
811
|
+
return current;
|
|
812
|
+
};
|
|
813
|
+
}
|
|
814
|
+
|
|
815
|
+
// src/template/build.ts
|
|
816
|
+
var REF_ATTR = "data-node-ref";
|
|
817
|
+
function collectOperableNodes(fragment) {
|
|
818
|
+
const walker = document.createTreeWalker(fragment, NodeFilter.SHOW_ELEMENT);
|
|
819
|
+
const knownIgnores = new Set;
|
|
820
|
+
const nodes = [];
|
|
821
|
+
let currentNode = walker.nextNode();
|
|
822
|
+
while (currentNode) {
|
|
823
|
+
if (!(currentNode instanceof HTMLElement)) {
|
|
824
|
+
currentNode = walker.nextNode();
|
|
825
|
+
continue;
|
|
826
|
+
}
|
|
827
|
+
let parent = currentNode.parentNode;
|
|
828
|
+
let isInList = false;
|
|
829
|
+
while (parent && !isInList) {
|
|
830
|
+
if (knownIgnores.has(parent) || parent instanceof HTMLElement && parent.hasAttribute("::each")) {
|
|
831
|
+
isInList = true;
|
|
832
|
+
knownIgnores.add(currentNode);
|
|
833
|
+
break;
|
|
834
|
+
}
|
|
835
|
+
parent = parent.parentNode;
|
|
836
|
+
}
|
|
837
|
+
if (!isInList) {
|
|
838
|
+
nodes.push(currentNode);
|
|
839
|
+
}
|
|
840
|
+
currentNode = walker.nextNode();
|
|
841
|
+
}
|
|
842
|
+
return nodes;
|
|
843
|
+
}
|
|
844
|
+
function compileTemplate(source) {
|
|
845
|
+
const parsedNotations = new Map;
|
|
846
|
+
const schemes = [];
|
|
847
|
+
let elementId = 0;
|
|
848
|
+
const ensureNodeRef = (el) => {
|
|
849
|
+
let ref = el.getAttribute(REF_ATTR);
|
|
850
|
+
if (!ref) {
|
|
851
|
+
ref = `el-${elementId++}`;
|
|
852
|
+
el.setAttribute(REF_ATTR, ref);
|
|
853
|
+
}
|
|
854
|
+
return ref;
|
|
855
|
+
};
|
|
856
|
+
const clonedTemplateElement = source.content.cloneNode(true);
|
|
857
|
+
const nodes = collectOperableNodes(clonedTemplateElement);
|
|
858
|
+
for (const node of nodes) {
|
|
859
|
+
const attributes2 = new Map;
|
|
860
|
+
for (const name of node.getAttributeNames()) {
|
|
861
|
+
attributes2.set(name, node.getAttribute(name));
|
|
862
|
+
}
|
|
863
|
+
for (const [attributeName, attributeValue] of attributes2) {
|
|
864
|
+
if (attributeValue === null || attributeName[0] !== ":") {
|
|
865
|
+
continue;
|
|
866
|
+
}
|
|
867
|
+
let notation = parsedNotations.get(attributeValue);
|
|
868
|
+
if (notation === undefined) {
|
|
869
|
+
notation = parseNotation(attributeValue);
|
|
870
|
+
parsedNotations.set(attributeValue, notation);
|
|
871
|
+
}
|
|
872
|
+
node.removeAttribute(attributeName);
|
|
873
|
+
if (attributeName === "::key") {
|
|
874
|
+
continue;
|
|
875
|
+
}
|
|
876
|
+
if (attributeName === "::tag") {
|
|
877
|
+
schemes.push({
|
|
878
|
+
type: "tag",
|
|
879
|
+
notation,
|
|
880
|
+
nodeRef: ensureNodeRef(node)
|
|
881
|
+
});
|
|
882
|
+
} else if (attributeName === "::each") {
|
|
883
|
+
const nodeRef = ensureNodeRef(node);
|
|
884
|
+
const keyNotation = node.getAttribute("::key");
|
|
885
|
+
if (keyNotation === null) {
|
|
886
|
+
throw new PaletteError(206 /* TEMPLATE_MISSING_LIST_KEY */);
|
|
887
|
+
}
|
|
888
|
+
node.removeAttribute("::key");
|
|
889
|
+
const listContentHtml = node.innerHTML;
|
|
890
|
+
while (node.firstChild) {
|
|
891
|
+
node.removeChild(node.firstChild);
|
|
892
|
+
}
|
|
893
|
+
const scheme = {
|
|
894
|
+
type: "each",
|
|
895
|
+
notation,
|
|
896
|
+
keyNotation: parseNotation(keyNotation),
|
|
897
|
+
nodeRef,
|
|
898
|
+
listContentHtml
|
|
899
|
+
};
|
|
900
|
+
schemes.push(scheme);
|
|
901
|
+
} else if (attributeName === "::swap") {
|
|
902
|
+
const scheme = {
|
|
903
|
+
type: "swap",
|
|
904
|
+
notation,
|
|
905
|
+
nodeRef: ensureNodeRef(node)
|
|
906
|
+
};
|
|
907
|
+
schemes.push(scheme);
|
|
908
|
+
} else if (isBoundAttribute(attributeName)) {
|
|
909
|
+
const bindName = attributeName.slice(1);
|
|
910
|
+
const scheme = {
|
|
911
|
+
type: "attr",
|
|
912
|
+
notation,
|
|
913
|
+
attribute: bindName,
|
|
914
|
+
nodeRef: ensureNodeRef(node)
|
|
915
|
+
};
|
|
916
|
+
schemes.push(scheme);
|
|
917
|
+
} else {
|
|
918
|
+
if (true) {
|
|
919
|
+
console.warn(`Template: ${attributeName} is not a valid directive`);
|
|
920
|
+
}
|
|
921
|
+
}
|
|
922
|
+
}
|
|
923
|
+
}
|
|
924
|
+
const notations = {};
|
|
925
|
+
for (const [key, val] of parsedNotations) {
|
|
926
|
+
notations[key] = val;
|
|
927
|
+
}
|
|
928
|
+
const temp = document.createElement("div");
|
|
929
|
+
temp.appendChild(clonedTemplateElement.cloneNode(true));
|
|
930
|
+
const html2 = temp.innerHTML;
|
|
931
|
+
const compiled = {
|
|
932
|
+
html: html2,
|
|
933
|
+
schemes,
|
|
934
|
+
notations
|
|
935
|
+
};
|
|
936
|
+
return compiled;
|
|
937
|
+
}
|
|
938
|
+
function prepareCompiledTemplate(compiled, prototypeFragment) {
|
|
939
|
+
const plansByNotation = new Map;
|
|
940
|
+
const notationAccessors = new Map;
|
|
941
|
+
const nodeRefs = new Map;
|
|
942
|
+
for (const [notation, parsed] of Object.entries(compiled.notations)) {
|
|
943
|
+
notationAccessors.set(notation, buildValueAccessor(parsed));
|
|
944
|
+
}
|
|
945
|
+
const fragment = prototypeFragment ?? htmlToFragment(compiled.html);
|
|
946
|
+
const walker = document.createTreeWalker(fragment, NodeFilter.SHOW_ELEMENT, {
|
|
947
|
+
acceptNode(node) {
|
|
948
|
+
return node.hasAttribute(REF_ATTR) ? NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_SKIP;
|
|
949
|
+
}
|
|
950
|
+
});
|
|
951
|
+
let markNode = walker.nextNode();
|
|
952
|
+
while (markNode) {
|
|
953
|
+
if (!(markNode instanceof HTMLElement)) {
|
|
954
|
+
continue;
|
|
955
|
+
}
|
|
956
|
+
const ref = markNode.getAttribute(REF_ATTR);
|
|
957
|
+
if (ref === null) {
|
|
958
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
959
|
+
}
|
|
960
|
+
nodeRefs.set(ref, markNode);
|
|
961
|
+
markNode.removeAttribute(REF_ATTR);
|
|
962
|
+
markNode = walker.nextNode();
|
|
963
|
+
}
|
|
964
|
+
const associatePlan = (plan, notation) => {
|
|
965
|
+
let notationPlans = plansByNotation.get(notation);
|
|
966
|
+
if (!notationPlans) {
|
|
967
|
+
notationPlans = new Set;
|
|
968
|
+
plansByNotation.set(notation, notationPlans);
|
|
969
|
+
}
|
|
970
|
+
notationPlans.add(plan);
|
|
971
|
+
};
|
|
972
|
+
const getByNodeRef = (nodeRef) => {
|
|
973
|
+
const cached = nodeRefs.get(nodeRef);
|
|
974
|
+
if (cached) {
|
|
975
|
+
return cached;
|
|
976
|
+
}
|
|
977
|
+
throw new PaletteError(204 /* TEMPLATE_INVALID_NODE_REF */);
|
|
978
|
+
};
|
|
979
|
+
for (const scheme of compiled.schemes) {
|
|
980
|
+
if (scheme.type === "each") {
|
|
981
|
+
const itemPrototype = getByNodeRef(scheme.nodeRef);
|
|
982
|
+
const listContainer = itemPrototype.parentElement;
|
|
983
|
+
if (!listContainer) {
|
|
984
|
+
throw new PaletteError(205 /* TEMPLATE_MISSING_LIST_PARENT */);
|
|
985
|
+
}
|
|
986
|
+
const startMarker = document.createComment("list-start");
|
|
987
|
+
const endMarker = document.createComment("list-end");
|
|
988
|
+
listContainer.insertBefore(startMarker, itemPrototype);
|
|
989
|
+
listContainer.insertBefore(endMarker, itemPrototype.nextSibling);
|
|
990
|
+
const rowRootElement = itemPrototype.cloneNode(false);
|
|
991
|
+
const templateContentSrc = htmlToFragment(scheme.listContentHtml);
|
|
992
|
+
const rowTemplateElement = document.createElement("template");
|
|
993
|
+
rowTemplateElement.appendChild(templateContentSrc);
|
|
994
|
+
itemPrototype.remove();
|
|
995
|
+
const plan = {
|
|
996
|
+
type: "each",
|
|
997
|
+
notation: scheme.notation,
|
|
998
|
+
keyNotation: scheme.keyNotation,
|
|
999
|
+
listContainerElement: listContainer,
|
|
1000
|
+
rowTemplateElement,
|
|
1001
|
+
rowRootElement,
|
|
1002
|
+
startMarker,
|
|
1003
|
+
endMarker,
|
|
1004
|
+
rowTemplatesByRoot: new WeakMap,
|
|
1005
|
+
rowRootsByKey: new Map
|
|
1006
|
+
};
|
|
1007
|
+
associatePlan(plan, scheme.notation.base);
|
|
1008
|
+
} else if (scheme.type === "swap") {
|
|
1009
|
+
const plan = {
|
|
1010
|
+
type: "swap",
|
|
1011
|
+
notation: scheme.notation,
|
|
1012
|
+
node: getByNodeRef(scheme.nodeRef)
|
|
1013
|
+
};
|
|
1014
|
+
associatePlan(plan, scheme.notation.base);
|
|
1015
|
+
} else if (scheme.type === "tag") {
|
|
1016
|
+
const plan = {
|
|
1017
|
+
type: "tag",
|
|
1018
|
+
notation: scheme.notation,
|
|
1019
|
+
node: getByNodeRef(scheme.nodeRef)
|
|
1020
|
+
};
|
|
1021
|
+
associatePlan(plan, plan.notation.base);
|
|
1022
|
+
} else if (scheme.type === "attr") {
|
|
1023
|
+
const plan = {
|
|
1024
|
+
type: "attr",
|
|
1025
|
+
attribute: scheme.attribute,
|
|
1026
|
+
notation: scheme.notation,
|
|
1027
|
+
node: getByNodeRef(scheme.nodeRef)
|
|
1028
|
+
};
|
|
1029
|
+
associatePlan(plan, plan.notation.base);
|
|
1030
|
+
}
|
|
1031
|
+
}
|
|
1032
|
+
return {
|
|
1033
|
+
plansByNotation,
|
|
1034
|
+
notationAccessors,
|
|
1035
|
+
fragment
|
|
1036
|
+
};
|
|
1037
|
+
}
|
|
1038
|
+
|
|
1039
|
+
// src/template/jobs.ts
|
|
1040
|
+
var SKIPPED_JOB = Symbol("noop");
|
|
1041
|
+
function prepareAttributeJob(plan, values) {
|
|
1042
|
+
const rawValue = values.get(plan.notation.base);
|
|
1043
|
+
let value = typeof rawValue === "function" ? rawValue(plan.node) : rawValue;
|
|
1044
|
+
if (plan.attribute === "class" && typeof value !== "string") {
|
|
1045
|
+
value = classify(value);
|
|
1046
|
+
}
|
|
1047
|
+
const serializedValue = serializeAttribute(value);
|
|
1048
|
+
const current = plan.node.getAttribute(plan.attribute);
|
|
1049
|
+
if (serializedValue !== current) {
|
|
1050
|
+
return [plan.node, plan.attribute, serializedValue];
|
|
1051
|
+
}
|
|
1052
|
+
return SKIPPED_JOB;
|
|
1053
|
+
}
|
|
1054
|
+
function runAttributeJobs(jobs) {
|
|
1055
|
+
for (const [node, name, value] of jobs) {
|
|
1056
|
+
if (value === null) {
|
|
1057
|
+
node.removeAttribute(name);
|
|
1058
|
+
} else {
|
|
1059
|
+
node.setAttribute(name, value);
|
|
1060
|
+
}
|
|
1061
|
+
}
|
|
1062
|
+
}
|
|
1063
|
+
function prepareContentSwapJob(plan, values) {
|
|
1064
|
+
const rawValue = values.get(plan.notation.base);
|
|
1065
|
+
const value = typeof rawValue === "function" ? rawValue(plan.node) : rawValue;
|
|
1066
|
+
if (typeof value === "string" || typeof value === "number" || typeof value === "boolean") {
|
|
1067
|
+
const asStr = `${value}`;
|
|
1068
|
+
if (plan.node.textContent === asStr) {
|
|
1069
|
+
return SKIPPED_JOB;
|
|
1070
|
+
}
|
|
1071
|
+
return [plan, document.createTextNode(asStr)];
|
|
1072
|
+
} else if (value === null || value === undefined) {
|
|
1073
|
+
if (plan.node instanceof Comment) {
|
|
1074
|
+
return SKIPPED_JOB;
|
|
1075
|
+
}
|
|
1076
|
+
return [plan, document.createComment("empty")];
|
|
1077
|
+
} else if (value instanceof HTMLElement) {
|
|
1078
|
+
if (Object.is(plan.node, value)) {
|
|
1079
|
+
return SKIPPED_JOB;
|
|
1080
|
+
}
|
|
1081
|
+
return [plan, value.cloneNode(true)];
|
|
1082
|
+
} else {
|
|
1083
|
+
throw new PaletteError(302 /* INVALID_CONTENT */, typeof value, JSON.stringify(value));
|
|
1084
|
+
}
|
|
1085
|
+
}
|
|
1086
|
+
function runContentSwapJobs(jobs) {
|
|
1087
|
+
for (const [plan, replace] of jobs) {
|
|
1088
|
+
const parent = plan.node.parentNode;
|
|
1089
|
+
if (!parent) {
|
|
1090
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
1091
|
+
}
|
|
1092
|
+
parent.replaceChild(replace, plan.node);
|
|
1093
|
+
plan.node = replace;
|
|
1094
|
+
}
|
|
1095
|
+
}
|
|
1096
|
+
function prepareTagChangeJob(plan, values) {
|
|
1097
|
+
const rawValue = values.get(plan.notation.base) ?? "";
|
|
1098
|
+
const value = typeof rawValue === "function" ? rawValue(plan.node) : rawValue;
|
|
1099
|
+
const nextTag = `${value}`.toLowerCase();
|
|
1100
|
+
const currentTag = plan.node.tagName.toLowerCase();
|
|
1101
|
+
if (!nextTag || nextTag === currentTag) {
|
|
1102
|
+
return SKIPPED_JOB;
|
|
1103
|
+
}
|
|
1104
|
+
return [plan, nextTag];
|
|
1105
|
+
}
|
|
1106
|
+
function runTagChangeJobs(jobs) {
|
|
1107
|
+
for (const [plan, nextTag] of jobs) {
|
|
1108
|
+
plan.node = changeTag(plan.node, nextTag);
|
|
1109
|
+
}
|
|
1110
|
+
}
|
|
1111
|
+
function fetchListData(plan, values) {
|
|
1112
|
+
const notationValue = values.get(plan.notation.base);
|
|
1113
|
+
const evaluatedValue = typeof notationValue === "function" ? notationValue(plan.startMarker.parentNode) : notationValue;
|
|
1114
|
+
if (evaluatedValue === null || evaluatedValue === undefined) {
|
|
1115
|
+
return [];
|
|
1116
|
+
}
|
|
1117
|
+
if (Array.isArray(evaluatedValue)) {
|
|
1118
|
+
return evaluatedValue;
|
|
1119
|
+
}
|
|
1120
|
+
throw new PaletteError(203 /* TEMPLATE_INVALID_LIST */);
|
|
1121
|
+
}
|
|
1122
|
+
function collectHTMLElements(start, end) {
|
|
1123
|
+
let node = start;
|
|
1124
|
+
const nodeSet = new Set;
|
|
1125
|
+
while (node && node !== end) {
|
|
1126
|
+
if (node instanceof HTMLElement) {
|
|
1127
|
+
nodeSet.add(node);
|
|
1128
|
+
}
|
|
1129
|
+
node = node.nextSibling;
|
|
1130
|
+
}
|
|
1131
|
+
return nodeSet;
|
|
1132
|
+
}
|
|
1133
|
+
function prepareListRenderJob(plan, values) {
|
|
1134
|
+
const listData = fetchListData(plan, values);
|
|
1135
|
+
if (listData.length === 0) {
|
|
1136
|
+
return [plan, []];
|
|
1137
|
+
}
|
|
1138
|
+
const { rowTemplatesByRoot, rowTemplateElement, rowRootElement } = plan;
|
|
1139
|
+
const keyAccessor = buildValueAccessor(plan.keyNotation);
|
|
1140
|
+
const listRenderContexts = new Map;
|
|
1141
|
+
for (const item of listData) {
|
|
1142
|
+
const context = {
|
|
1143
|
+
item
|
|
1144
|
+
};
|
|
1145
|
+
const key = keyAccessor(context);
|
|
1146
|
+
if (listRenderContexts.has(key)) {
|
|
1147
|
+
throw new PaletteError(207 /* TEMPLATE_DUPLICATE_LIST_KEY */, String(key));
|
|
1148
|
+
}
|
|
1149
|
+
let rowRoot = plan.rowRootsByKey.get(key);
|
|
1150
|
+
if (rowRoot === undefined) {
|
|
1151
|
+
rowRoot = rowRootElement.cloneNode(false);
|
|
1152
|
+
plan.rowRootsByKey.set(key, rowRoot);
|
|
1153
|
+
}
|
|
1154
|
+
let rowTemplate = rowTemplatesByRoot.get(rowRoot);
|
|
1155
|
+
if (rowTemplate === undefined) {
|
|
1156
|
+
rowTemplate = new Template(rowTemplateElement);
|
|
1157
|
+
rowTemplatesByRoot.set(rowRoot, rowTemplate);
|
|
1158
|
+
}
|
|
1159
|
+
const jobItem = {
|
|
1160
|
+
context,
|
|
1161
|
+
root: rowRoot,
|
|
1162
|
+
template: rowTemplate,
|
|
1163
|
+
attrMap: createAttributeMap(item)
|
|
1164
|
+
};
|
|
1165
|
+
listRenderContexts.set(key, jobItem);
|
|
1166
|
+
}
|
|
1167
|
+
for (const [key, root] of plan.rowRootsByKey) {
|
|
1168
|
+
if (!listRenderContexts.has(key)) {
|
|
1169
|
+
plan.rowRootsByKey.delete(key);
|
|
1170
|
+
plan.rowTemplatesByRoot.delete(root);
|
|
1171
|
+
}
|
|
1172
|
+
}
|
|
1173
|
+
return [plan, Array.from(listRenderContexts.values())];
|
|
1174
|
+
}
|
|
1175
|
+
function runListRenderJobs(jobs) {
|
|
1176
|
+
for (const [plan, renderItems] of jobs) {
|
|
1177
|
+
const { endMarker, listContainerElement } = plan;
|
|
1178
|
+
if (renderItems.length === 0) {
|
|
1179
|
+
for (const root of plan.rowRootsByKey.values()) {
|
|
1180
|
+
root.remove();
|
|
1181
|
+
}
|
|
1182
|
+
plan.rowRootsByKey.clear();
|
|
1183
|
+
continue;
|
|
1184
|
+
}
|
|
1185
|
+
const existingDOMNodes = Array.from(collectHTMLElements(plan.startMarker, plan.endMarker));
|
|
1186
|
+
const rootsToKeep = new Set;
|
|
1187
|
+
let existingDOMNodeIdx = 0;
|
|
1188
|
+
for (const renderItem of renderItems) {
|
|
1189
|
+
const { root, attrMap, context, template } = renderItem;
|
|
1190
|
+
const existingDOMNode = existingDOMNodes[existingDOMNodeIdx];
|
|
1191
|
+
applyAttributeMap(root, attrMap);
|
|
1192
|
+
template.render(root, context);
|
|
1193
|
+
rootsToKeep.add(root);
|
|
1194
|
+
if (Object.is(existingDOMNode, root)) {
|
|
1195
|
+
existingDOMNodeIdx += 1;
|
|
1196
|
+
continue;
|
|
1197
|
+
}
|
|
1198
|
+
if (!existingDOMNode) {
|
|
1199
|
+
listContainerElement.insertBefore(root, endMarker);
|
|
1200
|
+
continue;
|
|
1201
|
+
}
|
|
1202
|
+
listContainerElement.insertBefore(root, existingDOMNode);
|
|
1203
|
+
}
|
|
1204
|
+
for (const root of existingDOMNodes) {
|
|
1205
|
+
if (!rootsToKeep.has(root)) {
|
|
1206
|
+
root.remove();
|
|
1207
|
+
plan.rowTemplatesByRoot.delete(root);
|
|
1208
|
+
}
|
|
1209
|
+
}
|
|
1210
|
+
}
|
|
1211
|
+
}
|
|
1212
|
+
|
|
1213
|
+
// src/template/template.ts
|
|
1214
|
+
var compileCache = new LRUCache;
|
|
1215
|
+
if (typeof window !== "undefined") {
|
|
1216
|
+
window.__PALETTE_DEVTOOLS__ = {
|
|
1217
|
+
dumpCacheMetrics: () => {
|
|
1218
|
+
console.log("Compiled Templates Cache", compileCache.metrics);
|
|
1219
|
+
console.log("HTML Fragments Cache", fragmentCache.metrics);
|
|
1220
|
+
}
|
|
1221
|
+
};
|
|
1222
|
+
}
|
|
1223
|
+
|
|
1224
|
+
class Template {
|
|
1225
|
+
static cache = {
|
|
1226
|
+
clear: () => {
|
|
1227
|
+
fragmentCache.clear();
|
|
1228
|
+
compileCache.clear();
|
|
1229
|
+
},
|
|
1230
|
+
setCapacity: (maxSize) => {
|
|
1231
|
+
fragmentCache.setCapacity(maxSize);
|
|
1232
|
+
compileCache.setCapacity(maxSize);
|
|
1233
|
+
}
|
|
1234
|
+
};
|
|
1235
|
+
_compiled;
|
|
1236
|
+
_templateElement;
|
|
1237
|
+
_mountedRoot;
|
|
1238
|
+
_latestRenderedValues = new Map;
|
|
1239
|
+
_fragment;
|
|
1240
|
+
_plansByNotation = new Map;
|
|
1241
|
+
_notationAccessors = new Map;
|
|
1242
|
+
constructor(template) {
|
|
1243
|
+
if (template instanceof HTMLTemplateElement) {
|
|
1244
|
+
this._templateElement = template;
|
|
1245
|
+
const cached = compileCache.get(template);
|
|
1246
|
+
const compiled = cached ?? compileTemplate(template);
|
|
1247
|
+
this._compiled = compiled;
|
|
1248
|
+
compileCache.set(template, compiled);
|
|
1249
|
+
} else {
|
|
1250
|
+
this._compiled = template;
|
|
1251
|
+
}
|
|
1252
|
+
const fragment = htmlToFragment(this._compiled.html);
|
|
1253
|
+
const hydrated = prepareCompiledTemplate(this._compiled, fragment);
|
|
1254
|
+
this._plansByNotation = hydrated.plansByNotation;
|
|
1255
|
+
this._notationAccessors = hydrated.notationAccessors;
|
|
1256
|
+
this._fragment = hydrated.fragment;
|
|
1257
|
+
}
|
|
1258
|
+
clone = () => {
|
|
1259
|
+
return new Template(this.element());
|
|
1260
|
+
};
|
|
1261
|
+
element = () => {
|
|
1262
|
+
if (this._templateElement) {
|
|
1263
|
+
return this._templateElement;
|
|
1264
|
+
}
|
|
1265
|
+
const template = document.createElement("template");
|
|
1266
|
+
template.innerHTML = this._compiled.html;
|
|
1267
|
+
return template;
|
|
1268
|
+
};
|
|
1269
|
+
_collectUpdatedValues = (context) => {
|
|
1270
|
+
const updatedValues = new Map;
|
|
1271
|
+
for (const [rawNotation, accessor] of this._notationAccessors) {
|
|
1272
|
+
const previousValue = this._latestRenderedValues.get(rawNotation);
|
|
1273
|
+
const newValue = accessor(context);
|
|
1274
|
+
if (previousValue !== newValue && !Object.is(previousValue, newValue)) {
|
|
1275
|
+
updatedValues.set(rawNotation, newValue);
|
|
1276
|
+
}
|
|
1277
|
+
}
|
|
1278
|
+
return updatedValues;
|
|
1279
|
+
};
|
|
1280
|
+
_update(newValues) {
|
|
1281
|
+
const scheduledPlans = new Set;
|
|
1282
|
+
const attributeJobs = [];
|
|
1283
|
+
const contentSwapJobs = [];
|
|
1284
|
+
const tagChangeJobs = [];
|
|
1285
|
+
const listRenderJobs = [];
|
|
1286
|
+
for (const [notation] of newValues) {
|
|
1287
|
+
const affectedPlans = this._plansByNotation.get(notation);
|
|
1288
|
+
if (affectedPlans) {
|
|
1289
|
+
for (const plan of affectedPlans) {
|
|
1290
|
+
if (scheduledPlans.has(plan)) {
|
|
1291
|
+
continue;
|
|
1292
|
+
}
|
|
1293
|
+
scheduledPlans.add(plan);
|
|
1294
|
+
switch (plan.type) {
|
|
1295
|
+
case "attr": {
|
|
1296
|
+
const job = prepareAttributeJob(plan, newValues);
|
|
1297
|
+
if (job !== SKIPPED_JOB) {
|
|
1298
|
+
attributeJobs.push(job);
|
|
1299
|
+
}
|
|
1300
|
+
break;
|
|
1301
|
+
}
|
|
1302
|
+
case "each": {
|
|
1303
|
+
const job = prepareListRenderJob(plan, newValues);
|
|
1304
|
+
if (job !== SKIPPED_JOB) {
|
|
1305
|
+
listRenderJobs.push(job);
|
|
1306
|
+
}
|
|
1307
|
+
break;
|
|
1308
|
+
}
|
|
1309
|
+
case "swap": {
|
|
1310
|
+
const job = prepareContentSwapJob(plan, newValues);
|
|
1311
|
+
if (job !== SKIPPED_JOB) {
|
|
1312
|
+
contentSwapJobs.push(job);
|
|
1313
|
+
}
|
|
1314
|
+
break;
|
|
1315
|
+
}
|
|
1316
|
+
case "tag": {
|
|
1317
|
+
const job = prepareTagChangeJob(plan, newValues);
|
|
1318
|
+
if (job !== SKIPPED_JOB) {
|
|
1319
|
+
tagChangeJobs.push(job);
|
|
1320
|
+
}
|
|
1321
|
+
break;
|
|
1322
|
+
}
|
|
1323
|
+
default:
|
|
1324
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
1325
|
+
}
|
|
1326
|
+
}
|
|
1327
|
+
}
|
|
1328
|
+
}
|
|
1329
|
+
runAttributeJobs(attributeJobs);
|
|
1330
|
+
runListRenderJobs(listRenderJobs);
|
|
1331
|
+
runContentSwapJobs(contentSwapJobs);
|
|
1332
|
+
runTagChangeJobs(tagChangeJobs);
|
|
1333
|
+
}
|
|
1334
|
+
render = (root, context) => {
|
|
1335
|
+
if (root !== this._mountedRoot) {
|
|
1336
|
+
this._mountedRoot = root;
|
|
1337
|
+
while (root.firstChild) {
|
|
1338
|
+
root.removeChild(root.firstChild);
|
|
1339
|
+
}
|
|
1340
|
+
root.append(this._fragment);
|
|
1341
|
+
}
|
|
1342
|
+
const resolvedValues = this._collectUpdatedValues(context);
|
|
1343
|
+
if (resolvedValues.size !== 0) {
|
|
1344
|
+
this._update(resolvedValues);
|
|
1345
|
+
for (const [key, val] of resolvedValues) {
|
|
1346
|
+
this._latestRenderedValues.set(key, val);
|
|
1347
|
+
}
|
|
1348
|
+
}
|
|
1349
|
+
};
|
|
1350
|
+
}
|
|
1351
|
+
// src/component/delegator.ts
|
|
1352
|
+
class EventDelegator {
|
|
1353
|
+
_root;
|
|
1354
|
+
_rootListeners = new Map;
|
|
1355
|
+
_events = new Map;
|
|
1356
|
+
constructor(root = document) {
|
|
1357
|
+
this._root = root;
|
|
1358
|
+
}
|
|
1359
|
+
_handlersForEvent(event) {
|
|
1360
|
+
let handlerMap = this._events.get(event);
|
|
1361
|
+
if (handlerMap === undefined) {
|
|
1362
|
+
handlerMap = new Map;
|
|
1363
|
+
this._events.set(event, handlerMap);
|
|
1364
|
+
}
|
|
1365
|
+
return handlerMap;
|
|
1366
|
+
}
|
|
1367
|
+
_handlersForEventAndQuery(event, query) {
|
|
1368
|
+
const handlerMap = this._handlersForEvent(event);
|
|
1369
|
+
let handlerSet = handlerMap.get(query);
|
|
1370
|
+
if (handlerSet === undefined) {
|
|
1371
|
+
handlerSet = new Set;
|
|
1372
|
+
handlerMap.set(query, handlerSet);
|
|
1373
|
+
}
|
|
1374
|
+
return handlerSet;
|
|
1375
|
+
}
|
|
1376
|
+
_createRootListener(eventName) {
|
|
1377
|
+
if (this._rootListeners.has(eventName)) {
|
|
1378
|
+
return;
|
|
1379
|
+
}
|
|
1380
|
+
const listener = (event) => {
|
|
1381
|
+
const handlersForEvent = this._handlersForEvent(eventName);
|
|
1382
|
+
const { target } = event;
|
|
1383
|
+
if (target === null || !(target instanceof Node)) {
|
|
1384
|
+
return;
|
|
1385
|
+
}
|
|
1386
|
+
let node = target;
|
|
1387
|
+
let stopImmediateCalled = false;
|
|
1388
|
+
const originalFn = event.stopImmediatePropagation;
|
|
1389
|
+
event.stopImmediatePropagation = () => {
|
|
1390
|
+
originalFn.call(event);
|
|
1391
|
+
stopImmediateCalled = true;
|
|
1392
|
+
};
|
|
1393
|
+
while (node) {
|
|
1394
|
+
if (node === this._root) {
|
|
1395
|
+
break;
|
|
1396
|
+
}
|
|
1397
|
+
if (!(node instanceof Element)) {
|
|
1398
|
+
node = node.parentNode;
|
|
1399
|
+
continue;
|
|
1400
|
+
}
|
|
1401
|
+
for (const [query, handlers] of handlersForEvent) {
|
|
1402
|
+
if (query === "*" || elementMatches(node, query)) {
|
|
1403
|
+
for (const handler of handlers) {
|
|
1404
|
+
handler.call(node, event);
|
|
1405
|
+
if (stopImmediateCalled || event.defaultPrevented) {
|
|
1406
|
+
return;
|
|
1407
|
+
}
|
|
1408
|
+
}
|
|
1409
|
+
}
|
|
1410
|
+
}
|
|
1411
|
+
if (event.cancelBubble) {
|
|
1412
|
+
return;
|
|
1413
|
+
}
|
|
1414
|
+
node = node.parentNode;
|
|
1415
|
+
if (node === this._root || node instanceof ShadowRoot) {
|
|
1416
|
+
break;
|
|
1417
|
+
}
|
|
1418
|
+
}
|
|
1419
|
+
};
|
|
1420
|
+
this._rootListeners.set(eventName, listener);
|
|
1421
|
+
this._root.addEventListener(eventName, listener);
|
|
1422
|
+
}
|
|
1423
|
+
addListener(eventName, handler, queryFilter = "*") {
|
|
1424
|
+
this._handlersForEventAndQuery(eventName, queryFilter).add(handler);
|
|
1425
|
+
this._createRootListener(eventName);
|
|
1426
|
+
}
|
|
1427
|
+
removeListener(eventName, handler, queryFilter) {
|
|
1428
|
+
const handlerSet = this._handlersForEventAndQuery(eventName, queryFilter);
|
|
1429
|
+
handlerSet.delete(handler);
|
|
1430
|
+
if (handlerSet.size === 0) {
|
|
1431
|
+
const eventHandlerSets = this._handlersForEvent(eventName);
|
|
1432
|
+
eventHandlerSets.delete(queryFilter);
|
|
1433
|
+
}
|
|
1434
|
+
}
|
|
1435
|
+
cleanup = () => {
|
|
1436
|
+
for (const [event, handler] of this._rootListeners) {
|
|
1437
|
+
this._root.removeEventListener(event, handler);
|
|
1438
|
+
}
|
|
1439
|
+
this._rootListeners.clear();
|
|
1440
|
+
};
|
|
1441
|
+
}
|
|
1442
|
+
|
|
1443
|
+
// src/component/component.ts
|
|
1444
|
+
var MAX_SEQUENTIAL_RENDERS = 100;
|
|
1445
|
+
var BLANK_TEMPLATE = document.createElement("template");
|
|
1446
|
+
|
|
1447
|
+
class Component extends HTMLElement {
|
|
1448
|
+
static register(component, tagName) {
|
|
1449
|
+
if (typeof window === "undefined" || typeof window.customElements === "undefined") {
|
|
1450
|
+
throw new PaletteError(1 /* INVALID_ENV */);
|
|
1451
|
+
}
|
|
1452
|
+
const componentTagname = "tagName" in component && typeof component.tagName === "string" ? component.tagName : null;
|
|
1453
|
+
const computedTagname = tagName ?? componentTagname;
|
|
1454
|
+
if (computedTagname === null || computedTagname.length === 0 || !computedTagname.includes("-")) {
|
|
1455
|
+
throw new PaletteError(101 /* INVALID_TAGNAME */);
|
|
1456
|
+
}
|
|
1457
|
+
if (window.customElements.get(computedTagname)) {
|
|
1458
|
+
throw new PaletteError(100 /* DUPE_TAGNAME */, tagName ?? "");
|
|
1459
|
+
}
|
|
1460
|
+
window.customElements.define(computedTagname, component);
|
|
1461
|
+
}
|
|
1462
|
+
static tagName = "";
|
|
1463
|
+
static template = BLANK_TEMPLATE;
|
|
1464
|
+
static styles = [];
|
|
1465
|
+
static observedAttributes = [];
|
|
1466
|
+
static shadowRootMode = "closed";
|
|
1467
|
+
initialState;
|
|
1468
|
+
computedProperties;
|
|
1469
|
+
liveAttributes;
|
|
1470
|
+
_isComponentMounted = false;
|
|
1471
|
+
_template;
|
|
1472
|
+
_root;
|
|
1473
|
+
_state;
|
|
1474
|
+
_delegator;
|
|
1475
|
+
_ownListeners = new Set;
|
|
1476
|
+
_liveAttributeConfigs = new Map;
|
|
1477
|
+
_nextListenerAutoId = 0;
|
|
1478
|
+
_cleanupFn;
|
|
1479
|
+
_renderInternals = {
|
|
1480
|
+
willRender: false,
|
|
1481
|
+
isRendering: false,
|
|
1482
|
+
sequentialRenders: 0,
|
|
1483
|
+
postRenderCallbacks: new Set,
|
|
1484
|
+
lastRenderedAttributeMap: new Map,
|
|
1485
|
+
trackedAttributeChanges: new Map
|
|
1486
|
+
};
|
|
1487
|
+
constructor() {
|
|
1488
|
+
super();
|
|
1489
|
+
const { template: template2, styles, shadowRootMode } = this.constructor;
|
|
1490
|
+
const newStyles = Array.isArray(styles) ? styles : typeof styles === "object" ? [styles] : [css`${styles}`];
|
|
1491
|
+
this._root = this.attachShadow({ mode: shadowRootMode });
|
|
1492
|
+
this._root.adoptedStyleSheets.push(...newStyles);
|
|
1493
|
+
this._template = template2 instanceof Template ? template2 : new Template(template2);
|
|
1494
|
+
}
|
|
1495
|
+
_getRenderContext(attr) {
|
|
1496
|
+
const data = typeof this.computedProperties === "function" ? this.computedProperties() : this.computedProperties;
|
|
1497
|
+
const state = this.initialState ? this._safeGetState().current : undefined;
|
|
1498
|
+
return {
|
|
1499
|
+
data,
|
|
1500
|
+
attr: Object.fromEntries(attr),
|
|
1501
|
+
state
|
|
1502
|
+
};
|
|
1503
|
+
}
|
|
1504
|
+
_safeGetState() {
|
|
1505
|
+
if (!this.initialState || !this._state) {
|
|
1506
|
+
throw new PaletteError(301 /* MISSING_STATE */);
|
|
1507
|
+
}
|
|
1508
|
+
return this._state;
|
|
1509
|
+
}
|
|
1510
|
+
_reportError(error) {
|
|
1511
|
+
if (!this.onError) {
|
|
1512
|
+
throw error;
|
|
1513
|
+
}
|
|
1514
|
+
this.onError(error);
|
|
1515
|
+
}
|
|
1516
|
+
_executeAttributeChangeHandlers = () => {
|
|
1517
|
+
const prevAttrs = this._renderInternals.lastRenderedAttributeMap;
|
|
1518
|
+
const trackedChanges = new Map(this._renderInternals.trackedAttributeChanges);
|
|
1519
|
+
this._renderInternals.trackedAttributeChanges.clear();
|
|
1520
|
+
for (const [name, val] of trackedChanges) {
|
|
1521
|
+
const config = this._liveAttributeConfigs.get(name);
|
|
1522
|
+
if (!config) {
|
|
1523
|
+
continue;
|
|
1524
|
+
}
|
|
1525
|
+
if ("onChange" in config) {
|
|
1526
|
+
const previousValue = prevAttrs.get(name) ?? null;
|
|
1527
|
+
if (config && previousValue !== val) {
|
|
1528
|
+
config.onChange?.call(this, val, previousValue);
|
|
1529
|
+
}
|
|
1530
|
+
}
|
|
1531
|
+
}
|
|
1532
|
+
};
|
|
1533
|
+
_reflectLiveAttributes = () => {
|
|
1534
|
+
const toSet = [];
|
|
1535
|
+
for (const [name, config] of this._liveAttributeConfigs) {
|
|
1536
|
+
if ("reflect" in config) {
|
|
1537
|
+
const val = serializeAttribute(config.reflect?.call(this));
|
|
1538
|
+
const current = this.getAttribute(name);
|
|
1539
|
+
if (val !== current) {
|
|
1540
|
+
toSet.push([name, val]);
|
|
1541
|
+
}
|
|
1542
|
+
}
|
|
1543
|
+
}
|
|
1544
|
+
for (const [name, val] of toSet) {
|
|
1545
|
+
this.reflectAttribute(name, val);
|
|
1546
|
+
}
|
|
1547
|
+
};
|
|
1548
|
+
_performUpdate = () => {
|
|
1549
|
+
try {
|
|
1550
|
+
this._renderInternals.isRendering = true;
|
|
1551
|
+
this._renderInternals.willRender = false;
|
|
1552
|
+
this._executeAttributeChangeHandlers();
|
|
1553
|
+
const prevAttrs = this._renderInternals.lastRenderedAttributeMap;
|
|
1554
|
+
this.beforeUpdate?.call(this);
|
|
1555
|
+
const newAttributes = createAttributeMap(this);
|
|
1556
|
+
this._template.render(this.root, this._getRenderContext(newAttributes));
|
|
1557
|
+
this._renderInternals.lastRenderedAttributeMap = newAttributes;
|
|
1558
|
+
this.afterUpdate?.call(this, prevAttrs);
|
|
1559
|
+
this._reflectLiveAttributes();
|
|
1560
|
+
if (this._renderInternals.postRenderCallbacks.size) {
|
|
1561
|
+
const callbacks = Array.from(this._renderInternals.postRenderCallbacks);
|
|
1562
|
+
this._renderInternals.postRenderCallbacks.clear();
|
|
1563
|
+
for (const fn of callbacks) {
|
|
1564
|
+
fn();
|
|
1565
|
+
}
|
|
1566
|
+
}
|
|
1567
|
+
if (this._renderInternals.willRender) {
|
|
1568
|
+
this._renderInternals.sequentialRenders += 1;
|
|
1569
|
+
if (this._renderInternals.sequentialRenders >= MAX_SEQUENTIAL_RENDERS) {
|
|
1570
|
+
throw new PaletteError(306 /* MAX_SEQUENTIAL_RENDERS */, `${this._renderInternals.sequentialRenders}`);
|
|
1571
|
+
}
|
|
1572
|
+
this._performUpdate();
|
|
1573
|
+
} else {
|
|
1574
|
+
this._renderInternals.sequentialRenders = 0;
|
|
1575
|
+
}
|
|
1576
|
+
} catch (error) {
|
|
1577
|
+
this._renderInternals.willRender = false;
|
|
1578
|
+
this._reportError(error);
|
|
1579
|
+
} finally {
|
|
1580
|
+
this._renderInternals.isRendering = false;
|
|
1581
|
+
}
|
|
1582
|
+
};
|
|
1583
|
+
_scheduleUpdate = () => {
|
|
1584
|
+
if (this._renderInternals.willRender) {
|
|
1585
|
+
return;
|
|
1586
|
+
}
|
|
1587
|
+
this._renderInternals.willRender = true;
|
|
1588
|
+
if (this._renderInternals.isRendering) {
|
|
1589
|
+
return;
|
|
1590
|
+
}
|
|
1591
|
+
requestAnimationFrame(this._performUpdate);
|
|
1592
|
+
};
|
|
1593
|
+
_adoptState(state) {
|
|
1594
|
+
if (this._state) {
|
|
1595
|
+
this._state.removeListener(this._scheduleUpdate);
|
|
1596
|
+
this._scheduleUpdate();
|
|
1597
|
+
}
|
|
1598
|
+
const incomingStateObj = state instanceof State ? state : new State(state);
|
|
1599
|
+
this._state = incomingStateObj;
|
|
1600
|
+
this._state.removeListener(this._scheduleUpdate);
|
|
1601
|
+
this._state.addListener(this._scheduleUpdate);
|
|
1602
|
+
}
|
|
1603
|
+
connectedCallback() {
|
|
1604
|
+
if (this._isComponentMounted) {
|
|
1605
|
+
return;
|
|
1606
|
+
}
|
|
1607
|
+
try {
|
|
1608
|
+
const attributes2 = createAttributeMap(this);
|
|
1609
|
+
this._liveAttributeConfigs = new Map(Object.entries(this.liveAttributes ?? {}));
|
|
1610
|
+
if (this.initialState) {
|
|
1611
|
+
const init = typeof this.initialState === "object" ? this.initialState : this.initialState();
|
|
1612
|
+
this._adoptState(init);
|
|
1613
|
+
}
|
|
1614
|
+
this._template.render(this.root, this._getRenderContext(attributes2));
|
|
1615
|
+
this._reflectLiveAttributes();
|
|
1616
|
+
this._renderInternals.lastRenderedAttributeMap = attributes2;
|
|
1617
|
+
this._isComponentMounted = true;
|
|
1618
|
+
this._cleanupFn = this.script?.call(this) ?? undefined;
|
|
1619
|
+
} catch (error) {
|
|
1620
|
+
this._reportError(error);
|
|
1621
|
+
}
|
|
1622
|
+
}
|
|
1623
|
+
disconnectedCallback() {
|
|
1624
|
+
queueMicrotask(() => {
|
|
1625
|
+
if (!this.isConnected && this._isComponentMounted) {
|
|
1626
|
+
try {
|
|
1627
|
+
this._delegator?.cleanup();
|
|
1628
|
+
if (this._ownListeners.size > 0) {
|
|
1629
|
+
for (const listener of this._ownListeners) {
|
|
1630
|
+
this.removeEventListener(listener.event, listener.handler);
|
|
1631
|
+
}
|
|
1632
|
+
}
|
|
1633
|
+
this._cleanupFn?.call(this);
|
|
1634
|
+
this._isComponentMounted = false;
|
|
1635
|
+
} catch (error) {
|
|
1636
|
+
this._reportError(error);
|
|
1637
|
+
} finally {
|
|
1638
|
+
try {
|
|
1639
|
+
this.finalize?.call(this);
|
|
1640
|
+
} catch (error) {
|
|
1641
|
+
this._reportError(error);
|
|
1642
|
+
}
|
|
1643
|
+
}
|
|
1644
|
+
}
|
|
1645
|
+
});
|
|
1646
|
+
}
|
|
1647
|
+
attributeChangedCallback(name, _, newValue) {
|
|
1648
|
+
if (this._renderInternals.isRendering) {
|
|
1649
|
+
return;
|
|
1650
|
+
}
|
|
1651
|
+
if (newValue === this._renderInternals.lastRenderedAttributeMap?.get(name)) {
|
|
1652
|
+
return;
|
|
1653
|
+
}
|
|
1654
|
+
this._renderInternals.trackedAttributeChanges.set(name, newValue);
|
|
1655
|
+
if (!this._renderInternals.willRender) {
|
|
1656
|
+
this._scheduleUpdate();
|
|
1657
|
+
}
|
|
1658
|
+
}
|
|
1659
|
+
get root() {
|
|
1660
|
+
return this._root;
|
|
1661
|
+
}
|
|
1662
|
+
get state() {
|
|
1663
|
+
return this._safeGetState().live;
|
|
1664
|
+
}
|
|
1665
|
+
set state(newState) {
|
|
1666
|
+
if (!this._state) {
|
|
1667
|
+
throw new PaletteError(301 /* MISSING_STATE */);
|
|
1668
|
+
} else if (newState instanceof State) {
|
|
1669
|
+
this._state = newState;
|
|
1670
|
+
this._state.addListener(() => {
|
|
1671
|
+
this.requestRender();
|
|
1672
|
+
});
|
|
1673
|
+
this.requestRender();
|
|
1674
|
+
} else {
|
|
1675
|
+
this._state.replace(newState);
|
|
1676
|
+
}
|
|
1677
|
+
}
|
|
1678
|
+
getState() {
|
|
1679
|
+
return this._safeGetState();
|
|
1680
|
+
}
|
|
1681
|
+
requestRender(callback) {
|
|
1682
|
+
if (callback) {
|
|
1683
|
+
this._renderInternals.postRenderCallbacks.add(callback);
|
|
1684
|
+
}
|
|
1685
|
+
this._scheduleUpdate();
|
|
1686
|
+
}
|
|
1687
|
+
listen(targetDescriptor, eventName, eventHandler) {
|
|
1688
|
+
let query = "";
|
|
1689
|
+
if (targetDescriptor instanceof HTMLElement) {
|
|
1690
|
+
if (targetDescriptor.id.length) {
|
|
1691
|
+
query = `#${targetDescriptor.id}`;
|
|
1692
|
+
} else {
|
|
1693
|
+
const generatedId = `listen-${this._nextListenerAutoId++}`;
|
|
1694
|
+
targetDescriptor.setAttribute("data-listener-id", generatedId);
|
|
1695
|
+
query = `[data-listener-id="${generatedId}"]`;
|
|
1696
|
+
}
|
|
1697
|
+
} else if (typeof targetDescriptor === "function") {
|
|
1698
|
+
query = targetDescriptor.tagName;
|
|
1699
|
+
if (true) {
|
|
1700
|
+
if (targetDescriptor.tagName.length === 0) {
|
|
1701
|
+
console.warn(`[Component] Automatic listener setup failure: Component ${targetDescriptor.name} has no defined tagname`);
|
|
1702
|
+
}
|
|
1703
|
+
}
|
|
1704
|
+
} else {
|
|
1705
|
+
query = targetDescriptor;
|
|
1706
|
+
}
|
|
1707
|
+
if (query === ":host" || query === ":HOST") {
|
|
1708
|
+
const handler = eventHandler.bind(this);
|
|
1709
|
+
this.addEventListener(eventName, handler);
|
|
1710
|
+
this._ownListeners.add({
|
|
1711
|
+
handler,
|
|
1712
|
+
event: eventName
|
|
1713
|
+
});
|
|
1714
|
+
return;
|
|
1715
|
+
}
|
|
1716
|
+
if (!this._delegator) {
|
|
1717
|
+
this._delegator = new EventDelegator(this.root);
|
|
1718
|
+
}
|
|
1719
|
+
this._delegator.addListener(eventName, eventHandler.bind(this), query);
|
|
1720
|
+
}
|
|
1721
|
+
dispatchEvent(event, detail, options) {
|
|
1722
|
+
if (event instanceof Event) {
|
|
1723
|
+
return super.dispatchEvent(event);
|
|
1724
|
+
}
|
|
1725
|
+
const opts = {
|
|
1726
|
+
bubbles: true,
|
|
1727
|
+
cancelable: true,
|
|
1728
|
+
composed: true,
|
|
1729
|
+
...options ?? {}
|
|
1730
|
+
};
|
|
1731
|
+
return super.dispatchEvent(new CustomEvent(event, {
|
|
1732
|
+
detail,
|
|
1733
|
+
...opts
|
|
1734
|
+
}));
|
|
1735
|
+
}
|
|
1736
|
+
querySelector(query) {
|
|
1737
|
+
return this._root.querySelector(query);
|
|
1738
|
+
}
|
|
1739
|
+
querySelectorAll(query) {
|
|
1740
|
+
return this._root.querySelectorAll(query);
|
|
1741
|
+
}
|
|
1742
|
+
getElementById(id) {
|
|
1743
|
+
if (this._root instanceof ShadowRoot) {
|
|
1744
|
+
return this._root.getElementById(id);
|
|
1745
|
+
} else {
|
|
1746
|
+
return HTMLElement.prototype.querySelector.call(this, `#${id}`);
|
|
1747
|
+
}
|
|
1748
|
+
}
|
|
1749
|
+
requireElementById(id) {
|
|
1750
|
+
const lookup = this.getElementById(id);
|
|
1751
|
+
if (!lookup) {
|
|
1752
|
+
throw new PaletteError(300 /* MISSING_ELEMENT */);
|
|
1753
|
+
}
|
|
1754
|
+
return lookup;
|
|
1755
|
+
}
|
|
1756
|
+
setAttribute(name, value) {
|
|
1757
|
+
const serialized = serializeAttribute(value);
|
|
1758
|
+
if (serialized === null) {
|
|
1759
|
+
HTMLElement.prototype.removeAttribute.call(this, name);
|
|
1760
|
+
} else {
|
|
1761
|
+
HTMLElement.prototype.setAttribute.call(this, name, serialized);
|
|
1762
|
+
}
|
|
1763
|
+
}
|
|
1764
|
+
getAttribute(name, defaultValue) {
|
|
1765
|
+
const value = HTMLElement.prototype.getAttribute.call(this, name);
|
|
1766
|
+
if (defaultValue === undefined) {
|
|
1767
|
+
return value;
|
|
1768
|
+
}
|
|
1769
|
+
if (typeof defaultValue === "number") {
|
|
1770
|
+
if (value === null) {
|
|
1771
|
+
return defaultValue;
|
|
1772
|
+
}
|
|
1773
|
+
const parsed = Number(value);
|
|
1774
|
+
if (Number.isNaN(parsed)) {
|
|
1775
|
+
return defaultValue;
|
|
1776
|
+
}
|
|
1777
|
+
return parsed;
|
|
1778
|
+
}
|
|
1779
|
+
throw new PaletteError(0 /* INVARIANT */);
|
|
1780
|
+
}
|
|
1781
|
+
reflectAttribute(name, value) {
|
|
1782
|
+
const current = this.getAttribute(name);
|
|
1783
|
+
const reflected = serializeAttribute(value);
|
|
1784
|
+
if (current !== reflected) {
|
|
1785
|
+
this.setAttribute(name, reflected);
|
|
1786
|
+
}
|
|
1787
|
+
}
|
|
1788
|
+
toString(full = false) {
|
|
1789
|
+
const name = this.constructor.name;
|
|
1790
|
+
const attrString = this.getAttributeNames().map((attrName) => {
|
|
1791
|
+
const val = this.getAttribute(attrName);
|
|
1792
|
+
return val ? `${attrName}="${val}"` : "";
|
|
1793
|
+
}).join(" ");
|
|
1794
|
+
const attrs = attrString.length > 0 ? ` ${attrString}` : "";
|
|
1795
|
+
if (full) {
|
|
1796
|
+
return `Component<${name}${attrs}>
|
|
1797
|
+
${this.root.innerHTML}
|
|
1798
|
+
</${name}>`;
|
|
1799
|
+
}
|
|
1800
|
+
return `Component<${name}${attrs}>`;
|
|
1801
|
+
}
|
|
1802
|
+
}
|
|
1803
|
+
// src/component/factory.ts
|
|
1804
|
+
var BLANK_TEMPLATE2 = document.createElement("template");
|
|
1805
|
+
function define(tagname, definition) {
|
|
1806
|
+
const {
|
|
1807
|
+
template: template2 = BLANK_TEMPLATE2,
|
|
1808
|
+
styles = [],
|
|
1809
|
+
shadowRootMode = "closed",
|
|
1810
|
+
initialState,
|
|
1811
|
+
computedProperties = {},
|
|
1812
|
+
liveAttributes = {},
|
|
1813
|
+
script,
|
|
1814
|
+
beforeUpdate,
|
|
1815
|
+
afterUpdate,
|
|
1816
|
+
finalize,
|
|
1817
|
+
onError
|
|
1818
|
+
} = definition;
|
|
1819
|
+
const fullObservedAttributes = Array.from(new Set([...Object.keys(liveAttributes)]));
|
|
1820
|
+
const cls = class extends Component {
|
|
1821
|
+
static tagName = tagname;
|
|
1822
|
+
static template = template2;
|
|
1823
|
+
static styles = styles;
|
|
1824
|
+
static observedAttributes = [...fullObservedAttributes];
|
|
1825
|
+
static shadowRootMode = shadowRootMode;
|
|
1826
|
+
constructor() {
|
|
1827
|
+
super();
|
|
1828
|
+
this.initialState = typeof initialState === "function" ? initialState.bind(this) : initialState;
|
|
1829
|
+
this.computedProperties = typeof computedProperties === "function" ? computedProperties.bind(this) : computedProperties;
|
|
1830
|
+
this.liveAttributes = liveAttributes;
|
|
1831
|
+
for (const def of Object.values(this.liveAttributes)) {
|
|
1832
|
+
if (def.onChange) {
|
|
1833
|
+
def.onChange = def.onChange.bind(this);
|
|
1834
|
+
}
|
|
1835
|
+
if (def.reflect) {
|
|
1836
|
+
def.reflect = def.reflect.bind(this);
|
|
1837
|
+
}
|
|
1838
|
+
}
|
|
1839
|
+
this.script = script?.bind(this) ?? undefined;
|
|
1840
|
+
this.beforeUpdate = beforeUpdate?.bind(this) ?? undefined;
|
|
1841
|
+
this.afterUpdate = afterUpdate?.bind(this) ?? undefined;
|
|
1842
|
+
this.finalize = finalize?.bind(this) ?? undefined;
|
|
1843
|
+
this.onError = onError?.bind(this) ?? undefined;
|
|
1844
|
+
}
|
|
1845
|
+
};
|
|
1846
|
+
Component.register(cls, tagname);
|
|
1847
|
+
return cls;
|
|
1848
|
+
}
|
|
1849
|
+
export {
|
|
1850
|
+
html,
|
|
1851
|
+
define,
|
|
1852
|
+
css,
|
|
1853
|
+
classify,
|
|
1854
|
+
Template,
|
|
1855
|
+
State,
|
|
1856
|
+
PaletteError,
|
|
1857
|
+
Component
|
|
1858
|
+
};
|
|
1859
|
+
|
|
1860
|
+
//# debugId=331008C45ADD766764756E2164756E21
|
|
1861
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/errors.ts", "src/helper/css.ts", "src/util/attributes.ts", "src/util/change-tag.ts", "src/util/element-match.ts", "src/util/lru-cache.ts", "src/util/fragments.ts", "src/helper/html.ts", "src/state/state.ts", "src/template/syntax.ts", "src/template/build.ts", "src/template/jobs.ts", "src/template/template.ts", "src/component/delegator.ts", "src/component/component.ts", "src/component/factory.ts"],
  "sourcesContent": [
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n/**\n * Possible error types for {@link PaletteError}\n *\n * Codes are categorized into groups by the 100's:\n *\n * - 0 - 99: Invariants\n * - 100 - 199: Custom Elements / Environment Errors\n * - 200 - 299: Templating Errors\n * - 300 - 399: Runtime Errors\n */\nexport const enum ErrorCode {\n  // 0 - 99: Invariants\n  INVARIANT = 0,\n  INVALID_ENV = 1,\n\n  // 100 - 199: Custom Elements Errors\n  DUPE_TAGNAME = 100,\n  INVALID_TAGNAME = 101,\n\n  // 200 - 299: Template Errors\n  TEMPLATE_INVALID = 200,\n  TEMPLATE_INVALID_NOTATION = 201,\n  TEMPLATE_INVALID_COMPONENT = 202,\n  TEMPLATE_INVALID_LIST = 203,\n  TEMPLATE_INVALID_NODE_REF = 204,\n  TEMPLATE_MISSING_LIST_PARENT = 205,\n  TEMPLATE_MISSING_LIST_KEY = 206,\n  TEMPLATE_DUPLICATE_LIST_KEY = 207,\n\n  // 300 - 399: Runtime Errors\n  MISSING_ELEMENT = 300,\n  MISSING_STATE = 301,\n  INVALID_CONTENT = 302,\n  INVALID_STATE_KEY = 303,\n  INVALID_STATE_UPDATE = 304,\n  STATE_LOCKED = 305,\n  MAX_SEQUENTIAL_RENDERS = 306,\n}\n\nfunction getDevError(code: ErrorCode) {\n  const DEV_ERROR_INFO: Record<ErrorCode, string> = {\n    [ErrorCode.INVARIANT]: `\nINVARIANT\nA condition has occurred which should never happen, such as a code path that\nshouldn't be reachable when using correct types, or a disallowed state.\n`,\n\n    [ErrorCode.INVALID_ENV]: `\nINVALID_ENV\nFailed to find a custom elements registry in this environment. Is a window\nglobal available with a customElements property?\n`,\n\n    [ErrorCode.DUPE_TAGNAME]: `\nDUPE_TAGNAME\nFailed to register a custom element with the tagname %s because another\ncustom element is already defined with that name.\n`,\n\n    [ErrorCode.INVALID_TAGNAME]: `\nINVALID_TAGNAME\nFailed to determine a valid HTML Tag to use when registering a component as a\ncustom element. Specify a static readonly tagName property on your component\nclass or provide a tagName argument to Component.register.\n\nAlternatively, you can directly register your element as you would any other\nHTML custom element: customElements.define(\"my-tag\", MyComponentClass);\n`,\n\n    [ErrorCode.MISSING_ELEMENT]: `\nMISSING_ELEMENT\nAn element was expected to exist but was not found when the DOM was queried.\n`,\n\n    [ErrorCode.MISSING_STATE]: `\nMISSING_STATE\nA Component's reactive state was accessed, but no initialState is defined for\nthe Component's class. Initial state must be defined as an object or function.\n`,\n\n    [ErrorCode.INVALID_CONTENT]: `\nINVALID_CONTENT\nTried to set a value as content in a template, but the value was not a supported\ntype. Values must be a string, HTMLElement.\n\nInstead, got %s: %s.\n`,\n\n    [ErrorCode.TEMPLATE_INVALID]: `\nTEMPLATE_INVALID\nThe value %s cannot be interpolated via the Template html string helper.\nValues may only be HTMLTemplateElements (template content is adopted) or\nstrings (shorthand for a ::swap directive on an element)\n`,\n\n    [ErrorCode.TEMPLATE_INVALID_NOTATION]: `\nTEMPLATE_INVALID_NOTATION\nA template value notation failed to parse. Notations must begin with one of\n@, $, *, or #, followed by a dot-separated path of accessor names. Optionally,\nnotations may have modifiers around them, such as the NOT() modifier.\n\nThe notation that failed to parse was \"%s\"\n`,\n    [ErrorCode.TEMPLATE_INVALID_COMPONENT]: `\nTEMPLATE_INVALID_COMPONENT\nFailed to interpolate a Component in an html template string due to the\nComponent having an invalid tagName property.\n\nComponent \"%s\" has an invalid tagName property \"%s\"\n    `,\n\n    [ErrorCode.TEMPLATE_MISSING_LIST_PARENT]: `\nTEMPLATE_MISSING_LIST_PARENT\nA template failed to prepare an ::each directive because no valid parent node\nwas found for the list to render into.\n`,\n\n    [ErrorCode.TEMPLATE_INVALID_LIST]: `\nTEMPLATE_INVALID_LIST\nA template failed to render because an ::each directive fetched a value which\nwas not an array of actionable data.\n`,\n\n    [ErrorCode.TEMPLATE_INVALID_NODE_REF]: `\nTEMPLATE_INVALID_NODE_REF\nA template failed to prepare a node from a compiled source. Typically, this\nmeans something is broken within the compiled template rather than an error with\nthe template content itself.\n`,\n\n    [ErrorCode.TEMPLATE_MISSING_LIST_KEY]: `\nTEMPLATE_MISSING_LIST_KEY\nA template failed to compile because an ::each directive was found on an element\nwithout a corresponding ::key directive. ::each directives must have a ::key\ndirective specified on the same element, whose notation evaluates to a unique\nvalue for each item in the list.\n\nExample: <li ::each=\"$items\" ::key=\"#id\"> ... </li>\n`,\n\n    [ErrorCode.TEMPLATE_DUPLICATE_LIST_KEY]: `\nTEMPLATE_DUPLICATE_LIST_KEY\nA list failed to render because item key %s appeared multiple times when\nresolving list item keys. Each ::key directive must reference a notation whose\nvalue is unique for each list item.\n`,\n\n    [ErrorCode.INVALID_STATE_KEY]: `\nINVALID_STATE_KEY\nTried to access a state property which does not exist\n\nKey path: %s\n`,\n\n    [ErrorCode.INVALID_STATE_UPDATE]: `\nINVALID_STATE_UPDATE\nA Component's setState() was called with a value which could not be patched in\nto the existing component state.\n\nsetState() takes a partial object describing state keys to update.\n\nInstead, received: %s\n`,\n\n    [ErrorCode.STATE_LOCKED]: `\nSTATE_LOCKED\nA State received a request to update or lock while already locked.\nStates may be locked using the .lock() method and unlocked using the .unlock()\nmethod. States lock during async mutations done through the .mutateAsync()\nmethod if \"true\" is passed as the second parameter.\n\nYou can check the lock status of a State instance with .isLocked\n`,\n\n    [ErrorCode.MAX_SEQUENTIAL_RENDERS]: `\nMAX_SEQUENTIAL_RENDERS\nA component has errored after re-rendering more than %s times in the same frame.\n\nThis typically happens because state or live attributes are being modified\nduring a render lifecycle method such as beforeUpdate() or afterUpdate(). If\nstate changes unconditionally during a render, the render will infinitely loop.\nUpdating render source data during a render cycle is supported, but infinite\nupdate loops may occur if unchecked.\n`,\n  };\n\n  return DEV_ERROR_INFO[code];\n}\n\n/**\n * An error originating from within a Palette subsystem.\n *\n * In the `development` environment, the error will surface with extended info\n * about the error and any details which may be relevant to help with debugging.\n *\n * In the `production` environment, a minimal error only surfacing the numeric\n * error code is thrown instead.\n *\n * Codes are categorized into groups by the 100's:\n *\n * - 0 - 99: Invariants\n * - 100 - 199: Custom Elements / Environment Errors\n * - 200 - 299: Templating Errors\n * - 300 - 399: Runtime Errors\n */\nexport class PaletteError extends Error {\n  name = \"PaletteError\";\n  code: number;\n\n  constructor(code: number, ...values: string[]) {\n    let message = `Code: ${code}`;\n    if (__DEV__) {\n      message += getDevError(code);\n\n      for (const val of values) {\n        message = message.replace(`%s`, String(val));\n      }\n    }\n\n    super(message);\n    this.code = code;\n  }\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n/**\n * @returns An array of {@link CSSStyleSheet}s containing the provided CSS\n */\nexport function css(\n  strings: TemplateStringsArray,\n  ...values: unknown[]\n): CSSStyleSheet {\n  let result = strings[0] ?? \"\";\n\n  for (let i = 0; i < values.length; i++) {\n    const nextVal = values[i] ?? \"\";\n    const nextStr = strings[i + 1] ?? \"\";\n    result += `${nextVal}${nextStr}`;\n  }\n\n  const sheet = new CSSStyleSheet();\n  sheet.replaceSync(result);\n  return sheet;\n}\n\ntype ClassValue =\n  | string\n  | number\n  | boolean\n  | undefined\n  | null\n  | ClassArray\n  | ClassObject;\ntype ClassArray = ClassValue[];\ntype ClassObject = Record<string, any>;\n\n/**\n * Given flexible input, generate a string suitable for setting as an html class\n *\n * @example\n *\n * ```typescript\n * classify(\"my-class\", \"another-class\") // \"my-class another-class\"\n * classify([\"my-class\", \"another-class\"]) // \"my-class another-class\"\n * classify({ enabled: true, disabled: false }) // \"enabled\"\n * ```\n */\nexport function classify(...args: ClassValue[]): string {\n  let resultString = \"\";\n\n  for (let index = 0; index < args.length; index++) {\n    const argument = args[index];\n\n    if (!argument) {\n      continue;\n    }\n\n    // Strings/nums -> Direct append\n    if (typeof argument === \"string\" || typeof argument === \"number\") {\n      if (resultString) {\n        resultString += \" \";\n      }\n      resultString += argument;\n      continue;\n    }\n\n    // Handle arrays recursively\n    if (Array.isArray(argument)) {\n      const nestedResult = classify(...argument);\n      if (nestedResult) {\n        if (resultString) {\n          resultString += \" \";\n        }\n        resultString += nestedResult;\n      }\n      continue;\n    }\n\n    // Handle objects with conditional class names\n    if (typeof argument === \"object\") {\n      for (const className in argument) {\n        // Only include class name if its value is truthy\n        if (argument[className]) {\n          if (resultString) {\n            resultString += \" \";\n          }\n          resultString += className;\n        }\n      }\n    }\n  }\n\n  return resultString;\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport { ErrorCode, PaletteError } from \"#error\";\n\n/** An attribute value as read from a DOM element */\ntype AttributeValue = string | null;\n\n/** A Map of string attribute names to {@link AttributeValue} */\ntype AttributeMap = Map<string, AttributeValue>;\n\nexport function serializeAttribute(value: unknown): AttributeValue {\n  if (typeof value === \"string\" || typeof value === \"number\") {\n    return `${value}`;\n  }\n\n  if (value === true) {\n    return \"\";\n  }\n\n  if (value === null || value === false || value === undefined) {\n    return null;\n  }\n\n  return value.toString();\n}\n\nexport function createAttributeMap(\n  source: object | Map<string, unknown> | HTMLElement,\n): AttributeMap {\n  const map = new Map<string, string | null>();\n\n  if (source instanceof HTMLElement) {\n    for (const name of source.getAttributeNames()) {\n      map.set(name, serializeAttribute(source.getAttribute(name)));\n    }\n    return map;\n  }\n\n  if (source instanceof Map) {\n    for (const [key, val] of source) {\n      map.set(key, serializeAttribute(val));\n    }\n    return map;\n  }\n\n  if (typeof source === \"object\" && source !== null) {\n    for (const [key, val] of Object.entries(source)) {\n      map.set(key, serializeAttribute(val));\n    }\n    return map;\n  }\n\n  throw new PaletteError(ErrorCode.INVARIANT);\n}\n\nexport function applyAttributeMap(\n  target: HTMLElement,\n  attrs: AttributeMap,\n): void {\n  const currentNames = new Set(target.getAttributeNames());\n  const incomingNames = new Set(attrs.keys());\n  const attributesToRemove = currentNames.difference(incomingNames);\n\n  for (const attr of attributesToRemove) {\n    target.removeAttribute(attr);\n  }\n\n  for (const [name, val] of attrs) {\n    if (val === null) {\n      target.removeAttribute(name);\n    } else {\n      target.setAttribute(name, val);\n    }\n  }\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n/**\n * Given an element, replace it with a different tag, adopting all attributes\n * and child elements.\n * @param $target The element to swap out\n * @param newTag The new HTML tagName to use\n * @returns The newly swapped-in element\n */\nexport function changeTag($target: HTMLElement, newTag: string): HTMLElement {\n  const $newEl = document.createElement(newTag);\n\n  while ($target.firstChild !== null) {\n    $newEl.appendChild($target.firstChild);\n  }\n\n  for (const attrName of $target.getAttributeNames()) {\n    const val = $target.getAttribute(attrName);\n\n    if (val !== null) {\n      $newEl.setAttribute(attrName, val);\n    }\n  }\n\n  $target.replaceWith($newEl);\n  return $newEl;\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport { CharCodes } from \"./character-codes\";\n\nexport function isTagName(str: string): boolean {\n  for (let i = 0; i < str.length; i++) {\n    // Ensure the character code is a letter or hyphen\n    const char = str.charCodeAt(i);\n\n    if (\n      !(\n        char === CharCodes.Hyphen ||\n        (char >= CharCodes.A && char <= CharCodes.Z) ||\n        (char >= CharCodes.a && char <= CharCodes.z)\n      )\n    ) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\n/**\n * A performance optimized version of `.matches()` which includes faster\n * matching strategies for common queries.\n *\n * @remarks\n * Optimized query types:\n *\n * - Single ID selector `\"#identifier\"`\n * - Single class selector `\".classname\"`\n * - Tagname selector `\"button\"`\n * - Simple attribute selector `\"[attr='value']\"`\n *\n * All other types of queries use the base .matches() for comparison\n */\nexport function elementMatches(element: Element, query: string) {\n  // If the query is a simple ID, check it directly\n  if (\n    query[0] === \"#\" &&\n    query.indexOf(\" \") === -1 &&\n    query.indexOf(\".\") === -1 &&\n    query.indexOf(\"[\") === -1 &&\n    query.indexOf(\":\") === -1 &&\n    query.indexOf(\">\") === -1\n  ) {\n    return element.id === query.slice(1);\n  }\n\n  // If the query is a simple classname, check it directly\n  if (\n    query[0] === \".\" &&\n    query.indexOf(\" \") === -1 &&\n    query.indexOf(\"#\") === -1 &&\n    query.indexOf(\"[\") === -1 &&\n    query.indexOf(\":\") === -1 &&\n    query.indexOf(\">\") === -1\n  ) {\n    return element.classList.contains(query.slice(1));\n  }\n\n  // If the query is for just a tagname, check it directly\n  if (isTagName(query)) {\n    return element.tagName.toLowerCase() === query.toLowerCase();\n  }\n\n  // If the query is for a single selector value, check it directly\n  if (\n    query[0] === \"[\" &&\n    query[query.length - 1] === \"]\" &&\n    query.indexOf(\" \") === -1 &&\n    query.indexOf(\":\") === -1\n  ) {\n    const attr = query.slice(1, -1);\n    const eqIndex = attr.indexOf(\"=\");\n    if (eqIndex === -1) {\n      return element.hasAttribute(attr);\n    }\n    const attrName = attr.slice(0, eqIndex);\n    const attrValue = attr.slice(eqIndex + 1).replace(/^[\"']|[\"']$/g, \"\");\n    return element.getAttribute(attrName) === attrValue;\n  }\n\n  // Otherwise, use native `.matches` for complex stuff\n  return element.matches(query);\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\ntype Metrics = {\n  hits: number;\n  misses: number;\n};\n\nexport type LRUCacheMetrics = {\n  lookups: number;\n  hits: number;\n  misses: number;\n  capacity: number;\n  entries: number;\n  hitRate: number;\n};\n\n/**\n * A simple in-memory LRU Cache using JavaScript maps.\n *\n * @example\n *\n * Cached network request\n *\n * ```typescript\n * // A cache of API URLs to their returned payloads\n * const cache = new LRUCache<string, APIPayload>();\n *\n * async function makeRequest(url: string) {\n *   let cached = cache.get(url);\n *   if (cached) {\n *     return cached;\n *   }\n *\n *   // ... Make the request and cache.set() the response\n * }\n * ```\n */\nexport class LRUCache<K, V> {\n  private _maxSize: number;\n  private _data: Map<K, V>;\n  private _metrics: Metrics = { hits: 0, misses: 0 };\n\n  constructor(maxSize: number = 500) {\n    if (maxSize <= 0) {\n      throw new Error(\"LRU Cache capacity must be >= 1\");\n    }\n\n    if (__DEV__) {\n      this._metrics = { hits: 0, misses: 0 };\n    }\n\n    this._maxSize = maxSize;\n    this._data = new Map<K, V>();\n  }\n\n  private _trim = () => {\n    while (this._data.size > this._maxSize) {\n      const firstKey = this._data.keys().next().value;\n      if (firstKey === undefined) {\n        throw new Error(\"Absurd\");\n      }\n      this._data.delete(firstKey);\n    }\n  };\n\n  /**\n   * Get cache performance metrics from this LRU Cache instance.\n   *\n   * In development mode, this returns insights into hits/misses of the cache,\n   * as well as the current configuration and entry count.\n   *\n   * In production mode, this returns a placeholder object, as metric collection\n   * is not active in production.\n   */\n  get metrics(): LRUCacheMetrics {\n    if (__DEV__) {\n      const { hits = 0, misses = 0 } = this._metrics ?? {};\n      const lookups = hits + misses;\n      const hitRate = lookups === 0 ? 0 : hits / lookups;\n\n      return {\n        lookups,\n        hits: hits,\n        misses: misses,\n        capacity: this._maxSize,\n        entries: this._data.size,\n        hitRate,\n      };\n    }\n\n    return {\n      lookups: 0,\n      hits: 0,\n      misses: 0,\n      capacity: 0,\n      entries: 0,\n      hitRate: 0,\n    };\n  }\n\n  /**\n   * Update the maximum cache size for this instance. Immediately triggers a\n   * purge of cached contents if the new capacity is less than the current\n   * number of cached entries.\n   */\n  setCapacity = (maxSize: number): void => {\n    if (maxSize <= 0 && __DEV__) {\n      console.warn(\n        \"[Palette LRU Cache] Cache size is <= 0. Cache is disabled.\",\n      );\n    }\n\n    this._maxSize = maxSize;\n    this._trim();\n  };\n\n  /**\n   * Fetch a value from the cache, returning `undefined` if no value was found\n   */\n  get(key: K): V | undefined {\n    const value = this._data.get(key);\n\n    if (value === undefined) {\n      if (__DEV__) {\n        this._metrics.misses += 1;\n      }\n      return undefined;\n    }\n\n    // Delete and re-set the value to move the key to the end of the key list\n    this._data.delete(key);\n    this._data.set(key, value);\n\n    if (__DEV__) {\n      this._metrics.hits += 1;\n    }\n\n    return value;\n  }\n\n  /**\n   * Set a value in the cache. This also moves the value to the back of the list\n   * of values to consider evicting when the cache is full.\n   */\n  set(key: K, value: V): void {\n    // Remove if exists to update position\n    if (this._data.has(key)) {\n      this._data.delete(key);\n    }\n\n    this._data.set(key, value);\n\n    if (this._data.size > this._maxSize) {\n      this._trim();\n    }\n  }\n\n  /**\n   * Clear all contents of this cache instance\n   */\n  clear(): void {\n    this._data.clear();\n  }\n\n  /**\n   * Get the number of entries in this cache instance\n   */\n  get size(): number {\n    return this._data.size;\n  }\n}\n\nexport default LRUCache;\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport LRUCache from \"./lru-cache\";\n\nexport const fragmentCache = new LRUCache<string, DocumentFragment>();\n\nexport function htmlToFragment(html: string): DocumentFragment {\n  const lookup = fragmentCache.get(html);\n\n  if (lookup !== undefined) {\n    return lookup.cloneNode(true) as DocumentFragment;\n  }\n\n  const temp = document.createElement(\"template\");\n  temp.innerHTML = html;\n  fragmentCache.set(html, temp.content);\n  return temp.content.cloneNode(true) as DocumentFragment;\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport { Component } from \"#component\";\nimport { ErrorCode, PaletteError } from \"#error\";\nimport { isTagName } from \"#util\";\n\nexport function parseTemplateStringValue(\n  val: string | typeof Component | HTMLTemplateElement,\n): string {\n  if (val instanceof HTMLTemplateElement) {\n    return val.innerHTML;\n  }\n\n  if (\n    typeof val === \"function\" &&\n    (val.prototype instanceof Component || val === Component)\n  ) {\n    const tagName = val.tagName;\n    if (!isTagName(tagName)) {\n      throw new PaletteError(\n        ErrorCode.TEMPLATE_INVALID_COMPONENT,\n        val.name,\n        val.tagName,\n      );\n    }\n    return tagName;\n  }\n\n  if (typeof val === \"string\") {\n    return `<span ::swap=\"${val}\"></span>`;\n  }\n\n  throw new PaletteError(ErrorCode.TEMPLATE_INVALID, String(val));\n}\n\n/**\n * Create an {@link HTMLTemplateElement} from the provided HTML string.\n *\n * These HTML Templates can be used as standard HTML Templates, or as the\n * foundation for a {@link Component} or standalone {@link Template}\n *\n * Interpolates values in the template string to provide helpers for composing\n * templates:\n *\n * - HTMLTemplateElement references are serialized to HTML and inlined\n * - Strings are modified to become <span> tags with the ::swap directive set\n *   to the string value. This is helpful shorthand for content placeholders.\n *\n * @example\n *\n * This template string\n *\n * ```typescript\n * const $template = html`\n * <div :class=\"*classnames\">\n *   ${\"@content\"}\n * </div>\n * `\n * ```\n *\n * Produces this template element (portrayed as HTML)\n *\n * ```html\n * <template>\n *   <div :class=\"*classnames\">\n *     <span ::swap=\"@content\"></span>\n *   </div>\n * </template>\n * ```\n */\nexport function html(\n  strings: TemplateStringsArray,\n  ...values: (\n    | `@${string}`\n    | `$${string}`\n    | `#${string}`\n    | `*${string}`\n    | HTMLTemplateElement\n    | typeof Component<any>\n  )[]\n): HTMLTemplateElement {\n  const fullString = strings.reduce((acc, str, i) => {\n    const parsedValue = values[i]\n      ? `${parseTemplateStringValue(values[i])}`\n      : \"\";\n    return acc + str + parsedValue;\n  }, \"\");\n\n  const templateElement = document.createElement(\"template\");\n  templateElement.innerHTML = fullString;\n  return templateElement;\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n\nimport { ErrorCode, PaletteError } from \"#error\";\n\nexport type StateListener<Shape> = (state: Shape) => void;\n\nconst PROXY_DEPTH_WARNING_LIMIT = 4;\n\nconst ArrayMutatorFunctions = new Set([\n  \"push\",\n  \"pop\",\n  \"shift\",\n  \"unshift\",\n  \"splice\",\n  \"sort\",\n  \"reverse\",\n  \"fill\",\n  \"copyWithin\",\n]);\n\n/**\n * Powerful and flexible state management with deep reactivity, transaction, and\n * locking support.\n *\n * The State class wraps stateful data and provides an API for mutating the data\n * while listeners subscribe to meaningful updates.\n *\n * Modify state with setters, mutators, transactions, or proxied updates with\n * deep reactivity and array mutation support built in.\n *\n * @example\n *\n * ```typescript\n * const state = new State({count: 1});\n * state.addListener(data => console.log(data))\n *\n * state.set(\"count\", 2); // Prints \"2\"\n * state.set(\"count\", 2); // Nothing happens, no change\n * state.live.count = 3; // Prints \"3\"\n * ```\n */\nexport class State<Shape extends object> {\n  private _data: Shape;\n  private _listeners: Set<StateListener<Shape>> = new Set();\n  private _proxy?: Shape;\n  private _proxyCache = new WeakMap<object, any>();\n  private _reverseProxyCache = new WeakMap<object, object>();\n  private _isLocked = false;\n\n  // Only allow warning in dev mode, and only print the warning once per inst\n  private _hasWarned = !__DEV__;\n\n  constructor(initialData: Shape, onChange?: StateListener<Shape>) {\n    this._data = initialData;\n    this.get = this.get.bind(this);\n\n    if (onChange) {\n      this._listeners.add(onChange);\n    }\n  }\n\n  /**\n   * Lazily create and cache proxies into the state for reactive updates on\n   * nested values.\n   */\n  private _createProxy<T>(target: T, path: string[] = []): T {\n    // Return primitives immediately\n    if (typeof target !== \"object\" || target === null) {\n      return target;\n    }\n\n    // Check for a cached proxy for this object\n    const cached = this._proxyCache.get(target as object);\n    if (cached) {\n      return cached;\n    }\n\n    if (__DEV__) {\n      if (!this._hasWarned && path.length > PROXY_DEPTH_WARNING_LIMIT) {\n        this._hasWarned = true;\n        console.warn(`\n[State] Deeply nested live state access may have performance impacts. For deeply\nnested or complex state updates, use .mutate(), .patch(), or .transaction() for\noptimal performance (Warning while evaluating live access to ${path.join(\".\")})\n`);\n      }\n    }\n\n    // Otherwise create a new proxy and cache it\n    const proxy = new Proxy(target as object, {\n      get: (obj, key: string | symbol) => {\n        // Symbols -> Direct return\n        if (typeof key === \"symbol\") {\n          return (obj as any)[key];\n        }\n\n        const value = (obj as any)[key];\n\n        // Array method invocations -> Run on a clone and special case mutations\n        if (Array.isArray(obj) && typeof value === \"function\") {\n          const isMutation = ArrayMutatorFunctions.has(key);\n\n          return (...args: any[]) => {\n            const snapshot = this.snapshot();\n\n            let current: any = snapshot;\n\n            for (const p of path) {\n              current = current[p];\n            }\n\n            const res = (current as any)[key](...args);\n\n            if (isMutation) {\n              if (this._isLocked) {\n                throw new PaletteError(ErrorCode.STATE_LOCKED);\n              }\n              this._processIncomingState(snapshot);\n            }\n\n            return res;\n          };\n        }\n\n        // Objects -> recursively create a proxy\n        if (typeof value === \"object\" && value !== null) {\n          return this._createProxy(value, [...path, key]);\n        }\n\n        // Primitives -> Return value\n        return value;\n      },\n\n      set: (_obj, key: string | symbol, value: any) => {\n        // Reject if currently locked\n        if (this._isLocked) {\n          throw new PaletteError(ErrorCode.STATE_LOCKED);\n        }\n\n        // Disallow setting symbol properties on target data values\n        if (typeof key === \"symbol\") {\n          return false;\n        }\n\n        // Compute and update the path with the value\n        const newPath = [...path, key];\n        this._setNested(newPath, this._unwrapProxy(value));\n\n        return true;\n      },\n\n      has: (obj, key: string | symbol) => {\n        return key in obj;\n      },\n\n      ownKeys: (obj) => {\n        return Reflect.ownKeys(obj);\n      },\n\n      getOwnPropertyDescriptor: (obj, key: string | symbol) => {\n        const desc = Object.getOwnPropertyDescriptor(obj, key);\n\n        if (desc) {\n          desc.configurable = true;\n        }\n\n        return desc;\n      },\n\n      deleteProperty: (obj, key) => {\n        if (typeof key === \"symbol\" || !(key in obj)) {\n          return false;\n        }\n\n        // Compute and update the path with the value\n        const newPath = [...path, key];\n\n        this._setNested(newPath, undefined);\n\n        return true;\n      },\n    }) as T;\n\n    this._proxyCache.set(target as object, proxy);\n    this._reverseProxyCache.set(proxy as object, target as object);\n    return proxy;\n  }\n\n  private _unwrapProxy<T>(value: T): T {\n    // Don't do anything for primitives\n    if (typeof value !== \"object\" || value === null) {\n      return value;\n    }\n\n    // Check if this is one of our proxies\n    const rawTarget = this._reverseProxyCache.get(value as object);\n    if (rawTarget) {\n      return rawTarget as T;\n    }\n\n    // Handle arrays\n    if (Array.isArray(value)) {\n      return value.map((item) => this._unwrapProxy(item)) as T;\n    }\n\n    // Handle plain objects\n    if (Object.getPrototypeOf(value) === Object.prototype) {\n      const unwrapped: any = {};\n      for (const key in value) {\n        if (Object.hasOwn(value, key)) {\n          unwrapped[key] = this._unwrapProxy((value as any)[key]);\n        }\n      }\n      return unwrapped;\n    }\n\n    // Other objects (Date, Map, Set, etc.) pass through\n    return value;\n  }\n\n  /**\n   * Handle setting nested values and emitting updated data if the value changed\n   */\n  private _setNested(path: string[], value: any): void {\n    // Nothing to do with empty paths\n    if (!path.length) {\n      return;\n    }\n\n    // Start at the root and seek to the value being set\n    let current: any = this._data;\n\n    for (let i = 0; i < path.length - 1; i++) {\n      const key = path[i] ?? \"\";\n      if (!(key in current)) {\n        throw new PaletteError(ErrorCode.INVALID_STATE_KEY, path.join(\".\"));\n      }\n      current = current[key];\n    }\n\n    const finalKey = path[path.length - 1] ?? \"\";\n    if (!(finalKey in current)) {\n      throw new PaletteError(ErrorCode.INVALID_STATE_KEY, path.join(\".\"));\n    }\n\n    // Set and emit only if the value is different. Otherwise no-op.\n    if (current[finalKey] !== value && !Object.is(current[finalKey], value)) {\n      current[finalKey] = value;\n      this._emit(this.current);\n    }\n  }\n\n  /**\n   * Process an incoming partial state and emit updates to listeners if there\n   * are any changes detected.\n   *\n   * @remarks\n   * Object.is() to check for differences. Does not attempt deep diffing.\n   */\n  private _processIncomingState = (incomingState: Partial<Shape>): void => {\n    if (\n      typeof incomingState !== \"object\" ||\n      incomingState === null ||\n      Array.isArray(incomingState)\n    ) {\n      throw new PaletteError(\n        ErrorCode.INVALID_STATE_UPDATE,\n        JSON.stringify(incomingState),\n      );\n    }\n    let hasChanges = false;\n\n    // Check if any key/value pair on the incoming state differs from the\n    // existing data\n    for (const [key, newValue] of Object.entries(incomingState)) {\n      const currentValue = this._data[key as keyof Shape];\n      if (!Object.is(currentValue, newValue)) {\n        hasChanges = true;\n        (this._data as any)[key] = newValue;\n      }\n    }\n\n    if (hasChanges) {\n      this._emit(this._data);\n    }\n  };\n\n  /**\n   * Emit the current state to all listeners\n   */\n  private _emit = (data: Shape): void => {\n    for (const listener of this._listeners) {\n      listener(data);\n    }\n  };\n\n  /**\n   * The current state\n   *\n   * @remarks\n   * This is a direct reference to the internal state. It is marked as Readonly\n   * to help prevent accidental changes. Never edit this value directly.\n   */\n  get current(): Readonly<Shape> {\n    return this._data;\n  }\n\n  /**\n   * Proxied access to state properties with automatic reconciliation.\n   *\n   * Allows direct property access for both reading and writing state values.\n   *\n   * @example\n   *\n   * Deep reactivity with the `.live` accessor\n   *\n   * ```typescript\n   * const state = new State({\n   *   nested: {\n   *     values: [1, 2, 3],\n   *   }\n   * });\n   *\n   * // Deep value assignment triggers state updates\n   * state.live.nested.values[1] = 5;\n   *\n   * // Array mutators are automatically reactive as well\n   * state.live.nested.values.push(4);\n   * state.live.nested.values.reverse();\n   * ```\n   */\n  get live(): Shape {\n    if (!this._proxy) {\n      this._proxy = this._createProxy(this._data);\n    }\n    return this._proxy;\n  }\n\n  /**\n   * Indicates if this state is currently locked (true) or not (false)\n   */\n  get isLocked(): boolean {\n    return this._isLocked;\n  }\n\n  /**\n   * Add a handler function for when this state changes.\n   *\n   * Listeners are invoked in the order they are registered and are passed a\n   * reference to the internally managed state object as a readonly object.\n   *\n   * **Do not modify state values directly**. Instead, use {@link set} or\n   * {@link patch} to make immutable updates.\n   */\n  addListener = (listener: StateListener<Shape>): void => {\n    this._listeners.add(listener);\n  };\n\n  /**\n   * Remove a previously registered state handler\n   */\n  removeListener = (listener: StateListener<Shape>): void => {\n    this._listeners.delete(listener);\n  };\n\n  /**\n   * Get a single value from the state object\n   *\n   * The value is returned as Readonly to prevent accidental state mutations.\n   * To mutate stateful properties, use {@link set} or {@link patch} instead.\n   *\n   * @example\n   *\n   * ```typescript\n   * const state = new State({ count: 1 });\n   * console.log(state.get(\"count\")); // 1\n   * ```\n   *\n   */\n  get = <K extends keyof Shape>(key: K): Readonly<Shape[K]> => {\n    return this._data[key] as Readonly<Shape[K]>;\n  };\n\n  /**\n   * Returns a deep clone of the current state data using `structuredClone()` to\n   * make the copy.\n   *\n   * The object returned from this function can be edited without modifying the\n   * actual internal state.\n   *\n   * @example\n   *\n   * ```typescript\n   * const snap = state.snapshot();\n   * snap.count += 1;\n   * state.patch(snap);\n   * ```\n   */\n  snapshot = (): Shape => {\n    return structuredClone(this._data);\n  };\n\n  /**\n   * Set a single stateful property while leaving other properties unchanged.\n   *\n   * Triggers an update to all listeners if the value is different from before\n   * when compared using shallow equality.\n   *\n   * @example\n   *\n   * ```typescript\n   * const state = new State({ count: 1, color: \"red\" });\n   * state.set(\"count\", 2); // State is now { count: 2, color: \"red\" }\n   * ```\n   */\n  set = <K extends keyof Shape>(key: K, value: Shape[K]): void => {\n    // Reject if currently locked\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    const partial: Partial<Shape> = {};\n    partial[key] = value;\n\n    this._processIncomingState(partial);\n  };\n\n  /**\n   * Set multiple stateful properties at once, leaving omitted properties\n   * unchanged.\n   *\n   * @example\n   *\n   * Patch a partial state, updating all listed properties at once\n   *\n   * ```typescript\n   * const state = new State({\n   *   weather: \"sunny\",\n   *   temperature: 30,\n   *   humidity: 70,\n   * });\n   *\n   * // Leaves `temperature` unchanged\n   * state.patch({\n   *   weather: \"cloudy\",\n   *   humidity: 50,\n   * });\n   * ```\n   */\n  patch = (patch: Partial<Shape>): State<Shape> => {\n    // Reject if currently locked\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    this._processIncomingState(patch);\n    return this;\n  };\n\n  /**\n   * Fully replace the current state data and force listeners to receive the\n   * updated data immediately.\n   */\n  replace = (state: Shape): State<Shape> => {\n    // Reject if currently locked\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    this._data = state;\n    this._emit(this._data);\n    return this;\n  };\n\n  /**\n   * Lock this State instance, preventing further external changes\n   */\n  lock = (): State<Shape> => {\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    this._isLocked = true;\n    return this;\n  };\n\n  /**\n   * Unlock this State instance, allowing further external changes\n   */\n  unlock = (): State<Shape> => {\n    this._isLocked = false;\n    return this;\n  };\n\n  /**\n   * Apply complex updates to the state using a mutator function.\n   *\n   * The mutator function takes one parameter which is a structuredClone copy of\n   * the current state object. Whatever is returned by the mutator is then\n   * patched in to the state.\n   */\n  mutate = (mutator: (current: Shape) => Shape): State<Shape> => {\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    this._processIncomingState(mutator(this.snapshot()));\n    return this;\n  };\n\n  /**\n   * Perform an async mutation of the state, optionally locking the state during\n   * the process.\n   * @param mutator A function to mutate and return the new state\n   * @param lock If `true`, lock the state until the mutator completes\n   */\n  mutateAsync = async (\n    mutator: (current: Shape) => Promise<Shape>,\n    lock = false,\n  ): Promise<State<Shape>> => {\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    if (lock) {\n      this._isLocked = true;\n    }\n\n    const mutationResult = await mutator(this.snapshot());\n    this._processIncomingState(mutationResult);\n\n    if (lock) {\n      this._isLocked = false;\n    }\n\n    return this;\n  };\n\n  /**\n   * Perform a transaction-style set of actions defined within a function.\n   *\n   * The provided function can do anything, beyond just setting state. Any\n   * uncaught errors thrown from within the function will cause the transaction\n   * to fail and the state to automatically roll back to the last valid state.\n   *\n   * During a transaction, this state instance will be locked, preventing other\n   * changes.\n   *\n   * Transactions will always result in a state update when successful.\n   *\n   * @example\n   *\n   * Transactions with locking and rollback support\n   *\n   * ```typescript\n   * const state = new State({count: 1});\n   *\n   * state.transaction(async (s) => {\n   *   // `s` is a full State object you can safely manipulate\n   *   s.set(\"count\", 10);\n   * });\n   * state.get(\"count\"); // => 10;\n   *\n   * // Errors inside the transaction roll back the state\n   * state.transaction(async (s) => {\n   *   s.set(\"count\", 100);\n   *   throw new Error();\n   * });\n   * state.get(\"count\"); // => 10;\n   */\n  transaction = (fn: (state: State<Shape>) => void): boolean => {\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    // Lock outside of the try/catch to throw immediately, separating the\n    // transaction function error from the actual call site errors\n    this._isLocked = true;\n\n    let success = true;\n\n    try {\n      // Create a temporary state object to use within the transaction\n      const transactionState = new State<Shape>(this.snapshot());\n\n      // Run the transaction itself\n      fn(transactionState);\n\n      // Adopt the resulting state of the temporary object\n      this._processIncomingState(transactionState.current);\n    } catch (error) {\n      if (__DEV__) {\n        console.warn(\"[State] Transaction failed due to caught error\", error);\n      }\n      success = false;\n    } finally {\n      this._isLocked = false;\n    }\n\n    return success;\n  };\n\n  /**\n   * Perform a transaction-style set of actions defined within an async function\n   *\n   * The provided function can do anything, beyond just setting state. Any\n   * uncaught errors thrown from within the function will cause the transaction\n   * to fail and the state to automatically roll back to the last valid state.\n   *\n   * During a transaction, this state instance will be locked, preventing other\n   * changes.\n   *\n   * Transactions will always result in a state update when successful.\n   *\n   * @example\n   *\n   * Transactions with locking and rollback support\n   *\n   * ```typescript\n   * const state = new State({count: 1});\n   *\n   * // Awaiting the result of a transaction\n   * await state.transactionAsync(async (s) => {\n   *   // `s` is a full State object you can safely manipulate\n   *   s.set(\"count\", 10);\n   * });\n   * state.get(\"count\"); // => 10;\n   *\n   * // Errors inside the transaction roll back the state\n   * await state.transactionAsync(async (s) => {\n   *   s.set(\"count\", 100);\n   *   throw new Error();\n   * });\n   * state.get(\"count\"); // => 10;\n   *\n   * // If you forget to await the transaction, its still locked\n   * state.transactionAsync(async (s) => {\n   *   await waitSeconds(1);\n   * });\n   * state.set(\"count\", 1); // Error: State is locked!\n   * ```\n   */\n  transactionAsync = async (\n    fn: (state: State<Shape>) => Promise<void>,\n  ): Promise<boolean> => {\n    if (this._isLocked) {\n      throw new PaletteError(ErrorCode.STATE_LOCKED);\n    }\n\n    // Lock outside of the try/catch to throw immediately, separating the\n    // transaction function error from the actual call site errors\n    this._isLocked = true;\n\n    let success = true;\n\n    try {\n      // Create a temporary state object to use within the transaction\n      const transactionState = new State<Shape>(this.snapshot());\n\n      // Run the transaction itself\n      await fn(transactionState);\n\n      // Adopt the resulting state of the temporary object\n      this._processIncomingState(transactionState.current);\n    } catch (error) {\n      if (__DEV__) {\n        console.warn(\"[State] Transaction failed due to caught error\", error);\n      }\n      success = false;\n    } finally {\n      this._isLocked = false;\n    }\n\n    return success;\n  };\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n\nimport { ErrorCode, PaletteError } from \"#error\";\nimport { CharCodes } from \"#util\";\nimport type { ParsedNotation } from \"./types\";\n\n/**\n * Check if an attribute name matches the \"bound attribute\" syntax\n *\n * @example\n * ```typescript\n * isBoundAttribute(\":class\") // true\n * isBoundAttribute(\"::class\") // false\n * isBoundAttribute(\"class\") // false\n * ```\n *\n * @remarks\n * This function uses loops and character code checks instead of a regex to\n * optimize for hot paths during initial parsing and rendering.\n */\nexport function isBoundAttribute(attrName: string): boolean {\n  if (attrName[0] !== \":\" || attrName.length < 2) {\n    return false;\n  }\n\n  for (let i = 1; i < attrName.length; i++) {\n    // Ensure the character code is a letter or hyphen\n    const char = attrName.charCodeAt(i);\n\n    if (\n      !(\n        char === CharCodes.Hyphen ||\n        (char >= CharCodes.A && char <= CharCodes.Z) ||\n        (char >= CharCodes.a && char <= CharCodes.z)\n      )\n    ) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\nenum NotationPrefix {\n  Attributes = \"@\",\n  State = \"$\",\n  Data = \"*\",\n  Item = \"#\",\n}\n\n/**\n * Parse a notation into an object with runtime information\n *\n * @remarks\n * Four notations are supported:\n *\n * - `@attrname` -> Root element attributes\n * - `$statevalue` -> Stateful content\n * - `*property.or.whatever` -> Computed data properties\n * - `#color` -> List item properties\n */\nexport function parseNotation(notation: string): ParsedNotation {\n  // Check for prefix modifiers\n  let workingNotation = notation;\n  let notModifier = false;\n\n  if (\n    notation[0] === \"N\" &&\n    notation[1] === \"O\" &&\n    notation[2] === \"T\" &&\n    notation[3] === \":\"\n  ) {\n    workingNotation = notation.slice(4);\n    notModifier = true;\n  }\n\n  // Base notation must begin with a supported identifier\n  const firstChar = workingNotation[0];\n  if (firstChar === NotationPrefix.Attributes) {\n    workingNotation = `attr.${workingNotation.slice(1)}`;\n  } else if (firstChar === NotationPrefix.State) {\n    workingNotation = `state.${workingNotation.slice(1)}`;\n  } else if (firstChar === NotationPrefix.Data) {\n    workingNotation = `data.${workingNotation.slice(1)}`;\n  } else if (firstChar === NotationPrefix.Item) {\n    workingNotation = `item.${workingNotation.slice(1)}`;\n  } else {\n    throw new PaletteError(\n      ErrorCode.TEMPLATE_INVALID_NOTATION,\n      workingNotation,\n    );\n  }\n\n  const path = workingNotation.split(\".\");\n\n  if (path.length < 2) {\n    throw new PaletteError(\n      ErrorCode.TEMPLATE_INVALID_NOTATION,\n      workingNotation,\n    );\n  }\n\n  return {\n    base: notation,\n    path,\n    modifiers: notModifier ? { not: true } : undefined,\n  };\n}\n\nexport type NotationAccessor = (source: unknown) => unknown;\n\nexport function buildValueAccessor(notation: ParsedNotation): NotationAccessor {\n  const { path, modifiers } = notation;\n\n  if (path.length < 2) {\n    // Notations must always have at least 2 parts: the namespace, and at least\n    // one property name on the namespace (example: [\"state\", \"visible\"])\n    throw new PaletteError(ErrorCode.INVARIANT);\n  }\n\n  // For 2 and 3 length paths, pre-bake accessors to optimize for common cases\n  if (path.length === 2) {\n    const [firstKey, secondKey] = path;\n\n    if (firstKey === undefined || secondKey === undefined) {\n      throw new PaletteError(ErrorCode.INVARIANT);\n    }\n\n    return (source: unknown) => {\n      if (modifiers?.not) {\n        const value = (source as any)?.[firstKey]?.[secondKey] ?? null;\n\n        // Attribute values must treat the empty string as truthy per DOM rules\n        if (firstKey === \"attr\") {\n          return value !== \"\" && !value;\n        }\n\n        return !value;\n      }\n      return (source as any)?.[firstKey]?.[secondKey] ?? null;\n    };\n  }\n\n  if (path.length === 3) {\n    const [firstKey, secondKey, thirdKey] = path;\n\n    if (\n      firstKey === undefined ||\n      secondKey === undefined ||\n      thirdKey === undefined\n    ) {\n      throw new PaletteError(ErrorCode.INVARIANT);\n    }\n\n    return (source: unknown) => {\n      if (modifiers?.not) {\n        const value =\n          (source as any)?.[firstKey]?.[secondKey]?.[thirdKey] ?? null;\n\n        if (firstKey === \"attr\") {\n          return value !== \"\" && !value;\n        }\n\n        return !value;\n      }\n      return (source as any)?.[firstKey]?.[secondKey]?.[thirdKey] ?? null;\n    };\n  }\n\n  // For longer paths, opt for a pre-built loop\n  return (source: unknown) => {\n    let current = source;\n\n    for (const part of path) {\n      if (!current || typeof current !== \"object\") {\n        return null;\n      }\n\n      current = current[part as keyof typeof current];\n    }\n\n    if (modifiers?.not) {\n      if (path[0] === \"attr\") {\n        return current !== \"\" && !current;\n      }\n      return !current;\n    }\n\n    return current;\n  };\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport { ErrorCode, PaletteError } from \"#error\";\nimport { htmlToFragment } from \"#util\";\nimport {\n  buildValueAccessor,\n  isBoundAttribute,\n  type NotationAccessor,\n  parseNotation,\n} from \"./syntax\";\nimport type {\n  AnyRenderPlan,\n  AnyScheme,\n  AttributeUpdatePlan,\n  AttributeUpdateScheme,\n  CompiledTemplate,\n  ContentSwapPlan,\n  ContentSwapScheme,\n  ListRenderPlan,\n  ListRenderScheme,\n  ParsedNotation,\n  PreparedTemplate,\n  TagChangePlan,\n} from \"./types\";\n\nconst REF_ATTR = \"data-node-ref\";\n\nfunction collectOperableNodes(fragment: DocumentFragment): HTMLElement[] {\n  const walker = document.createTreeWalker(fragment, NodeFilter.SHOW_ELEMENT);\n  const knownIgnores = new Set<Node>();\n  const nodes: HTMLElement[] = [];\n\n  let currentNode: Node | null = walker.nextNode();\n\n  while (currentNode) {\n    if (!(currentNode instanceof HTMLElement)) {\n      currentNode = walker.nextNode();\n      continue;\n    }\n\n    let parent = currentNode.parentNode;\n    let isInList = false;\n\n    while (parent && !isInList) {\n      if (\n        knownIgnores.has(parent) ||\n        (parent instanceof HTMLElement && parent.hasAttribute(\"::each\"))\n      ) {\n        isInList = true;\n        knownIgnores.add(currentNode);\n        break;\n      }\n\n      parent = parent.parentNode;\n    }\n\n    if (!isInList) {\n      nodes.push(currentNode);\n    }\n\n    currentNode = walker.nextNode();\n  }\n\n  return nodes;\n}\n\n/**\n * Compile an {@link HTMLTemplateElement} into a {@link CompiledTemplate} for\n * caching, cloning, or storage.\n *\n * Compiled template objects contain a JSON-serializable data representation of\n * a template and can be prepared using {@link prepareCompiledTemplate} to\n * create new Template instances.\n *\n * @example\n *\n * ```typescript\n * const compiled = compileTemplate($templateElement);\n * const stringified = JSON.stringify(compiled); // a JSON template bundle\n * ```\n */\nexport function compileTemplate(source: HTMLTemplateElement): CompiledTemplate {\n  const parsedNotations = new Map<string, ParsedNotation>();\n  const schemes: AnyScheme[] = [];\n  let elementId = 0;\n\n  const ensureNodeRef = (el: HTMLElement): string => {\n    let ref = el.getAttribute(REF_ATTR);\n    if (!ref) {\n      ref = `el-${elementId++}`;\n      el.setAttribute(REF_ATTR, ref);\n    }\n    return ref;\n  };\n\n  const clonedTemplateElement = source.content.cloneNode(\n    true,\n  ) as DocumentFragment;\n\n  const nodes: HTMLElement[] = collectOperableNodes(clonedTemplateElement);\n\n  for (const node of nodes) {\n    // Collect all attributes on this node\n    const attributes = new Map();\n    for (const name of node.getAttributeNames()) {\n      attributes.set(name, node.getAttribute(name));\n    }\n\n    // Check for directive attributes\n    for (const [attributeName, attributeValue] of attributes) {\n      // Bail quickly if this isn't a directive\n      if (attributeValue === null || attributeName[0] !== \":\") {\n        continue;\n      }\n\n      // If the notation has been parsed before, reuse that\n      let notation = parsedNotations.get(attributeValue);\n\n      if (notation === undefined) {\n        notation = parseNotation(attributeValue);\n        parsedNotations.set(attributeValue, notation);\n      }\n\n      node.removeAttribute(attributeName);\n\n      // ::key is handled in the ::each route\n      if (attributeName === \"::key\") {\n        continue;\n      }\n\n      // TAG CHANGE SCHEMES\n      if (attributeName === \"::tag\") {\n        schemes.push({\n          type: \"tag\",\n          notation,\n          nodeRef: ensureNodeRef(node),\n        });\n      }\n\n      // LIST RENDER SCHEMES\n      else if (attributeName === \"::each\") {\n        const nodeRef = ensureNodeRef(node);\n\n        // A ::key directive is required with any ::each\n        const keyNotation = node.getAttribute(\"::key\");\n        if (keyNotation === null) {\n          throw new PaletteError(ErrorCode.TEMPLATE_MISSING_LIST_KEY);\n        }\n\n        node.removeAttribute(\"::key\");\n\n        const listContentHtml = node.innerHTML;\n\n        while (node.firstChild) {\n          node.removeChild(node.firstChild);\n        }\n\n        const scheme: ListRenderScheme = {\n          type: \"each\",\n          notation,\n          keyNotation: parseNotation(keyNotation),\n          nodeRef,\n          listContentHtml,\n        };\n\n        schemes.push(scheme);\n      }\n\n      // CONTENT SWAP SCHEMES\n      else if (attributeName === \"::swap\") {\n        const scheme: ContentSwapScheme = {\n          type: \"swap\",\n          notation,\n          nodeRef: ensureNodeRef(node),\n        };\n\n        schemes.push(scheme);\n      }\n\n      // ATTRIBUTE UPDATE SCHEMES\n      else if (isBoundAttribute(attributeName)) {\n        const bindName = attributeName.slice(1);\n\n        const scheme: AttributeUpdateScheme = {\n          type: \"attr\",\n          notation,\n          attribute: bindName,\n          nodeRef: ensureNodeRef(node),\n        };\n\n        schemes.push(scheme);\n      } else {\n        if (__DEV__) {\n          console.warn(`Template: ${attributeName} is not a valid directive`);\n        }\n      }\n    }\n  }\n\n  const notations: Record<string, ParsedNotation> = {};\n\n  for (const [key, val] of parsedNotations) {\n    notations[key] = val;\n  }\n\n  const temp = document.createElement(\"div\");\n  temp.appendChild(clonedTemplateElement.cloneNode(true));\n  const html = temp.innerHTML;\n\n  const compiled: CompiledTemplate = {\n    html,\n    schemes,\n    notations,\n  };\n\n  return compiled;\n}\n\n/**\n * Given a {@link CompiledTemplate} structure, return an object containing the\n * data needed to create new Template instances.\n *\n * @example\n *\n * ```typescript\n * const compiledTemplate = getCompiledTemplateJSON();\n * const parsed = JSON.parse(compiledTemplate);\n * const prepared = prepareCompiledTemplate(parsed);\n *\n * // Elsewhere...\n *\n * const template = new Template(prepared);\n * ```\n */\nexport function prepareCompiledTemplate(\n  compiled: CompiledTemplate,\n  prototypeFragment?: DocumentFragment,\n): PreparedTemplate {\n  const plansByNotation: Map<string, Set<AnyRenderPlan>> = new Map();\n  const notationAccessors: Map<string, NotationAccessor> = new Map();\n  const nodeRefs: Map<string, HTMLElement> = new Map();\n\n  for (const [notation, parsed] of Object.entries(compiled.notations)) {\n    notationAccessors.set(notation, buildValueAccessor(parsed));\n  }\n\n  const fragment = prototypeFragment ?? htmlToFragment(compiled.html);\n\n  const walker = document.createTreeWalker(fragment, NodeFilter.SHOW_ELEMENT, {\n    acceptNode(node: HTMLElement) {\n      return node.hasAttribute(REF_ATTR)\n        ? NodeFilter.FILTER_ACCEPT\n        : NodeFilter.FILTER_SKIP;\n    },\n  });\n\n  let markNode: Node | null = walker.nextNode();\n\n  while (markNode) {\n    if (!(markNode instanceof HTMLElement)) {\n      continue;\n    }\n\n    const ref = markNode.getAttribute(REF_ATTR);\n\n    if (ref === null) {\n      // Should never happen in practice, as the tree walker only surfaces\n      // elements with an extant node ref attribute\n      throw new PaletteError(ErrorCode.INVARIANT);\n    }\n\n    nodeRefs.set(ref, markNode);\n    markNode.removeAttribute(REF_ATTR);\n    markNode = walker.nextNode();\n  }\n\n  const associatePlan = (plan: AnyRenderPlan, notation: string) => {\n    let notationPlans = plansByNotation.get(notation);\n    if (!notationPlans) {\n      notationPlans = new Set();\n      plansByNotation.set(notation, notationPlans);\n    }\n    notationPlans.add(plan);\n  };\n\n  const getByNodeRef = (nodeRef: string): HTMLElement => {\n    const cached = nodeRefs.get(nodeRef);\n\n    if (cached) {\n      return cached;\n    }\n\n    throw new PaletteError(ErrorCode.TEMPLATE_INVALID_NODE_REF);\n  };\n\n  for (const scheme of compiled.schemes) {\n    // Hydrate List schemes into List plans\n    if (scheme.type === \"each\") {\n      const itemPrototype = getByNodeRef(scheme.nodeRef);\n      const listContainer = itemPrototype.parentElement;\n\n      if (!listContainer) {\n        throw new PaletteError(ErrorCode.TEMPLATE_MISSING_LIST_PARENT);\n      }\n\n      // Create list start/end markers\n      const startMarker = document.createComment(\"list-start\");\n      const endMarker = document.createComment(\"list-end\");\n\n      listContainer.insertBefore(startMarker, itemPrototype);\n      listContainer.insertBefore(endMarker, itemPrototype.nextSibling);\n\n      // Shallow copy of the row root\n      const rowRootElement = itemPrototype.cloneNode(false) as HTMLElement;\n\n      // Copy the contents of the row into a template\n      const templateContentSrc = htmlToFragment(scheme.listContentHtml);\n      const rowTemplateElement = document.createElement(\"template\");\n      rowTemplateElement.appendChild(templateContentSrc);\n\n      // Remove the original root prototype since we will be rendering new items\n      itemPrototype.remove();\n\n      const plan: ListRenderPlan = {\n        type: \"each\",\n        notation: scheme.notation,\n        keyNotation: scheme.keyNotation,\n        listContainerElement: listContainer,\n        rowTemplateElement,\n        rowRootElement,\n        startMarker,\n        endMarker,\n        rowTemplatesByRoot: new WeakMap(),\n        rowRootsByKey: new Map(),\n      };\n\n      associatePlan(plan, scheme.notation.base);\n    }\n    // Hydrate Swap schemes into Swap plans\n    else if (scheme.type === \"swap\") {\n      const plan: ContentSwapPlan = {\n        type: \"swap\",\n        notation: scheme.notation,\n        node: getByNodeRef(scheme.nodeRef),\n      };\n\n      associatePlan(plan, scheme.notation.base);\n    }\n\n    // Hydrate Tag change plans\n    else if (scheme.type === \"tag\") {\n      const plan: TagChangePlan = {\n        type: \"tag\",\n        notation: scheme.notation,\n        node: getByNodeRef(scheme.nodeRef),\n      };\n\n      associatePlan(plan, plan.notation.base);\n    }\n\n    // Hydrate Attribute update plans\n    else if (scheme.type === \"attr\") {\n      const plan: AttributeUpdatePlan = {\n        type: \"attr\",\n        attribute: scheme.attribute,\n        notation: scheme.notation,\n        node: getByNodeRef(scheme.nodeRef),\n      };\n\n      associatePlan(plan, plan.notation.base);\n    }\n  }\n\n  return {\n    plansByNotation,\n    notationAccessors,\n    fragment,\n  };\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n\nimport { ErrorCode, PaletteError } from \"#error\";\nimport { classify } from \"#helper\";\nimport {\n  applyAttributeMap,\n  changeTag,\n  createAttributeMap,\n  serializeAttribute,\n} from \"#util\";\nimport { buildValueAccessor } from \"./syntax\";\nimport { Template, type TemplateContext } from \"./template\";\nimport type {\n  AttributeUpdateJob,\n  AttributeUpdatePlan,\n  ContentSwapJob,\n  ContentSwapPlan,\n  ListRenderJob,\n  ListRenderJobItem,\n  ListRenderPlan,\n  TagChangeJob,\n  TagChangePlan,\n} from \"./types\";\n\nexport const SKIPPED_JOB = Symbol(\"noop\");\nexport type SkippedJob = typeof SKIPPED_JOB;\n\n///////////////////////////\n// ATTRIBUTE UPDATE JOBS //\n///////////////////////////\n\nexport function prepareAttributeJob(\n  plan: AttributeUpdatePlan,\n  values: Map<string, unknown>,\n): AttributeUpdateJob | SkippedJob {\n  const rawValue = values.get(plan.notation.base);\n  let value = typeof rawValue === \"function\" ? rawValue(plan.node) : rawValue;\n\n  // Parse the value with `classify` if setting the class attribute\n  if (plan.attribute === \"class\" && typeof value !== \"string\") {\n    value = classify(value);\n  }\n\n  const serializedValue = serializeAttribute(value);\n  const current = plan.node.getAttribute(plan.attribute);\n\n  if (serializedValue !== current) {\n    return [plan.node, plan.attribute, serializedValue];\n  }\n\n  return SKIPPED_JOB;\n}\n\nexport function runAttributeJobs(jobs: AttributeUpdateJob[]): void {\n  for (const [node, name, value] of jobs) {\n    if (value === null) {\n      node.removeAttribute(name);\n    } else {\n      node.setAttribute(name, value);\n    }\n  }\n}\n\n///////////////////////\n// CONTENT SWAP JOBS //\n///////////////////////\n\nexport function prepareContentSwapJob(\n  plan: ContentSwapPlan,\n  values: Map<string, unknown>,\n): ContentSwapJob | SkippedJob {\n  const rawValue = values.get(plan.notation.base);\n  const value = typeof rawValue === \"function\" ? rawValue(plan.node) : rawValue;\n\n  // String, Number, Boolean -> Text Node\n  if (\n    typeof value === \"string\" ||\n    typeof value === \"number\" ||\n    typeof value === \"boolean\"\n  ) {\n    const asStr = `${value}`;\n    if (plan.node.textContent === asStr) {\n      return SKIPPED_JOB;\n    }\n\n    return [plan, document.createTextNode(asStr)];\n  }\n\n  // Nullish values -> Comment\n  else if (value === null || value === undefined) {\n    if (plan.node instanceof Comment) {\n      return SKIPPED_JOB;\n    }\n\n    return [plan, document.createComment(\"empty\")];\n  }\n\n  // HTMLElement -> Swap nodes\n  else if (value instanceof HTMLElement) {\n    if (Object.is(plan.node, value)) {\n      return SKIPPED_JOB;\n    }\n\n    return [plan, value.cloneNode(true)];\n  }\n\n  // Anything else -> Unsupported\n  else {\n    throw new PaletteError(\n      ErrorCode.INVALID_CONTENT,\n      typeof value,\n      JSON.stringify(value),\n    );\n  }\n}\n\nexport function runContentSwapJobs(jobs: ContentSwapJob[]): void {\n  for (const [plan, replace] of jobs) {\n    const parent = plan.node.parentNode;\n\n    if (!parent) {\n      // Should never happen in practice\n      throw new PaletteError(ErrorCode.INVARIANT);\n    }\n\n    parent.replaceChild(replace, plan.node);\n    plan.node = replace;\n  }\n}\n\n/////////////////////\n// TAG CHANGE JOBS //\n/////////////////////\n\nexport function prepareTagChangeJob(\n  plan: TagChangePlan,\n  values: Map<string, unknown>,\n): TagChangeJob | SkippedJob {\n  const rawValue = values.get(plan.notation.base) ?? \"\";\n  const value = typeof rawValue === \"function\" ? rawValue(plan.node) : rawValue;\n  const nextTag = `${value}`.toLowerCase();\n  const currentTag = plan.node.tagName.toLowerCase();\n\n  if (!nextTag || nextTag === currentTag) {\n    return SKIPPED_JOB;\n  }\n\n  return [plan, nextTag];\n}\n\nexport function runTagChangeJobs(jobs: TagChangeJob[]): void {\n  for (const [plan, nextTag] of jobs) {\n    plan.node = changeTag(plan.node, nextTag);\n  }\n}\n\n//////////////////////\n// LIST RENDER JOBS //\n//////////////////////\n\nfunction fetchListData(\n  plan: ListRenderPlan,\n  values: Map<string, unknown>,\n): Record<string, unknown>[] {\n  const notationValue = values.get(plan.notation.base);\n  const evaluatedValue =\n    typeof notationValue === \"function\"\n      ? notationValue(plan.startMarker.parentNode)\n      : notationValue;\n\n  if (evaluatedValue === null || evaluatedValue === undefined) {\n    return [];\n  }\n\n  if (Array.isArray(evaluatedValue)) {\n    return evaluatedValue;\n  }\n\n  throw new PaletteError(ErrorCode.TEMPLATE_INVALID_LIST);\n}\n\nfunction collectHTMLElements(start: Node, end: Node): Set<HTMLElement> {\n  let node: Node | null = start;\n  const nodeSet = new Set<HTMLElement>();\n\n  while (node && node !== end) {\n    if (node instanceof HTMLElement) {\n      nodeSet.add(node);\n    }\n\n    node = node.nextSibling;\n  }\n\n  return nodeSet;\n}\n\nexport function prepareListRenderJob(\n  plan: ListRenderPlan,\n  values: Map<string, unknown>,\n): ListRenderJob | SkippedJob {\n  // Fetch the data for rendering the updated list items\n  const listData = fetchListData(plan, values);\n\n  // Empty -> Remove the list contents entirely and exit\n  if (listData.length === 0) {\n    return [plan, []];\n  }\n\n  // Collect existing elements from the DOM for assessment\n  const { rowTemplatesByRoot, rowTemplateElement, rowRootElement } = plan;\n  const keyAccessor = buildValueAccessor(plan.keyNotation);\n  const listRenderContexts: Map<unknown, ListRenderJobItem> = new Map();\n\n  for (const item of listData) {\n    const context: TemplateContext = {\n      item,\n    };\n\n    const key = keyAccessor(context);\n\n    if (listRenderContexts.has(key)) {\n      throw new PaletteError(\n        ErrorCode.TEMPLATE_DUPLICATE_LIST_KEY,\n        String(key),\n      );\n    }\n\n    // Get or create the HTML Element we expect to see here\n    let rowRoot = plan.rowRootsByKey.get(key);\n\n    if (rowRoot === undefined) {\n      rowRoot = rowRootElement.cloneNode(false) as HTMLElement;\n      plan.rowRootsByKey.set(key, rowRoot);\n    }\n\n    // Get or create the template associated with the row root\n    let rowTemplate = rowTemplatesByRoot.get(rowRoot);\n\n    if (rowTemplate === undefined) {\n      rowTemplate = new Template(rowTemplateElement);\n      rowTemplatesByRoot.set(rowRoot, rowTemplate);\n    }\n\n    const jobItem: ListRenderJobItem = {\n      context,\n      root: rowRoot,\n      template: rowTemplate,\n      attrMap: createAttributeMap(item),\n    };\n\n    listRenderContexts.set(key, jobItem);\n  }\n\n  // Clean up cached data\n  for (const [key, root] of plan.rowRootsByKey) {\n    if (!listRenderContexts.has(key)) {\n      plan.rowRootsByKey.delete(key);\n      plan.rowTemplatesByRoot.delete(root);\n    }\n  }\n\n  return [plan, Array.from(listRenderContexts.values())];\n}\n\nexport function runListRenderJobs(jobs: ListRenderJob[]): void {\n  for (const [plan, renderItems] of jobs) {\n    const { endMarker, listContainerElement } = plan;\n\n    // Fast path for empty lists -> Remove everything and continue\n    if (renderItems.length === 0) {\n      for (const root of plan.rowRootsByKey.values()) {\n        root.remove();\n      }\n      plan.rowRootsByKey.clear();\n      continue;\n    }\n\n    // Otherwise, scan the current nodes and modify as needed\n    const existingDOMNodes = Array.from(\n      collectHTMLElements(plan.startMarker, plan.endMarker),\n    );\n\n    const rootsToKeep = new Set<HTMLElement>();\n\n    let existingDOMNodeIdx = 0;\n\n    for (const renderItem of renderItems) {\n      const { root, attrMap, context, template } = renderItem;\n      const existingDOMNode = existingDOMNodes[existingDOMNodeIdx];\n\n      // Spread item as attributes on the root and update the template\n      applyAttributeMap(root, attrMap);\n      template.render(root, context);\n\n      rootsToKeep.add(root);\n\n      // Node is the existing DOM Node, continue\n      if (Object.is(existingDOMNode, root)) {\n        existingDOMNodeIdx += 1;\n        continue;\n      }\n\n      // No existing node -> append item\n      if (!existingDOMNode) {\n        listContainerElement.insertBefore(root, endMarker);\n        continue;\n      }\n\n      // DOM node exists but isn't the one we're looking for. Insert before.\n      listContainerElement.insertBefore(root, existingDOMNode);\n    }\n\n    // Remove any remaining old elements\n    for (const root of existingDOMNodes) {\n      if (!rootsToKeep.has(root)) {\n        root.remove();\n        plan.rowTemplatesByRoot.delete(root);\n      }\n    }\n  }\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport { ErrorCode, PaletteError } from \"../errors\";\nimport { fragmentCache, htmlToFragment } from \"../util/fragments\";\nimport { LRUCache } from \"../util/lru-cache\";\nimport { compileTemplate, prepareCompiledTemplate } from \"./build\";\nimport {\n  prepareAttributeJob,\n  prepareContentSwapJob,\n  prepareListRenderJob,\n  prepareTagChangeJob,\n  runAttributeJobs,\n  runContentSwapJobs,\n  runListRenderJobs,\n  runTagChangeJobs,\n  SKIPPED_JOB,\n} from \"./jobs\";\nimport type { NotationAccessor } from \"./syntax\";\nimport type {\n  AnyRenderPlan,\n  AttributeUpdateJob,\n  CompiledTemplate,\n  ContentSwapJob,\n  ListRenderJob,\n  TagChangeJob,\n} from \"./types\";\n\nexport type TemplateRoot = HTMLElement | ShadowRoot;\nexport type TemplateContext = {\n  attr?: Record<string, string | null>;\n  state?: Record<string, unknown>;\n  data?: Record<string, unknown>;\n  item?: Record<string, unknown>;\n  index?: number;\n};\n\nconst compileCache = new LRUCache<HTMLTemplateElement, CompiledTemplate>();\n\nif (__DEV__ && typeof window !== \"undefined\") {\n  (window as any).__PALETTE_DEVTOOLS__ = {\n    dumpCacheMetrics: () => {\n      console.log(\"Compiled Templates Cache\", compileCache.metrics);\n      console.log(\"HTML Fragments Cache\", fragmentCache.metrics);\n    },\n  };\n}\n\n/**\n * A renderable template from an HTMLTemplateElement\n *\n * Leverages a simple syntax to provide accessors to template data from one of\n * four categories, based on use case:\n *\n * ```\n * `@` -> The `attr` namespace (the host element's attributes)\n * `$` -> The `state` namespace (the host element's reactive state)\n * `*` -> The `data` namespace (the host element's computed data)\n * `#` -> The `item` namespace (list item access with ::each)\n * ```\n *\n * Bind attributes to a value using the `:attribute` syntax:\n *\n * ```html\n * <div :id=\"$id\" :class=\"@classname\"></div>\n * ```\n */\nexport class Template {\n  /**\n   * Properties related to the global Template build cache.\n   *\n   * This is not something typically needed for general app development, but\n   * allows for advanced configuration to tune the caching system if need be.\n   *\n   * Cache sizes default to 500 HTML Template Fragments and compiled Palette\n   * templates (includes nested templates like lists) to cover most application\n   * needs without using much memory.\n   */\n  static cache = {\n    /**\n     * Wipe all cached templates and HTML fragments\n     */\n    clear: (): void => {\n      fragmentCache.clear();\n      compileCache.clear();\n    },\n\n    /**\n     * the default cache capacity of 500 items per cache\n     *\n     * General caching guidance:\n     * - 95% of the time, 500 works fine\n     * - For very large applications, up to 1000 is reasonable\n     * - For smaller apps, dropping down to 100 - 250 may save memory\n     *\n     * Ultimately the impact will be minimal unless your app has very specific\n     * rendering needs.\n     */\n    setCapacity: (maxSize: number): void => {\n      fragmentCache.setCapacity(maxSize);\n      compileCache.setCapacity(maxSize);\n    },\n  };\n\n  private _compiled: CompiledTemplate;\n\n  /**\n   * The original {@link HTMLTemplateElement} used to create this Template.\n   *\n   * @remarks\n   * Make sure to never operate directly on this element. Always clone.\n   */\n  private _templateElement?: HTMLTemplateElement;\n\n  /** The current known root element that this Template is rendered into */\n  private _mountedRoot?: TemplateRoot;\n\n  /** The most recent set value for each notation that has rendered */\n  private _latestRenderedValues: Map<string, unknown> = new Map();\n\n  /** The active HTMLElement which the template is updating on renders */\n  private _fragment: DocumentFragment;\n\n  /** Value Notations -> Set<Render Plans> */\n  private _plansByNotation: Map<string, Set<AnyRenderPlan>> = new Map();\n\n  /** Value Notations -> Accessor functions */\n  private _notationAccessors: Map<string, NotationAccessor> = new Map();\n\n  constructor(template: HTMLTemplateElement) {\n    if (template instanceof HTMLTemplateElement) {\n      this._templateElement = template;\n      const cached = compileCache.get(template);\n      const compiled = cached ?? compileTemplate(template);\n      this._compiled = compiled;\n\n      // Always set the compiled template to move it to the front of the cache\n      compileCache.set(template, compiled);\n    } else {\n      this._compiled = template;\n    }\n\n    // Prepare a fragment from the compiled HTML (cached)\n    const fragment = htmlToFragment(this._compiled.html);\n    const hydrated = prepareCompiledTemplate(this._compiled, fragment);\n\n    this._plansByNotation = hydrated.plansByNotation;\n    this._notationAccessors = hydrated.notationAccessors;\n    this._fragment = hydrated.fragment;\n  }\n\n  /**\n   * Create a full copy of this template for use in rendering to another root\n   */\n  clone = (): Template => {\n    return new Template(this.element());\n  };\n\n  /**\n   * @returns An {@link HTMLTemplateElement} representing this template\n   */\n  element = (): HTMLTemplateElement => {\n    if (this._templateElement) {\n      return this._templateElement;\n    }\n\n    const template = document.createElement(\"template\");\n    template.innerHTML = this._compiled.html;\n    return template;\n  };\n\n  private _collectUpdatedValues = (\n    context: TemplateContext,\n  ): Map<string, unknown> => {\n    const updatedValues = new Map<string, unknown>();\n\n    for (const [rawNotation, accessor] of this._notationAccessors) {\n      const previousValue = this._latestRenderedValues.get(rawNotation);\n      const newValue = accessor(context);\n\n      if (previousValue !== newValue && !Object.is(previousValue, newValue)) {\n        updatedValues.set(rawNotation, newValue);\n      }\n    }\n\n    return updatedValues;\n  };\n\n  private _update(newValues: Map<string, unknown>) {\n    const scheduledPlans: Set<AnyRenderPlan> = new Set();\n\n    const attributeJobs: AttributeUpdateJob[] = [];\n    const contentSwapJobs: ContentSwapJob[] = [];\n    const tagChangeJobs: TagChangeJob[] = [];\n    const listRenderJobs: ListRenderJob[] = [];\n\n    for (const [notation] of newValues) {\n      const affectedPlans = this._plansByNotation.get(notation);\n\n      if (affectedPlans) {\n        for (const plan of affectedPlans) {\n          if (scheduledPlans.has(plan)) {\n            continue;\n          }\n\n          scheduledPlans.add(plan);\n\n          switch (plan.type) {\n            case \"attr\": {\n              const job = prepareAttributeJob(plan, newValues);\n              if (job !== SKIPPED_JOB) {\n                attributeJobs.push(job);\n              }\n              break;\n            }\n            case \"each\": {\n              const job = prepareListRenderJob(plan, newValues);\n              if (job !== SKIPPED_JOB) {\n                listRenderJobs.push(job);\n              }\n              break;\n            }\n            case \"swap\": {\n              const job = prepareContentSwapJob(plan, newValues);\n              if (job !== SKIPPED_JOB) {\n                contentSwapJobs.push(job);\n              }\n              break;\n            }\n            case \"tag\": {\n              const job = prepareTagChangeJob(plan, newValues);\n              if (job !== SKIPPED_JOB) {\n                tagChangeJobs.push(job);\n              }\n              break;\n            }\n            default:\n              // Should be impossible given an exhaustive switch case\n              throw new PaletteError(ErrorCode.INVARIANT);\n          }\n        }\n      }\n    }\n\n    runAttributeJobs(attributeJobs);\n    runListRenderJobs(listRenderJobs);\n    runContentSwapJobs(contentSwapJobs);\n    runTagChangeJobs(tagChangeJobs);\n  }\n\n  /**\n   * Render this template into a target host element.\n   *\n   * @remarks\n   * If no context is provided, the root element is used as the context, which\n   * allows for accessing nested properties of the host element itself.\n   */\n  render = (root: TemplateRoot, context: TemplateContext): void => {\n    if (root !== this._mountedRoot) {\n      this._mountedRoot = root;\n      while (root.firstChild) {\n        root.removeChild(root.firstChild);\n      }\n\n      root.append(this._fragment);\n    }\n\n    const resolvedValues = this._collectUpdatedValues(context);\n\n    if (resolvedValues.size !== 0) {\n      this._update(resolvedValues);\n\n      // Save the values that were used for this render if the render succeeded\n      for (const [key, val] of resolvedValues) {\n        this._latestRenderedValues.set(key, val);\n      }\n    }\n  };\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\nimport { elementMatches } from \"#util\";\n\nexport type EventName = keyof GlobalEventHandlersEventMap;\nexport type EventHandler<K extends EventName = EventName> = (\n  event: GlobalEventHandlersEventMap[K],\n) => void;\n\ntype EventHandlerMap = Map<string, EventHandler>;\n\n/**\n * Managed event delegation to coalesce event listeners for an entire tree into\n * single root-level listeners.\n *\n * @example\n * Listen for events from any child element of the delegated root\n *\n * ```typescript\n * const delegator = new EventDelegator($root);\n * delegator.addDelegatedListener(\"click\", handleClick);\n * ```\n *\n * @example\n * Listen for events from child elements matching `.action`\n *\n * ```typescript\n * const delegator = new EventDelegator($root);\n * delegator.addDelegatedListener(\"click\", handleClick, \".action\");\n * ```\n */\nexport class EventDelegator {\n  private _root: HTMLElement | ShadowRoot | Document;\n  private _rootListeners: EventHandlerMap = new Map();\n  private _events: Map<string, Map<string, Set<EventHandler>>> = new Map();\n\n  constructor(root: HTMLElement | ShadowRoot | Document = document) {\n    this._root = root;\n  }\n\n  private _handlersForEvent(event: string): Map<string, Set<EventHandler>> {\n    let handlerMap = this._events.get(event);\n\n    if (handlerMap === undefined) {\n      handlerMap = new Map();\n      this._events.set(event, handlerMap);\n    }\n\n    return handlerMap;\n  }\n\n  private _handlersForEventAndQuery<E extends EventName>(\n    event: E,\n    query: string,\n  ): Set<EventHandler<E>> {\n    const handlerMap = this._handlersForEvent(event);\n    let handlerSet = handlerMap.get(query);\n\n    if (handlerSet === undefined) {\n      handlerSet = new Set();\n      handlerMap.set(query, handlerSet);\n    }\n\n    return handlerSet;\n  }\n\n  private _createRootListener(eventName: string): void {\n    if (this._rootListeners.has(eventName)) {\n      return;\n    }\n\n    const listener = (event: Event) => {\n      const handlersForEvent = this._handlersForEvent(eventName);\n      const { target } = event;\n\n      if (target === null || !(target instanceof Node)) {\n        return;\n      }\n\n      let node: Node | null = target;\n\n      let stopImmediateCalled = false;\n      const originalFn = event.stopImmediatePropagation;\n      event.stopImmediatePropagation = () => {\n        originalFn.call(event);\n        stopImmediateCalled = true;\n      };\n\n      while (node) {\n        if (node === this._root) {\n          break;\n        }\n\n        // Skip invalid elements\n        if (!(node instanceof Element)) {\n          node = node.parentNode;\n          continue;\n        }\n\n        // Invoke handlers\n        for (const [query, handlers] of handlersForEvent) {\n          if (query === \"*\" || elementMatches(node, query)) {\n            for (const handler of handlers) {\n              handler.call(node, event);\n\n              // Respect .stopImmediatePropagation() and .preventDefault()\n              if (stopImmediateCalled || event.defaultPrevented) {\n                return;\n              }\n            }\n          }\n        }\n\n        // Respect .stopPropagation()\n        if (event.cancelBubble) {\n          return;\n        }\n\n        node = node.parentNode;\n\n        if (node === this._root || node instanceof ShadowRoot) {\n          break;\n        }\n      }\n    };\n\n    this._rootListeners.set(eventName, listener);\n    this._root.addEventListener(eventName, listener);\n  }\n\n  addListener<E extends EventName>(\n    eventName: E,\n    handler: EventHandler<E>,\n    queryFilter: string = \"*\",\n  ): void {\n    this._handlersForEventAndQuery(eventName, queryFilter).add(handler);\n    this._createRootListener(eventName);\n  }\n\n  removeListener(\n    eventName: EventName,\n    handler: EventHandler,\n    queryFilter: string,\n  ): void {\n    const handlerSet = this._handlersForEventAndQuery(eventName, queryFilter);\n    handlerSet.delete(handler);\n\n    // If that was the last handler for that event/query combo, delete the set\n    if (handlerSet.size === 0) {\n      const eventHandlerSets = this._handlersForEvent(eventName);\n      eventHandlerSets.delete(queryFilter);\n    }\n  }\n\n  cleanup = (): void => {\n    for (const [event, handler] of this._rootListeners) {\n      this._root.removeEventListener(event, handler);\n    }\n\n    this._rootListeners.clear();\n  };\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n\nimport { ErrorCode, PaletteError } from \"#error\";\nimport { css } from \"#helper\";\nimport { State } from \"#state\";\nimport { Template, type TemplateContext } from \"#template\";\nimport { createAttributeMap, serializeAttribute } from \"#util\";\nimport { EventDelegator, type EventHandler, type EventName } from \"./delegator\";\nimport type {\n  AttributeMap,\n  ComponentStyles,\n  ComputedProperties,\n  LiveAttributes,\n  StateInitializer,\n} from \"./types\";\n\nconst MAX_SEQUENTIAL_RENDERS = 100;\nconst BLANK_TEMPLATE = document.createElement(\"template\");\n\ntype InternalRenderStatus = {\n  willRender: boolean;\n  isRendering: boolean;\n  sequentialRenders: number;\n  postRenderCallbacks: Set<VoidFunction>;\n  lastRenderedAttributeMap: AttributeMap;\n  trackedAttributeChanges: AttributeMap;\n};\n\n/**\n * The **Component** class extends the base HTMLElement class to provide an\n * ergonomic way to design and develop Web Components with reactive state,\n * managed attributes, and dynamic templating.\n */\nexport abstract class Component<\n  StateShape extends object = {},\n> extends HTMLElement {\n  // ╭────────────────────────────────────────────────────────╮\n  // │                 Static Helper Methods                  │\n  // ╰────────────────────────────────────────────────────────╯\n\n  /**\n   * Register a Component with the browser. Optionally provide an overriding\n   * tag name.\n   */\n  static register(\n    component: CustomElementConstructor | typeof Component,\n    tagName?: string,\n  ): void {\n    if (\n      typeof window === \"undefined\" ||\n      typeof window.customElements === \"undefined\"\n    ) {\n      throw new PaletteError(ErrorCode.INVALID_ENV);\n    }\n\n    const componentTagname =\n      \"tagName\" in component && typeof component.tagName === \"string\"\n        ? component.tagName\n        : null;\n    const computedTagname = tagName ?? componentTagname;\n\n    if (\n      computedTagname === null ||\n      computedTagname.length === 0 ||\n      !computedTagname.includes(\"-\")\n    ) {\n      throw new PaletteError(ErrorCode.INVALID_TAGNAME);\n    }\n\n    if (window.customElements.get(computedTagname)) {\n      throw new PaletteError(ErrorCode.DUPE_TAGNAME, tagName ?? \"\");\n    }\n\n    window.customElements.define(\n      computedTagname,\n      component as CustomElementConstructor,\n    );\n  }\n\n  // ╭────────────────────────────────────────────────────────╮\n  // │       Static Component Configuration Properties        │\n  // ╰────────────────────────────────────────────────────────╯\n\n  /**\n   * The HTML tag name to use for this Component\n   */\n  static tagName: string = \"\";\n\n  /**\n   * An HTML Template element for this element to use.\n   */\n  static template: HTMLTemplateElement | Template = BLANK_TEMPLATE;\n\n  /**\n   * A {@link CSSStyleSheet} array to adopt to the element's shadow DOM\n   */\n  static styles: ComponentStyles = [];\n\n  /**\n   * An array of html attribute names to observe to trigger updates.\n   *\n   * @remarks\n   * If you use the {@link define} factory helper, this value will be\n   * automatically merged with a list of any defined {@link liveAttributes}.\n   */\n  static observedAttributes: string[] = [];\n\n  /**\n   * The mode for this Component's ShadowRoot.\n   *\n   * - `\"open\"` allows JS access to the children of this component\n   * - `\"closed\"` (default) disallows internal JS access\n   */\n  static shadowRootMode: \"closed\" | \"open\" = \"closed\";\n\n  // ╭────────────────────────────────────────────────────────╮\n  // │                  Lifecycle Properties                  │\n  // ╰────────────────────────────────────────────────────────╯\n\n  /**\n   * Primary scripting for this component.\n   *\n   * Use `script()` to define primary behaviors, listeners, and initializations.\n   *\n   * If need be, return a function to automatically call when the component is\n   * unmounting for cleanup scripts. Alternatively, use {@link finalize} for\n   * cleanup tasks, which will always run last, even after the returned function\n   * from script().\n   */\n  protected script?(): VoidFunction | void;\n\n  /**\n   * Called immediately before each template update\n   */\n  protected beforeUpdate?(): void;\n\n  /**\n   * Called immediately after each template update\n   */\n  protected afterUpdate?(previousAttributes: Map<string, string | null>): void;\n\n  /**\n   * Called as the final lifecycle method at the end of the unmounting process.\n   *\n   * Content has been removed from the DOM and the component is no longer\n   * considered mounted when this function runs.\n   *\n   * Use this for final cleanup tasks and freeing resources.\n   */\n  protected finalize?(): void;\n\n  /**\n   * If defined, receives errors caught during the lifecycle methods of this\n   * component and it's children. If not defined, this component will instead\n   * throw errors as they are found.\n   *\n   * If the error cannot be handled, `onError()` should re-throw it for a higher\n   * up component to evaluate.\n   *\n   * To display an error message while removing an erroring component from the\n   * DOM, use `this.replaceWith()` and a newly constructed placeholder element.\n   *\n   * @example\n   *\n   * Recovering from errors or displaying a fallback\n   *\n   * ```typescript\n   * onError(error) {\n   *   // Handle known errors\n   *   if (error.message = \"Invalid Input\") {\n   *     this.state.input = \"\";\n   *     return;\n   *   }\n   *\n   *   // Display a fallback banner\n   *   const banner = document.createElement(\"div\");\n   *   banner.textContent = \"An unexpected error done did\";\n   *   this.replaceWith(banner);\n   * }\n   * ```\n   */\n  protected onError?(error: unknown): void;\n\n  /**\n   * The initial value for this Component's reactive internal state, if any.\n   *\n   * If `initialState` is defined as a function, the function is evaluated at\n   * mounting time and the return value is used as the initial state.\n   */\n  protected initialState?: StateInitializer<StateShape>;\n\n  /**\n   * An object or function returning an object which populates the `data` values\n   * in the templating engine. Such values are accessed using the `*` notation.\n   *\n   * When a hosted value is accessed, it's value is coerced based on the way it\n   * is being used. For attributes, values are serialized to a string or null,\n   * which will set the attribute (string) or remove it (null).\n   *\n   * For content swaps, strings, numbers and booleans are rendered as strings\n   * and set as textContent. null and undefined values render nothing (with a\n   * comment placeholder), an HTMLElement will be swapped in as a cloned node.\n   *\n   * Functions are evaluated at render/update time, and their return value is\n   * used instead. Functions are passed the current target node accessing the\n   * property as its first and only argument.\n   *\n   * Other types are unsupported.\n   *\n   * @example\n   *\n   * ```typescript\n   * host = {\n   *   avatarUrl: \"http://...\",\n   *   renderTime: () => new Date().toISOString(),\n   * }\n   * ```\n   *\n   * ```html\n   * <img :src=\"*avatarUrl\" />\n   * <span>Rendered at ${\"*renderTime\"}</span>\n   * ```\n   */\n  protected computedProperties?: ComputedProperties<StateShape>;\n\n  /**\n   * Declaration of attribute behaviors during updates.\n   *\n   * Each key is the name of an attribute. Values are an object with two\n   * optional function definitions:\n   *\n   * `onChange(newValue, oldValue): void`, a handler function for when the value\n   * of this attribute has changed. Runs once per update cycle, *before* the\n   * update occurs.\n   *\n   * `reflect(): unknown`, a function which returns a value to serialize and set\n   * as this attribute *after* an update completes.\n   *\n   * @example\n   *\n   * ```typescript\n   * liveAttributes = {\n   *   done: {\n   *     onChange(newValue) {\n   *       this.state.set(\"done\", !!newValue);\n   *     },\n   *     reflect() { return this.state.current.done }\n   *   }\n   * };\n   * ```\n   *\n   * @remarks\n   *\n   * If using the {@link define} helper, the keys of this property are\n   * automatically merged with any defined {@link observedAttributes}.\n   *\n   * If you are directly extending the class, `observedAttributes` must be\n   * explicitly defined with any attributes that should be monitored, regardless\n   * of if they're defined here.\n   */\n  protected liveAttributes?: LiveAttributes<StateShape>;\n\n  // ╭────────────────────────────────────────────────────────╮\n  // │               Internal State Management                │\n  // ╰────────────────────────────────────────────────────────╯\n\n  private _isComponentMounted: boolean = false;\n\n  private _template: Template;\n  private _root: ShadowRoot;\n  private _state?: State<StateShape>;\n  private _delegator?: EventDelegator;\n  private _ownListeners: Set<{\n    handler: EventHandler;\n    event: EventName;\n  }> = new Set();\n  private _liveAttributeConfigs: Map<\n    string,\n    LiveAttributes<StateShape>[string]\n  > = new Map();\n  private _nextListenerAutoId = 0;\n  private _cleanupFn?: VoidFunction;\n\n  private _renderInternals: InternalRenderStatus = {\n    willRender: false,\n\n    /** When set to `true`, the render/update process is actively running */\n    isRendering: false,\n    sequentialRenders: 0,\n    postRenderCallbacks: new Set(),\n    lastRenderedAttributeMap: new Map(),\n    trackedAttributeChanges: new Map(),\n  };\n\n  // ╭────────────────────────────────────────────────────────╮\n  // │             Internals and Private Methods              │\n  // ╰────────────────────────────────────────────────────────╯\n\n  constructor() {\n    super();\n\n    const { template, styles, shadowRootMode } = this\n      .constructor as typeof Component;\n\n    // Normalize styles to an array of CSSStyleSheet objects\n    const newStyles = Array.isArray(styles)\n      ? styles\n      : typeof styles === \"object\"\n        ? [styles]\n        : [css`${styles}`];\n\n    this._root = this.attachShadow({ mode: shadowRootMode });\n    this._root.adoptedStyleSheets.push(...newStyles);\n    this._template =\n      template instanceof Template ? template : new Template(template);\n  }\n\n  /** Produce the data object to pass to the template's render function */\n  private _getRenderContext(attr: Map<string, string | null>): TemplateContext {\n    // Computed properties\n    const data =\n      typeof this.computedProperties === \"function\"\n        ? this.computedProperties()\n        : this.computedProperties;\n\n    // Current state\n    const state = this.initialState ? this._safeGetState().current : undefined;\n\n    return {\n      data,\n      attr: Object.fromEntries(attr),\n      state,\n    };\n  }\n\n  private _safeGetState(): State<StateShape> {\n    if (!this.initialState || !this._state) {\n      throw new PaletteError(ErrorCode.MISSING_STATE);\n    }\n\n    return this._state;\n  }\n\n  private _reportError(error: unknown): void {\n    // Errors during resurrection cause an immediate throw, as does a lack of\n    // an onError handler\n    if (!this.onError) {\n      throw error;\n    }\n\n    this.onError(error);\n  }\n\n  private _executeAttributeChangeHandlers = () => {\n    const prevAttrs = this._renderInternals.lastRenderedAttributeMap;\n\n    // Copy the changes to a new map and clear the tracking map\n    const trackedChanges = new Map(\n      this._renderInternals.trackedAttributeChanges,\n    );\n    this._renderInternals.trackedAttributeChanges.clear();\n\n    for (const [name, val] of trackedChanges) {\n      const config = this._liveAttributeConfigs.get(name);\n      if (!config) {\n        continue;\n      }\n\n      if (\"onChange\" in config) {\n        const previousValue = prevAttrs.get(name) ?? null;\n\n        if (config && previousValue !== val) {\n          config.onChange?.call(this, val, previousValue);\n        }\n      }\n    }\n  };\n\n  private _reflectLiveAttributes = () => {\n    const toSet: [name: string, val: string | null][] = [];\n\n    for (const [name, config] of this._liveAttributeConfigs) {\n      if (\"reflect\" in config) {\n        const val = serializeAttribute(config.reflect?.call(this));\n        const current = this.getAttribute(name);\n\n        if (val !== current) {\n          toSet.push([name, val]);\n        }\n      }\n    }\n\n    for (const [name, val] of toSet) {\n      this.reflectAttribute(name, val);\n    }\n  };\n\n  /** The actual update behavior */\n  private _performUpdate = (): void => {\n    try {\n      // Immediately flip flags to denote a render is in progress and allow\n      // queueing another render request.\n      this._renderInternals.isRendering = true;\n      this._renderInternals.willRender = false;\n\n      // Attributes: Run any `read` hooks for changed attributes\n      this._executeAttributeChangeHandlers();\n\n      // Cache the attributes from the previous render\n      const prevAttrs = this._renderInternals.lastRenderedAttributeMap;\n\n      this.beforeUpdate?.call(this);\n\n      // Update the current snapshot of attributes\n      const newAttributes = createAttributeMap(this);\n      this._template.render(this.root, this._getRenderContext(newAttributes));\n      this._renderInternals.lastRenderedAttributeMap = newAttributes;\n      this.afterUpdate?.call(this, prevAttrs);\n\n      // Attributes: Run any `write` hooks for updated values\n      this._reflectLiveAttributes();\n\n      // Run any requested render callbacks\n      // These are considered lifecycle methods as well, and will report errors\n      // as any other lifecycle method would\n      if (this._renderInternals.postRenderCallbacks.size) {\n        const callbacks = Array.from(this._renderInternals.postRenderCallbacks);\n        this._renderInternals.postRenderCallbacks.clear();\n\n        for (const fn of callbacks) {\n          fn();\n        }\n      }\n\n      // If the `willRender` flag was flipped, immediately perform a render\n      if (this._renderInternals.willRender) {\n        this._renderInternals.sequentialRenders += 1;\n\n        if (this._renderInternals.sequentialRenders >= MAX_SEQUENTIAL_RENDERS) {\n          throw new PaletteError(\n            ErrorCode.MAX_SEQUENTIAL_RENDERS,\n            `${this._renderInternals.sequentialRenders}`,\n          );\n        }\n\n        this._performUpdate();\n      } else {\n        this._renderInternals.sequentialRenders = 0;\n      }\n    } catch (error) {\n      this._renderInternals.willRender = false;\n      this._reportError(error);\n    } finally {\n      this._renderInternals.isRendering = false;\n    }\n  };\n\n  /** Safely request for an update to run on the next frame */\n  private _scheduleUpdate = () => {\n    // Prevent scheduling multiple renders\n    if (this._renderInternals.willRender) {\n      return;\n    }\n\n    this._renderInternals.willRender = true;\n\n    // If currently rendering, don't request a render, it will auto-continue\n    // until there are no changes to render\n    if (this._renderInternals.isRendering) {\n      return;\n    }\n\n    // If a render is requested and we're not rendering, request a render frame\n    requestAnimationFrame(this._performUpdate);\n  };\n\n  private _adoptState(state: StateShape | State<StateShape>) {\n    // If state exists, remove the listener and schedule a render\n    if (this._state) {\n      this._state.removeListener(this._scheduleUpdate);\n      this._scheduleUpdate();\n    }\n\n    // Adopt the incoming state obj or wrap and adopt it\n    const incomingStateObj = state instanceof State ? state : new State(state);\n    this._state = incomingStateObj;\n\n    // Ensure there is not accidentally two of the same render listeners\n    this._state.removeListener(this._scheduleUpdate);\n    this._state.addListener(this._scheduleUpdate);\n  }\n\n  // ╭────────────────────────────────────────────────────────╮\n  // │           Custom Element Callback Overrides            │\n  // ╰────────────────────────────────────────────────────────╯\n\n  /**\n   * @deprecated Use {@link script} instead\n   */\n  protected connectedCallback(): void {\n    if (this._isComponentMounted) {\n      return;\n    }\n\n    try {\n      const attributes = createAttributeMap(this);\n      this._liveAttributeConfigs = new Map(\n        Object.entries(this.liveAttributes ?? {}),\n      );\n\n      // If this is a stateful component, set up that state\n      if (this.initialState) {\n        const init =\n          typeof this.initialState === \"object\"\n            ? this.initialState\n            : this.initialState();\n\n        this._adoptState(init);\n      }\n\n      this._template.render(this.root, this._getRenderContext(attributes));\n      this._reflectLiveAttributes();\n      this._renderInternals.lastRenderedAttributeMap = attributes;\n      this._isComponentMounted = true;\n      this._cleanupFn = this.script?.call(this) ?? undefined;\n    } catch (error) {\n      this._reportError(error);\n    }\n  }\n\n  /**\n   * @deprecated use {@link finalize}\n   */\n  protected disconnectedCallback(): void {\n    queueMicrotask(() => {\n      // Only run the disconnect if the element is actually unmounted and not\n      // just moved. Using this in place of connectedMoveCallback() for compat\n      if (!this.isConnected && this._isComponentMounted) {\n        try {\n          this._delegator?.cleanup();\n\n          // Clear any listeners set up on self\n          if (this._ownListeners.size > 0) {\n            for (const listener of this._ownListeners) {\n              this.removeEventListener(listener.event, listener.handler);\n            }\n          }\n\n          this._cleanupFn?.call(this);\n          this._isComponentMounted = false;\n        } catch (error) {\n          this._reportError(error);\n        } finally {\n          // Run the finalize call after everything, even if the cleanup fn failed\n          try {\n            this.finalize?.call(this);\n          } catch (error) {\n            this._reportError(error);\n          }\n        }\n      }\n    });\n  }\n\n  /**\n   * @deprecated Use {@link liveAttributes} instead\n   */\n  protected attributeChangedCallback(\n    name: string,\n    _: string | null,\n    newValue: string | null,\n  ): void {\n    // Ignore attribute changes that occur during an update, as they are\n    // coming from inside the house\n    if (this._renderInternals.isRendering) {\n      return;\n    }\n\n    // Skip non-changes using the stored attributes as reference to compare\n    // to the render state rather than the dom\n    if (\n      newValue === this._renderInternals.lastRenderedAttributeMap?.get(name)\n    ) {\n      return;\n    }\n\n    // Cache the updated attribute value\n    this._renderInternals.trackedAttributeChanges.set(name, newValue);\n\n    // Schedule a render\n    if (!this._renderInternals.willRender) {\n      this._scheduleUpdate();\n    }\n  }\n\n  // ╭────────────────────────────────────────────────────────╮\n  // │                    Public Interface                    │\n  // ╰────────────────────────────────────────────────────────╯\n\n  /**\n   * The ShadowRoot node of this element\n   */\n  get root(): ShadowRoot {\n    return this._root;\n  }\n\n  /**\n   * Directly access and manipulate this Component's reactive state with deep\n   * reactivity enabled.\n   *\n   * @example\n   *\n   * ```typescript\n   * initialState = { values: { count: 1, time: 0 } };\n   *\n   * increment() {\n   *   this.state.values.count = 2;\n   * }\n   * ```\n   *\n   * @remarks\n   * This is a shorthand getter for `this.getState().live`\n   */\n  get state(): StateShape {\n    return this._safeGetState().live;\n  }\n\n  /**\n   * Replace the current state with a new value or adopt a whole different State\n   * instance\n   *\n   * @example\n   *\n   * State replacement\n   *\n   * ```typescript\n   * // Always triggers a render\n   * this.state = { some: \"newState\" }\n   *\n   * // Adopt a new state instance altogether\n   * const otherState = new State({some: \"newState\"});\n   * this.state = otherState;\n   *\n   * // Causes a render\n   * otherState.set(\"some\", \"value\");\n   * ```\n   */\n  set state(newState: StateShape | State<StateShape>) {\n    if (!this._state) {\n      throw new PaletteError(ErrorCode.MISSING_STATE);\n    } else if (newState instanceof State) {\n      // Change current internal state for new object\n      this._state = newState;\n      this._state.addListener(() => {\n        this.requestRender();\n      });\n      this.requestRender();\n    } else {\n      this._state.replace(newState);\n    }\n  }\n\n  /**\n   * Access the full {@link State} instance for this Component.\n   *\n   * Using this method on a component which is not using reactive state\n   * will result in an error being thrown.\n   *\n   * @example\n   *\n   * Retrieve stateful value\n   *\n   * ```typescript\n   * class Example extends Component {\n   *   initialState = { count: 1 };\n   *\n   *   script() {\n   *     const state = this.getState();\n   *     console.log(`The count is ${state.get(\"count\")}`);\n   *   }\n   *\n   * }\n   * ```\n   *\n   * @example\n   *\n   * Set a stateful value\n   *\n   * ```typescript\n   * class Example extends Component {\n   *   initialState = { count: 1 };\n   *\n   *   increment() {\n   *     const state = this.getState();\n   *     const current = state.get(\"count\");\n   *     state.set(\"count\", current + 1);\n   *   }\n   * }\n   * ```\n   */\n  getState(): State<StateShape> {\n    return this._safeGetState();\n  }\n\n  /**\n   * Manually schedule a render to occur\n   *\n   * @remarks\n   * Calling {@link requestRender} multiple times in the same event loop won't\n   * schedule multiple renders. Renders are scheduled to occur on the next\n   * animation frame.\n   */\n  requestRender(callback?: VoidFunction): void {\n    if (callback) {\n      this._renderInternals.postRenderCallbacks.add(callback);\n    }\n\n    this._scheduleUpdate();\n  }\n\n  /**\n   * Create a managed event listener on this Component with an optional query\n   * filter string to specify which elements to listen to events from.\n   *\n   * Event handlers set up with `listen()` are automatically cleaned up when\n   * the component is unmounted. Event listeners are executed with this instance\n   * as the `this` binding.\n   *\n   * Handlers are delegated to a single listener per event on the root of the\n   * Component, which is the closed ShadowRoot by default.\n   *\n   * @example\n   *\n   * Set up an event listener for a click event from a button with ID \"submit\"\n   *\n   * ```typescript\n   * this.listen(\"#submit\", \"click\", this.handleClick);\n   * ```\n   *\n   * @example\n   *\n   * Set up an event listener on the host element itself (`this`).\n   *\n   * ```typescript\n   * // Use \":host\" or \":HOST\" as the query selector to targe the host element\n   * this.listen(\":host\", \"animationend\", this.handler);\n   * ```\n   */\n  listen<E extends EventName>(\n    targetDescriptor: string | typeof Component<any> | HTMLElement,\n    eventName: E,\n    eventHandler: EventHandler<E>,\n  ): void {\n    let query = \"\";\n\n    // Create a query filter from the target descriptor\n    if (targetDescriptor instanceof HTMLElement) {\n      // If the element has an ID, use that\n      if (targetDescriptor.id.length) {\n        query = `#${targetDescriptor.id}`;\n      } else {\n        const generatedId = `listen-${this._nextListenerAutoId++}`;\n        targetDescriptor.setAttribute(\"data-listener-id\", generatedId);\n        query = `[data-listener-id=\"${generatedId}\"]`;\n      }\n    } else if (typeof targetDescriptor === \"function\") {\n      // If the target descriptor is a Component class, pull the tagname\n      query = targetDescriptor.tagName;\n\n      if (__DEV__) {\n        if (targetDescriptor.tagName.length === 0) {\n          console.warn(\n            `[Component] Automatic listener setup failure: Component ${targetDescriptor.name} has no defined tagname`,\n          );\n        }\n      }\n    } else {\n      query = targetDescriptor;\n    }\n\n    // If the query filter is `\":host\"`, set up an ownListener\n    if (query === \":host\" || query === \":HOST\") {\n      const handler = eventHandler.bind(this) as EventHandler;\n      this.addEventListener(eventName, handler);\n      this._ownListeners.add({\n        handler,\n        event: eventName,\n      });\n      return;\n    }\n\n    // Otherwise, set up a root listener\n    if (!this._delegator) {\n      this._delegator = new EventDelegator(this.root);\n    }\n\n    this._delegator.addListener(eventName, eventHandler.bind(this), query);\n  }\n\n  dispatchEvent(event: Event): boolean;\n\n  /**\n   * Dispatch a {@link CustomEvent} with the specified name and detail\n   *\n   * @example Defining custom events\n   *\n   * ```typescript\n   * // Define types. This can happen in the same file or elsewhere for common\n   * // and shared events.\n   * export type MyEvent = CustomEvent<T> // Set `T` to the event detail type\n   *\n   * declare global {\n   *  interface GlobalEventHandlersEventMap {\n   *    \"my-event-name\": MyEvent\n   *  }\n   * }\n   *\n   * // Dispatch a custom event\n   * this.dispatchEvent(\"my-event-name\", T);\n   *\n   * // Listen to a custom event\n   * $el.addEventListener(\"my-event-name\", ...);\n   * ```\n   */\n  dispatchEvent<T>(\n    event: string,\n    detail?: T,\n    options?: {\n      bubbles: boolean;\n      cancelable: boolean;\n      composed: boolean;\n    },\n  ): boolean;\n  dispatchEvent<T>(\n    event: string | Event,\n    detail?: T,\n    options?: {\n      bubbles: boolean;\n      cancelable: boolean;\n      composed: boolean;\n    },\n  ): boolean {\n    if (event instanceof Event) {\n      return super.dispatchEvent(event);\n    }\n\n    const opts = {\n      bubbles: true,\n      cancelable: true,\n      composed: true,\n      ...(options ?? {}),\n    };\n\n    return super.dispatchEvent(\n      new CustomEvent<T>(event, {\n        detail,\n        ...opts,\n      }),\n    );\n  }\n\n  querySelector<E extends HTMLElement>(query: string): E | null {\n    return this._root.querySelector(query);\n  }\n\n  querySelectorAll<E extends HTMLElement>(query: string): NodeListOf<E> {\n    return this._root.querySelectorAll(query) as NodeListOf<E>;\n  }\n\n  getElementById<E extends HTMLElement>(id: string): E | null {\n    if (this._root instanceof ShadowRoot) {\n      return this._root.getElementById(id) as E | null;\n    } else {\n      return HTMLElement.prototype.querySelector.call(\n        this,\n        `#${id}`,\n      ) as E | null;\n    }\n  }\n\n  requireElementById<E extends HTMLElement>(id: string): E {\n    const lookup = this.getElementById<E>(id);\n    if (!lookup) {\n      throw new PaletteError(ErrorCode.MISSING_ELEMENT);\n    }\n    return lookup;\n  }\n\n  setAttribute(name: string, value: string): void;\n\n  /**\n   * Set a value as an attribute, serializing to a string or `null`.\n   *\n   * null values cause the attribute to be removed from this element\n   */\n  setAttribute(name: string, value: unknown): void;\n  setAttribute(name: string, value: unknown): void {\n    const serialized = serializeAttribute(value);\n    if (serialized === null) {\n      HTMLElement.prototype.removeAttribute.call(this, name);\n    } else {\n      HTMLElement.prototype.setAttribute.call(this, name, serialized);\n    }\n  }\n\n  getAttribute(name: string): string | null;\n\n  /**\n   * Get an attribute as a number, falling back to a specified default.\n   *\n   *\n   * `null` attributes are returned as `null`, while string attributes are\n   * parsed using the `Number()` constructor.\n   *\n   * Values which evaluate to NaN are returned as `null`.\n   */\n  getAttribute(name: string, defaultValue: number): number;\n\n  /**\n   * Get an attribute as a string, falling back to a specified default\n   */\n  getAttribute(name: string, defaultValue: string): string;\n  getAttribute(\n    name: string,\n    defaultValue?: string | number,\n  ): string | number | null {\n    const value = HTMLElement.prototype.getAttribute.call(this, name);\n\n    if (defaultValue === undefined) {\n      return value;\n    }\n\n    if (typeof defaultValue === \"number\") {\n      if (value === null) {\n        return defaultValue;\n      }\n\n      const parsed = Number(value);\n\n      if (Number.isNaN(parsed)) {\n        return defaultValue;\n      }\n\n      return parsed;\n    }\n\n    // Invalid function invocation\n    throw new PaletteError(ErrorCode.INVARIANT);\n  }\n\n  /**\n   * Set the value of an attribute on this element if the provided value\n   * differs from the attribute when serialized\n   */\n  reflectAttribute(name: string, value: unknown): void {\n    const current = this.getAttribute(name);\n    const reflected = serializeAttribute(value);\n\n    if (current !== reflected) {\n      this.setAttribute(name, reflected);\n    }\n  }\n\n  /**\n   * Serialize this element to a string\n   */\n  toString(full: boolean = false): string {\n    const name = (this.constructor as typeof Component).name;\n    const attrString = this.getAttributeNames()\n      .map((attrName) => {\n        const val = this.getAttribute(attrName);\n        return val ? `${attrName}=\"${val}\"` : \"\";\n      })\n      .join(\" \");\n    const attrs = attrString.length > 0 ? ` ${attrString}` : \"\";\n\n    if (full) {\n      return `Component<${name}${attrs}>\\n${this.root.innerHTML}\\n</${name}>`;\n    }\n\n    return `Component<${name}${attrs}>`;\n  }\n}\n",
    "/*\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at https://mozilla.org/MPL/2.0/.\n */\n\nimport { Component } from \"./component\";\nimport type {\n  AttributeMap,\n  ComponentStyles,\n  ComputedProperties,\n  LiveAttributes,\n  StateInitializer,\n} from \"./types\";\n\nconst BLANK_TEMPLATE = document.createElement(\"template\");\n\nexport interface ComponentShorthand<StateShape extends object = {}> {\n  /**\n   * An HTML Template element for this element to use.\n   */\n  template?: HTMLTemplateElement;\n\n  /**\n   * A {@link CSSStyleSheet} array to adopt to the element's shadow DOM\n   */\n  styles?: ComponentStyles;\n\n  /**\n   * The mode for this Component's ShadowRoot.\n   *\n   * - `\"open\"` allows JS access to the children of this component\n   * - `\"closed\"` (default) disallows internal JS access\n   */\n  shadowRootMode?: \"closed\" | \"open\";\n\n  /**\n   * The initial state for this Component's reactive internal state, if any.\n   */\n  initialState?: StateInitializer<StateShape>;\n\n  /**\n   * An object or function returning an object which populates the `data` values\n   * in the templating engine. Such values are accessed using the `*` notation.\n   */\n  computedProperties?: ComputedProperties<StateShape>;\n\n  /**\n   * Declaration of attribute behaviors during updates.\n   */\n  liveAttributes?: LiveAttributes<StateShape>;\n\n  /**\n   * Initialization scripting for this Component\n   */\n  script?: (this: Component<StateShape>) => VoidFunction | void;\n\n  /**\n   * Called immediately before each template update\n   */\n  beforeUpdate?: (this: Component<StateShape>) => void;\n\n  /**\n   * Called immediately after each template update\n   */\n  afterUpdate?: (\n    this: Component<StateShape>,\n    previousAttributes: AttributeMap,\n  ) => void;\n\n  /**\n   * Called as the final lifecycle method at the end of the unmounting process.\n   *\n   * Content has been removed from the DOM and the component is no longer\n   * considered mounted when this function runs.\n   *\n   * Use this for final cleanup tasks and freeing resources.\n   */\n  finalize?: (this: Component<StateShape>) => void;\n\n  /**\n   * If defined, receives errors caught during the lifecycle methods of this\n   * component and it's children. If not defined, this component will instead\n   * throw errors as they are found.\n   *\n   * If the error cannot be handled, `onError()` should re-throw it for a higher\n   * up component to evaluate.\n   *\n   * To display an error message while removing an erroring component from the\n   * DOM, use `this.replaceWith()` and a newly constructed placeholder element.\n   *\n   * @example\n   *\n   * Recovering from errors or displaying a fallback\n   *\n   * ```typescript\n   * onError(error) {\n   *   // Handle known errors\n   *   if (error.message = \"Invalid Input\") {\n   *     this.state.input = \"\";\n   *     return;\n   *   }\n   *\n   *   // Display a fallback banner\n   *   const banner = document.createElement(\"div\");\n   *   banner.textContent = \"An unexpected error done did\";\n   *   this.replaceWith(banner);\n   * }\n   * ```\n   */\n  onError?: (this: Component<StateShape>, error: unknown) => void;\n}\n\nexport function define<StateShape extends object = {}>(\n  tagname: string,\n  definition: ComponentShorthand<StateShape>,\n): typeof Component<StateShape> {\n  const {\n    // Config\n    template = BLANK_TEMPLATE,\n    styles = [],\n    shadowRootMode = \"closed\",\n    initialState,\n    computedProperties = {},\n    liveAttributes = {},\n\n    // Lifecycle\n    script,\n    beforeUpdate,\n    afterUpdate,\n    finalize,\n    onError,\n  } = definition;\n\n  const fullObservedAttributes = Array.from(\n    new Set([...Object.keys(liveAttributes)]),\n  );\n\n  const cls = class extends Component<StateShape> {\n    static tagName = tagname;\n    static template = template;\n    static styles = styles;\n    static observedAttributes = [...fullObservedAttributes];\n    static shadowRootMode = shadowRootMode;\n\n    constructor() {\n      super();\n\n      this.initialState =\n        typeof initialState === \"function\"\n          ? initialState.bind(this)\n          : initialState;\n      this.computedProperties =\n        typeof computedProperties === \"function\"\n          ? computedProperties.bind(this)\n          : computedProperties;\n      this.liveAttributes = liveAttributes;\n\n      for (const def of Object.values(this.liveAttributes)) {\n        if (def.onChange) {\n          def.onChange = def.onChange.bind(this);\n        }\n\n        if (def.reflect) {\n          def.reflect = def.reflect.bind(this);\n        }\n      }\n\n      this.script = script?.bind(this) ?? undefined;\n      this.beforeUpdate = beforeUpdate?.bind(this) ?? undefined;\n      this.afterUpdate = afterUpdate?.bind(this) ?? undefined;\n      this.finalize = finalize?.bind(this) ?? undefined;\n      this.onError = onError?.bind(this) ?? undefined;\n    }\n  };\n\n  Component.register(cls, tagname);\n\n  return cls;\n}\n"
  ],
  "mappings": "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",
  "debugId": "331008C45ADD766764756E2164756E21",
  "names": []
}
|