mvc-kit 2.12.5 → 2.13.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/agent-config/bin/postinstall.mjs +4 -3
- package/agent-config/bin/setup.mjs +5 -1
- package/agent-config/claude-code/agents/mvc-kit-architect.md +11 -8
- package/agent-config/claude-code/skills/guide/SKILL.md +20 -7
- package/agent-config/claude-code/skills/guide/patterns.md +12 -0
- package/agent-config/claude-code/skills/guide/recipes.md +510 -0
- package/agent-config/claude-code/skills/guide/testing.md +297 -0
- package/agent-config/claude-code/skills/review/SKILL.md +3 -13
- package/agent-config/claude-code/skills/review/checklist.md +30 -5
- package/agent-config/claude-code/skills/scaffold/SKILL.md +4 -13
- package/agent-config/lib/install-claude.mjs +84 -25
- package/dist/Channel.cjs +276 -300
- package/dist/Channel.cjs.map +1 -1
- package/dist/Channel.js +275 -299
- package/dist/Channel.js.map +1 -1
- package/dist/Collection.cjs +424 -504
- package/dist/Collection.cjs.map +1 -1
- package/dist/Collection.js +423 -503
- package/dist/Collection.js.map +1 -1
- package/dist/Controller.cjs +70 -67
- package/dist/Controller.cjs.map +1 -1
- package/dist/Controller.js +69 -66
- package/dist/Controller.js.map +1 -1
- package/dist/EventBus.cjs +77 -88
- package/dist/EventBus.cjs.map +1 -1
- package/dist/EventBus.js +76 -87
- package/dist/EventBus.js.map +1 -1
- package/dist/Feed.cjs +81 -77
- package/dist/Feed.cjs.map +1 -1
- package/dist/Feed.js +80 -76
- package/dist/Feed.js.map +1 -1
- package/dist/Model.cjs +181 -207
- package/dist/Model.cjs.map +1 -1
- package/dist/Model.js +179 -205
- package/dist/Model.js.map +1 -1
- package/dist/Pagination.cjs +75 -73
- package/dist/Pagination.cjs.map +1 -1
- package/dist/Pagination.js +74 -72
- package/dist/Pagination.js.map +1 -1
- package/dist/Pending.cjs +255 -287
- package/dist/Pending.cjs.map +1 -1
- package/dist/Pending.js +253 -285
- package/dist/Pending.js.map +1 -1
- package/dist/PersistentCollection.cjs +242 -285
- package/dist/PersistentCollection.cjs.map +1 -1
- package/dist/PersistentCollection.js +241 -284
- package/dist/PersistentCollection.js.map +1 -1
- package/dist/Resource.cjs +166 -174
- package/dist/Resource.cjs.map +1 -1
- package/dist/Resource.js +164 -172
- package/dist/Resource.js.map +1 -1
- package/dist/Selection.cjs +84 -94
- package/dist/Selection.cjs.map +1 -1
- package/dist/Selection.js +83 -93
- package/dist/Selection.js.map +1 -1
- package/dist/Service.cjs +54 -55
- package/dist/Service.cjs.map +1 -1
- package/dist/Service.js +53 -54
- package/dist/Service.js.map +1 -1
- package/dist/Sorting.cjs +102 -101
- package/dist/Sorting.cjs.map +1 -1
- package/dist/Sorting.js +102 -101
- package/dist/Sorting.js.map +1 -1
- package/dist/Trackable.cjs +112 -80
- package/dist/Trackable.cjs.map +1 -1
- package/dist/Trackable.js +111 -79
- package/dist/Trackable.js.map +1 -1
- package/dist/ViewModel.cjs +528 -576
- package/dist/ViewModel.cjs.map +1 -1
- package/dist/ViewModel.js +525 -573
- package/dist/ViewModel.js.map +1 -1
- package/dist/bindPublicMethods.cjs +43 -24
- package/dist/bindPublicMethods.cjs.map +1 -1
- package/dist/bindPublicMethods.js +43 -24
- package/dist/bindPublicMethods.js.map +1 -1
- package/dist/errors.cjs +67 -68
- package/dist/errors.cjs.map +1 -1
- package/dist/errors.js +68 -71
- package/dist/errors.js.map +1 -1
- package/dist/mvc-kit.cjs +44 -46
- package/dist/mvc-kit.js +5 -32
- package/dist/produceDraft.cjs +105 -95
- package/dist/produceDraft.cjs.map +1 -1
- package/dist/produceDraft.js +106 -97
- package/dist/produceDraft.js.map +1 -1
- package/dist/react/components/CardList.cjs +30 -40
- package/dist/react/components/CardList.cjs.map +1 -1
- package/dist/react/components/CardList.js +31 -41
- package/dist/react/components/CardList.js.map +1 -1
- package/dist/react/components/DataTable.cjs +146 -169
- package/dist/react/components/DataTable.cjs.map +1 -1
- package/dist/react/components/DataTable.js +147 -170
- package/dist/react/components/DataTable.js.map +1 -1
- package/dist/react/components/InfiniteScroll.cjs +51 -42
- package/dist/react/components/InfiniteScroll.cjs.map +1 -1
- package/dist/react/components/InfiniteScroll.js +52 -43
- package/dist/react/components/InfiniteScroll.js.map +1 -1
- package/dist/react/components/types.cjs +10 -6
- package/dist/react/components/types.cjs.map +1 -1
- package/dist/react/components/types.js +11 -9
- package/dist/react/components/types.js.map +1 -1
- package/dist/react/guards.cjs +10 -6
- package/dist/react/guards.cjs.map +1 -1
- package/dist/react/guards.js +11 -9
- package/dist/react/guards.js.map +1 -1
- package/dist/react/provider.cjs +23 -20
- package/dist/react/provider.cjs.map +1 -1
- package/dist/react/provider.js +23 -21
- package/dist/react/provider.js.map +1 -1
- package/dist/react/use-event-bus.cjs +24 -20
- package/dist/react/use-event-bus.cjs.map +1 -1
- package/dist/react/use-event-bus.js +24 -21
- package/dist/react/use-event-bus.js.map +1 -1
- package/dist/react/use-instance.cjs +43 -36
- package/dist/react/use-instance.cjs.map +1 -1
- package/dist/react/use-instance.js +43 -36
- package/dist/react/use-instance.js.map +1 -1
- package/dist/react/use-local.cjs +48 -64
- package/dist/react/use-local.cjs.map +1 -1
- package/dist/react/use-local.js +47 -63
- package/dist/react/use-local.js.map +1 -1
- package/dist/react/use-model.cjs +84 -98
- package/dist/react/use-model.cjs.map +1 -1
- package/dist/react/use-model.js +84 -100
- package/dist/react/use-model.js.map +1 -1
- package/dist/react/use-singleton.cjs +19 -23
- package/dist/react/use-singleton.cjs.map +1 -1
- package/dist/react/use-singleton.js +16 -20
- package/dist/react/use-singleton.js.map +1 -1
- package/dist/react/use-subscribe-only.cjs +28 -22
- package/dist/react/use-subscribe-only.cjs.map +1 -1
- package/dist/react/use-subscribe-only.js +28 -22
- package/dist/react/use-subscribe-only.js.map +1 -1
- package/dist/react/use-teardown.cjs +20 -19
- package/dist/react/use-teardown.cjs.map +1 -1
- package/dist/react/use-teardown.js +20 -19
- package/dist/react/use-teardown.js.map +1 -1
- package/dist/react-native/NativeCollection.cjs +98 -78
- package/dist/react-native/NativeCollection.cjs.map +1 -1
- package/dist/react-native/NativeCollection.js +97 -77
- package/dist/react-native/NativeCollection.js.map +1 -1
- package/dist/react-native.cjs +2 -4
- package/dist/react-native.js +1 -4
- package/dist/react.cjs +24 -26
- package/dist/react.js +1 -17
- package/dist/singleton.cjs +28 -22
- package/dist/singleton.cjs.map +1 -1
- package/dist/singleton.js +29 -26
- package/dist/singleton.js.map +1 -1
- package/dist/walkPrototypeChain.cjs +20 -12
- package/dist/walkPrototypeChain.cjs.map +1 -1
- package/dist/walkPrototypeChain.js +21 -13
- package/dist/walkPrototypeChain.js.map +1 -1
- package/dist/web/IndexedDBCollection.cjs +53 -36
- package/dist/web/IndexedDBCollection.cjs.map +1 -1
- package/dist/web/IndexedDBCollection.js +52 -35
- package/dist/web/IndexedDBCollection.js.map +1 -1
- package/dist/web/WebStorageCollection.cjs +82 -84
- package/dist/web/WebStorageCollection.cjs.map +1 -1
- package/dist/web/WebStorageCollection.js +81 -83
- package/dist/web/WebStorageCollection.js.map +1 -1
- package/dist/web/idb.cjs +107 -99
- package/dist/web/idb.cjs.map +1 -1
- package/dist/web/idb.js +108 -105
- package/dist/web/idb.js.map +1 -1
- package/dist/web.cjs +4 -6
- package/dist/web.js +1 -5
- package/dist/wrapAsyncMethods.cjs +141 -168
- package/dist/wrapAsyncMethods.cjs.map +1 -1
- package/dist/wrapAsyncMethods.js +141 -168
- package/dist/wrapAsyncMethods.js.map +1 -1
- package/package.json +8 -8
- package/src/Pending.test.ts +1 -2
- package/src/Sorting.test.ts +1 -1
- package/src/produceDraft.test.ts +3 -3
- package/src/react/components/CardList.test.tsx +1 -1
- package/src/react/components/DataTable.test.tsx +1 -1
- package/src/react/components/InfiniteScroll.test.tsx +5 -5
- package/dist/mvc-kit.cjs.map +0 -1
- package/dist/mvc-kit.js.map +0 -1
- package/dist/react-native.cjs.map +0 -1
- package/dist/react-native.js.map +0 -1
- package/dist/react.cjs.map +0 -1
- package/dist/react.js.map +0 -1
- package/dist/web.cjs.map +0 -1
- package/dist/web.js.map +0 -1
package/dist/Sorting.cjs
CHANGED
|
@@ -1,104 +1,105 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
}
|
|
1
|
+
const require_Trackable = require("./Trackable.cjs");
|
|
2
|
+
//#region src/Sorting.ts
|
|
3
|
+
/**
|
|
4
|
+
* Multi-column sort state manager with a comparator pipeline.
|
|
5
|
+
* Maintains an ordered list of sort descriptors and applies them to arrays.
|
|
6
|
+
* Subscribable — auto-tracked when used as a ViewModel property.
|
|
7
|
+
*/
|
|
8
|
+
var Sorting = class extends require_Trackable.Trackable {
|
|
9
|
+
_sorts;
|
|
10
|
+
constructor(options) {
|
|
11
|
+
super();
|
|
12
|
+
this._sorts = Object.freeze(options?.sorts?.map((s) => ({ ...s })) ?? []);
|
|
13
|
+
}
|
|
14
|
+
/** Current list of active sort descriptors, in priority order. */
|
|
15
|
+
get sorts() {
|
|
16
|
+
return this._sorts;
|
|
17
|
+
}
|
|
18
|
+
/** Primary sort key (first descriptor), or null when empty. */
|
|
19
|
+
get key() {
|
|
20
|
+
return this._sorts.length > 0 ? this._sorts[0].key : null;
|
|
21
|
+
}
|
|
22
|
+
/** Primary sort direction. Defaults to 'asc' when empty. */
|
|
23
|
+
get direction() {
|
|
24
|
+
return this._sorts.length > 0 ? this._sorts[0].direction : "asc";
|
|
25
|
+
}
|
|
26
|
+
/** Whether the given key is currently sorted. */
|
|
27
|
+
isSorted(key) {
|
|
28
|
+
return this._sorts.some((s) => s.key === key);
|
|
29
|
+
}
|
|
30
|
+
/** Returns the sort direction for a key, or null if not sorted. */
|
|
31
|
+
directionOf(key) {
|
|
32
|
+
const found = this._sorts.find((s) => s.key === key);
|
|
33
|
+
return found ? found.direction : null;
|
|
34
|
+
}
|
|
35
|
+
/** Returns the priority index of a sorted key, or -1 if not sorted. */
|
|
36
|
+
indexOf(key) {
|
|
37
|
+
return this._sorts.findIndex((s) => s.key === key);
|
|
38
|
+
}
|
|
39
|
+
/** 3-click cycle: not sorted → asc → desc → removed. */
|
|
40
|
+
toggle(key) {
|
|
41
|
+
const idx = this.indexOf(key);
|
|
42
|
+
if (idx === -1) this._sorts = Object.freeze([...this._sorts, {
|
|
43
|
+
key,
|
|
44
|
+
direction: "asc"
|
|
45
|
+
}]);
|
|
46
|
+
else if (this._sorts[idx].direction === "asc") {
|
|
47
|
+
const next = this._sorts.map((s, i) => i === idx ? {
|
|
48
|
+
key: s.key,
|
|
49
|
+
direction: "desc"
|
|
50
|
+
} : s);
|
|
51
|
+
this._sorts = Object.freeze(next);
|
|
52
|
+
} else this._sorts = Object.freeze(this._sorts.filter((_, i) => i !== idx));
|
|
53
|
+
this.notify();
|
|
54
|
+
}
|
|
55
|
+
/** Replace all with a single sort. */
|
|
56
|
+
setSort(key, direction) {
|
|
57
|
+
this._sorts = Object.freeze([{
|
|
58
|
+
key,
|
|
59
|
+
direction
|
|
60
|
+
}]);
|
|
61
|
+
this.notify();
|
|
62
|
+
}
|
|
63
|
+
/** Replace all sorts. */
|
|
64
|
+
setSorts(sorts) {
|
|
65
|
+
this._sorts = Object.freeze(sorts.map((s) => ({ ...s })));
|
|
66
|
+
this.notify();
|
|
67
|
+
}
|
|
68
|
+
/** Clear all sort descriptors. */
|
|
69
|
+
reset() {
|
|
70
|
+
this._sorts = Object.freeze([]);
|
|
71
|
+
this.notify();
|
|
72
|
+
}
|
|
73
|
+
/** Sort an array using the current descriptors. Returns a new sorted array. */
|
|
74
|
+
apply(items, compareFn) {
|
|
75
|
+
if (this._sorts.length === 0) return items;
|
|
76
|
+
const sorted = items.slice();
|
|
77
|
+
const sorts = this._sorts;
|
|
78
|
+
sorted.sort((a, b) => {
|
|
79
|
+
for (const { key, direction } of sorts) {
|
|
80
|
+
let cmp;
|
|
81
|
+
if (compareFn) cmp = compareFn(a, b, key, direction);
|
|
82
|
+
else cmp = defaultCompare(a, b, key);
|
|
83
|
+
if (direction === "desc") cmp = -cmp;
|
|
84
|
+
if (cmp !== 0) return cmp;
|
|
85
|
+
}
|
|
86
|
+
return 0;
|
|
87
|
+
});
|
|
88
|
+
return sorted;
|
|
89
|
+
}
|
|
90
|
+
};
|
|
90
91
|
function defaultCompare(a, b, key) {
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
if (aVal > bVal) return 1;
|
|
101
|
-
return 0;
|
|
92
|
+
const aVal = a[key];
|
|
93
|
+
const bVal = b[key];
|
|
94
|
+
if (aVal == null && bVal == null) return 0;
|
|
95
|
+
if (aVal == null) return -1;
|
|
96
|
+
if (bVal == null) return 1;
|
|
97
|
+
if (typeof aVal === "string" && typeof bVal === "string") return aVal.localeCompare(bVal);
|
|
98
|
+
if (aVal < bVal) return -1;
|
|
99
|
+
if (aVal > bVal) return 1;
|
|
100
|
+
return 0;
|
|
102
101
|
}
|
|
102
|
+
//#endregion
|
|
103
103
|
exports.Sorting = Sorting;
|
|
104
|
-
|
|
104
|
+
|
|
105
|
+
//# sourceMappingURL=Sorting.cjs.map
|
package/dist/Sorting.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Sorting.cjs","sources":["../src/Sorting.ts"],"sourcesContent":["import { Trackable } from './Trackable';\n\n/** Describes a single sort column with key and direction. */\nexport interface SortDescriptor {\n key: string;\n direction: 'asc' | 'desc';\n}\n\n/**\n * Multi-column sort state manager with a comparator pipeline.\n * Maintains an ordered list of sort descriptors and applies them to arrays.\n * Subscribable — auto-tracked when used as a ViewModel property.\n */\nexport class Sorting<T = any> extends Trackable {\n private _sorts: readonly SortDescriptor[];\n\n constructor(options?: { sorts?: SortDescriptor[] }) {\n super();\n this._sorts = Object.freeze(options?.sorts?.map(s => ({ ...s })) ?? []);\n }\n\n // ── Readable state ──\n\n /** Current list of active sort descriptors, in priority order. */\n get sorts(): readonly SortDescriptor[] {\n return this._sorts;\n }\n\n /** Primary sort key (first descriptor), or null when empty. */\n get key(): string | null {\n return this._sorts.length > 0 ? this._sorts[0].key : null;\n }\n\n /** Primary sort direction. Defaults to 'asc' when empty. */\n get direction(): 'asc' | 'desc' {\n return this._sorts.length > 0 ? this._sorts[0].direction : 'asc';\n }\n\n // ── Query ──\n\n /** Whether the given key is currently sorted. */\n isSorted(key: string): boolean {\n return this._sorts.some(s => s.key === key);\n }\n\n /** Returns the sort direction for a key, or null if not sorted. */\n directionOf(key: string): 'asc' | 'desc' | null {\n const found = this._sorts.find(s => s.key === key);\n return found ? found.direction : null;\n }\n\n /** Returns the priority index of a sorted key, or -1 if not sorted. */\n indexOf(key: string): number {\n return this._sorts.findIndex(s => s.key === key);\n }\n\n // ── Actions ──\n\n /** 3-click cycle: not sorted → asc → desc → removed. */\n toggle(key: string): void {\n const idx = this.indexOf(key);\n if (idx === -1) {\n // Add as asc\n this._sorts = Object.freeze([...this._sorts, { key, direction: 'asc' as const }]);\n } else if (this._sorts[idx].direction === 'asc') {\n // Flip to desc\n const next = this._sorts.map((s, i) =>\n i === idx ? { key: s.key, direction: 'desc' as const } : s\n );\n this._sorts = Object.freeze(next);\n } else {\n // Remove\n this._sorts = Object.freeze(this._sorts.filter((_, i) => i !== idx));\n }\n this.notify();\n }\n\n /** Replace all with a single sort. */\n setSort(key: string, direction: 'asc' | 'desc'): void {\n this._sorts = Object.freeze([{ key, direction }]);\n this.notify();\n }\n\n /** Replace all sorts. */\n setSorts(sorts: SortDescriptor[]): void {\n this._sorts = Object.freeze(sorts.map(s => ({ ...s })));\n this.notify();\n }\n\n /** Clear all sort descriptors. */\n reset(): void {\n this._sorts = Object.freeze([]);\n this.notify();\n }\n\n // ── Pipeline ──\n\n /** Sort an array using the current descriptors. Returns a new sorted array. */\n apply(\n items: T[],\n compareFn?: (a: T, b: T, key: string, dir: 'asc' | 'desc') => number,\n ): T[] {\n if (this._sorts.length === 0) return items;\n const sorted = items.slice();\n const sorts = this._sorts;\n sorted.sort((a, b) => {\n for (const { key, direction } of sorts) {\n let cmp: number;\n if (compareFn) {\n cmp = compareFn(a, b, key, direction);\n } else {\n cmp = defaultCompare(a, b, key);\n }\n if (direction === 'desc') cmp = -cmp;\n if (cmp !== 0) return cmp;\n }\n return 0;\n });\n return sorted;\n }\n}\n\nfunction defaultCompare(a: any, b: any, key: string): number {\n const aVal = a[key];\n const bVal = b[key];\n if (aVal == null && bVal == null) return 0;\n if (aVal == null) return -1;\n if (bVal == null) return 1;\n if (typeof aVal === 'string' && typeof bVal === 'string') {\n return aVal.localeCompare(bVal);\n }\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n}\n"],"
|
|
1
|
+
{"version":3,"file":"Sorting.cjs","names":[],"sources":["../src/Sorting.ts"],"sourcesContent":["import { Trackable } from './Trackable';\n\n/** Describes a single sort column with key and direction. */\nexport interface SortDescriptor {\n key: string;\n direction: 'asc' | 'desc';\n}\n\n/**\n * Multi-column sort state manager with a comparator pipeline.\n * Maintains an ordered list of sort descriptors and applies them to arrays.\n * Subscribable — auto-tracked when used as a ViewModel property.\n */\nexport class Sorting<T = any> extends Trackable {\n private _sorts: readonly SortDescriptor[];\n\n constructor(options?: { sorts?: SortDescriptor[] }) {\n super();\n this._sorts = Object.freeze(options?.sorts?.map(s => ({ ...s })) ?? []);\n }\n\n // ── Readable state ──\n\n /** Current list of active sort descriptors, in priority order. */\n get sorts(): readonly SortDescriptor[] {\n return this._sorts;\n }\n\n /** Primary sort key (first descriptor), or null when empty. */\n get key(): string | null {\n return this._sorts.length > 0 ? this._sorts[0].key : null;\n }\n\n /** Primary sort direction. Defaults to 'asc' when empty. */\n get direction(): 'asc' | 'desc' {\n return this._sorts.length > 0 ? this._sorts[0].direction : 'asc';\n }\n\n // ── Query ──\n\n /** Whether the given key is currently sorted. */\n isSorted(key: string): boolean {\n return this._sorts.some(s => s.key === key);\n }\n\n /** Returns the sort direction for a key, or null if not sorted. */\n directionOf(key: string): 'asc' | 'desc' | null {\n const found = this._sorts.find(s => s.key === key);\n return found ? found.direction : null;\n }\n\n /** Returns the priority index of a sorted key, or -1 if not sorted. */\n indexOf(key: string): number {\n return this._sorts.findIndex(s => s.key === key);\n }\n\n // ── Actions ──\n\n /** 3-click cycle: not sorted → asc → desc → removed. */\n toggle(key: string): void {\n const idx = this.indexOf(key);\n if (idx === -1) {\n // Add as asc\n this._sorts = Object.freeze([...this._sorts, { key, direction: 'asc' as const }]);\n } else if (this._sorts[idx].direction === 'asc') {\n // Flip to desc\n const next = this._sorts.map((s, i) =>\n i === idx ? { key: s.key, direction: 'desc' as const } : s\n );\n this._sorts = Object.freeze(next);\n } else {\n // Remove\n this._sorts = Object.freeze(this._sorts.filter((_, i) => i !== idx));\n }\n this.notify();\n }\n\n /** Replace all with a single sort. */\n setSort(key: string, direction: 'asc' | 'desc'): void {\n this._sorts = Object.freeze([{ key, direction }]);\n this.notify();\n }\n\n /** Replace all sorts. */\n setSorts(sorts: SortDescriptor[]): void {\n this._sorts = Object.freeze(sorts.map(s => ({ ...s })));\n this.notify();\n }\n\n /** Clear all sort descriptors. */\n reset(): void {\n this._sorts = Object.freeze([]);\n this.notify();\n }\n\n // ── Pipeline ──\n\n /** Sort an array using the current descriptors. Returns a new sorted array. */\n apply(\n items: T[],\n compareFn?: (a: T, b: T, key: string, dir: 'asc' | 'desc') => number,\n ): T[] {\n if (this._sorts.length === 0) return items;\n const sorted = items.slice();\n const sorts = this._sorts;\n sorted.sort((a, b) => {\n for (const { key, direction } of sorts) {\n let cmp: number;\n if (compareFn) {\n cmp = compareFn(a, b, key, direction);\n } else {\n cmp = defaultCompare(a, b, key);\n }\n if (direction === 'desc') cmp = -cmp;\n if (cmp !== 0) return cmp;\n }\n return 0;\n });\n return sorted;\n }\n}\n\nfunction defaultCompare(a: any, b: any, key: string): number {\n const aVal = a[key];\n const bVal = b[key];\n if (aVal == null && bVal == null) return 0;\n if (aVal == null) return -1;\n if (bVal == null) return 1;\n if (typeof aVal === 'string' && typeof bVal === 'string') {\n return aVal.localeCompare(bVal);\n }\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n}\n"],"mappings":";;;;;;;AAaA,IAAa,UAAb,cAAsC,kBAAA,UAAU;CAC9C;CAEA,YAAY,SAAwC;AAClD,SAAO;AACP,OAAK,SAAS,OAAO,OAAO,SAAS,OAAO,KAAI,OAAM,EAAE,GAAG,GAAG,EAAE,IAAI,EAAE,CAAC;;;CAMzE,IAAI,QAAmC;AACrC,SAAO,KAAK;;;CAId,IAAI,MAAqB;AACvB,SAAO,KAAK,OAAO,SAAS,IAAI,KAAK,OAAO,GAAG,MAAM;;;CAIvD,IAAI,YAA4B;AAC9B,SAAO,KAAK,OAAO,SAAS,IAAI,KAAK,OAAO,GAAG,YAAY;;;CAM7D,SAAS,KAAsB;AAC7B,SAAO,KAAK,OAAO,MAAK,MAAK,EAAE,QAAQ,IAAI;;;CAI7C,YAAY,KAAoC;EAC9C,MAAM,QAAQ,KAAK,OAAO,MAAK,MAAK,EAAE,QAAQ,IAAI;AAClD,SAAO,QAAQ,MAAM,YAAY;;;CAInC,QAAQ,KAAqB;AAC3B,SAAO,KAAK,OAAO,WAAU,MAAK,EAAE,QAAQ,IAAI;;;CAMlD,OAAO,KAAmB;EACxB,MAAM,MAAM,KAAK,QAAQ,IAAI;AAC7B,MAAI,QAAQ,GAEV,MAAK,SAAS,OAAO,OAAO,CAAC,GAAG,KAAK,QAAQ;GAAE;GAAK,WAAW;GAAgB,CAAC,CAAC;WACxE,KAAK,OAAO,KAAK,cAAc,OAAO;GAE/C,MAAM,OAAO,KAAK,OAAO,KAAK,GAAG,MAC/B,MAAM,MAAM;IAAE,KAAK,EAAE;IAAK,WAAW;IAAiB,GAAG,EAC1D;AACD,QAAK,SAAS,OAAO,OAAO,KAAK;QAGjC,MAAK,SAAS,OAAO,OAAO,KAAK,OAAO,QAAQ,GAAG,MAAM,MAAM,IAAI,CAAC;AAEtE,OAAK,QAAQ;;;CAIf,QAAQ,KAAa,WAAiC;AACpD,OAAK,SAAS,OAAO,OAAO,CAAC;GAAE;GAAK;GAAW,CAAC,CAAC;AACjD,OAAK,QAAQ;;;CAIf,SAAS,OAA+B;AACtC,OAAK,SAAS,OAAO,OAAO,MAAM,KAAI,OAAM,EAAE,GAAG,GAAG,EAAE,CAAC;AACvD,OAAK,QAAQ;;;CAIf,QAAc;AACZ,OAAK,SAAS,OAAO,OAAO,EAAE,CAAC;AAC/B,OAAK,QAAQ;;;CAMf,MACE,OACA,WACK;AACL,MAAI,KAAK,OAAO,WAAW,EAAG,QAAO;EACrC,MAAM,SAAS,MAAM,OAAO;EAC5B,MAAM,QAAQ,KAAK;AACnB,SAAO,MAAM,GAAG,MAAM;AACpB,QAAK,MAAM,EAAE,KAAK,eAAe,OAAO;IACtC,IAAI;AACJ,QAAI,UACF,OAAM,UAAU,GAAG,GAAG,KAAK,UAAU;QAErC,OAAM,eAAe,GAAG,GAAG,IAAI;AAEjC,QAAI,cAAc,OAAQ,OAAM,CAAC;AACjC,QAAI,QAAQ,EAAG,QAAO;;AAExB,UAAO;IACP;AACF,SAAO;;;AAIX,SAAS,eAAe,GAAQ,GAAQ,KAAqB;CAC3D,MAAM,OAAO,EAAE;CACf,MAAM,OAAO,EAAE;AACf,KAAI,QAAQ,QAAQ,QAAQ,KAAM,QAAO;AACzC,KAAI,QAAQ,KAAM,QAAO;AACzB,KAAI,QAAQ,KAAM,QAAO;AACzB,KAAI,OAAO,SAAS,YAAY,OAAO,SAAS,SAC9C,QAAO,KAAK,cAAc,KAAK;AAEjC,KAAI,OAAO,KAAM,QAAO;AACxB,KAAI,OAAO,KAAM,QAAO;AACxB,QAAO"}
|
package/dist/Sorting.js
CHANGED
|
@@ -1,104 +1,105 @@
|
|
|
1
1
|
import { Trackable } from "./Trackable.js";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
}
|
|
2
|
+
//#region src/Sorting.ts
|
|
3
|
+
/**
|
|
4
|
+
* Multi-column sort state manager with a comparator pipeline.
|
|
5
|
+
* Maintains an ordered list of sort descriptors and applies them to arrays.
|
|
6
|
+
* Subscribable — auto-tracked when used as a ViewModel property.
|
|
7
|
+
*/
|
|
8
|
+
var Sorting = class extends Trackable {
|
|
9
|
+
_sorts;
|
|
10
|
+
constructor(options) {
|
|
11
|
+
super();
|
|
12
|
+
this._sorts = Object.freeze(options?.sorts?.map((s) => ({ ...s })) ?? []);
|
|
13
|
+
}
|
|
14
|
+
/** Current list of active sort descriptors, in priority order. */
|
|
15
|
+
get sorts() {
|
|
16
|
+
return this._sorts;
|
|
17
|
+
}
|
|
18
|
+
/** Primary sort key (first descriptor), or null when empty. */
|
|
19
|
+
get key() {
|
|
20
|
+
return this._sorts.length > 0 ? this._sorts[0].key : null;
|
|
21
|
+
}
|
|
22
|
+
/** Primary sort direction. Defaults to 'asc' when empty. */
|
|
23
|
+
get direction() {
|
|
24
|
+
return this._sorts.length > 0 ? this._sorts[0].direction : "asc";
|
|
25
|
+
}
|
|
26
|
+
/** Whether the given key is currently sorted. */
|
|
27
|
+
isSorted(key) {
|
|
28
|
+
return this._sorts.some((s) => s.key === key);
|
|
29
|
+
}
|
|
30
|
+
/** Returns the sort direction for a key, or null if not sorted. */
|
|
31
|
+
directionOf(key) {
|
|
32
|
+
const found = this._sorts.find((s) => s.key === key);
|
|
33
|
+
return found ? found.direction : null;
|
|
34
|
+
}
|
|
35
|
+
/** Returns the priority index of a sorted key, or -1 if not sorted. */
|
|
36
|
+
indexOf(key) {
|
|
37
|
+
return this._sorts.findIndex((s) => s.key === key);
|
|
38
|
+
}
|
|
39
|
+
/** 3-click cycle: not sorted → asc → desc → removed. */
|
|
40
|
+
toggle(key) {
|
|
41
|
+
const idx = this.indexOf(key);
|
|
42
|
+
if (idx === -1) this._sorts = Object.freeze([...this._sorts, {
|
|
43
|
+
key,
|
|
44
|
+
direction: "asc"
|
|
45
|
+
}]);
|
|
46
|
+
else if (this._sorts[idx].direction === "asc") {
|
|
47
|
+
const next = this._sorts.map((s, i) => i === idx ? {
|
|
48
|
+
key: s.key,
|
|
49
|
+
direction: "desc"
|
|
50
|
+
} : s);
|
|
51
|
+
this._sorts = Object.freeze(next);
|
|
52
|
+
} else this._sorts = Object.freeze(this._sorts.filter((_, i) => i !== idx));
|
|
53
|
+
this.notify();
|
|
54
|
+
}
|
|
55
|
+
/** Replace all with a single sort. */
|
|
56
|
+
setSort(key, direction) {
|
|
57
|
+
this._sorts = Object.freeze([{
|
|
58
|
+
key,
|
|
59
|
+
direction
|
|
60
|
+
}]);
|
|
61
|
+
this.notify();
|
|
62
|
+
}
|
|
63
|
+
/** Replace all sorts. */
|
|
64
|
+
setSorts(sorts) {
|
|
65
|
+
this._sorts = Object.freeze(sorts.map((s) => ({ ...s })));
|
|
66
|
+
this.notify();
|
|
67
|
+
}
|
|
68
|
+
/** Clear all sort descriptors. */
|
|
69
|
+
reset() {
|
|
70
|
+
this._sorts = Object.freeze([]);
|
|
71
|
+
this.notify();
|
|
72
|
+
}
|
|
73
|
+
/** Sort an array using the current descriptors. Returns a new sorted array. */
|
|
74
|
+
apply(items, compareFn) {
|
|
75
|
+
if (this._sorts.length === 0) return items;
|
|
76
|
+
const sorted = items.slice();
|
|
77
|
+
const sorts = this._sorts;
|
|
78
|
+
sorted.sort((a, b) => {
|
|
79
|
+
for (const { key, direction } of sorts) {
|
|
80
|
+
let cmp;
|
|
81
|
+
if (compareFn) cmp = compareFn(a, b, key, direction);
|
|
82
|
+
else cmp = defaultCompare(a, b, key);
|
|
83
|
+
if (direction === "desc") cmp = -cmp;
|
|
84
|
+
if (cmp !== 0) return cmp;
|
|
85
|
+
}
|
|
86
|
+
return 0;
|
|
87
|
+
});
|
|
88
|
+
return sorted;
|
|
89
|
+
}
|
|
90
|
+
};
|
|
88
91
|
function defaultCompare(a, b, key) {
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
if (aVal > bVal) return 1;
|
|
99
|
-
return 0;
|
|
92
|
+
const aVal = a[key];
|
|
93
|
+
const bVal = b[key];
|
|
94
|
+
if (aVal == null && bVal == null) return 0;
|
|
95
|
+
if (aVal == null) return -1;
|
|
96
|
+
if (bVal == null) return 1;
|
|
97
|
+
if (typeof aVal === "string" && typeof bVal === "string") return aVal.localeCompare(bVal);
|
|
98
|
+
if (aVal < bVal) return -1;
|
|
99
|
+
if (aVal > bVal) return 1;
|
|
100
|
+
return 0;
|
|
100
101
|
}
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
//# sourceMappingURL=Sorting.js.map
|
|
102
|
+
//#endregion
|
|
103
|
+
export { Sorting };
|
|
104
|
+
|
|
105
|
+
//# sourceMappingURL=Sorting.js.map
|
package/dist/Sorting.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Sorting.js","sources":["../src/Sorting.ts"],"sourcesContent":["import { Trackable } from './Trackable';\n\n/** Describes a single sort column with key and direction. */\nexport interface SortDescriptor {\n key: string;\n direction: 'asc' | 'desc';\n}\n\n/**\n * Multi-column sort state manager with a comparator pipeline.\n * Maintains an ordered list of sort descriptors and applies them to arrays.\n * Subscribable — auto-tracked when used as a ViewModel property.\n */\nexport class Sorting<T = any> extends Trackable {\n private _sorts: readonly SortDescriptor[];\n\n constructor(options?: { sorts?: SortDescriptor[] }) {\n super();\n this._sorts = Object.freeze(options?.sorts?.map(s => ({ ...s })) ?? []);\n }\n\n // ── Readable state ──\n\n /** Current list of active sort descriptors, in priority order. */\n get sorts(): readonly SortDescriptor[] {\n return this._sorts;\n }\n\n /** Primary sort key (first descriptor), or null when empty. */\n get key(): string | null {\n return this._sorts.length > 0 ? this._sorts[0].key : null;\n }\n\n /** Primary sort direction. Defaults to 'asc' when empty. */\n get direction(): 'asc' | 'desc' {\n return this._sorts.length > 0 ? this._sorts[0].direction : 'asc';\n }\n\n // ── Query ──\n\n /** Whether the given key is currently sorted. */\n isSorted(key: string): boolean {\n return this._sorts.some(s => s.key === key);\n }\n\n /** Returns the sort direction for a key, or null if not sorted. */\n directionOf(key: string): 'asc' | 'desc' | null {\n const found = this._sorts.find(s => s.key === key);\n return found ? found.direction : null;\n }\n\n /** Returns the priority index of a sorted key, or -1 if not sorted. */\n indexOf(key: string): number {\n return this._sorts.findIndex(s => s.key === key);\n }\n\n // ── Actions ──\n\n /** 3-click cycle: not sorted → asc → desc → removed. */\n toggle(key: string): void {\n const idx = this.indexOf(key);\n if (idx === -1) {\n // Add as asc\n this._sorts = Object.freeze([...this._sorts, { key, direction: 'asc' as const }]);\n } else if (this._sorts[idx].direction === 'asc') {\n // Flip to desc\n const next = this._sorts.map((s, i) =>\n i === idx ? { key: s.key, direction: 'desc' as const } : s\n );\n this._sorts = Object.freeze(next);\n } else {\n // Remove\n this._sorts = Object.freeze(this._sorts.filter((_, i) => i !== idx));\n }\n this.notify();\n }\n\n /** Replace all with a single sort. */\n setSort(key: string, direction: 'asc' | 'desc'): void {\n this._sorts = Object.freeze([{ key, direction }]);\n this.notify();\n }\n\n /** Replace all sorts. */\n setSorts(sorts: SortDescriptor[]): void {\n this._sorts = Object.freeze(sorts.map(s => ({ ...s })));\n this.notify();\n }\n\n /** Clear all sort descriptors. */\n reset(): void {\n this._sorts = Object.freeze([]);\n this.notify();\n }\n\n // ── Pipeline ──\n\n /** Sort an array using the current descriptors. Returns a new sorted array. */\n apply(\n items: T[],\n compareFn?: (a: T, b: T, key: string, dir: 'asc' | 'desc') => number,\n ): T[] {\n if (this._sorts.length === 0) return items;\n const sorted = items.slice();\n const sorts = this._sorts;\n sorted.sort((a, b) => {\n for (const { key, direction } of sorts) {\n let cmp: number;\n if (compareFn) {\n cmp = compareFn(a, b, key, direction);\n } else {\n cmp = defaultCompare(a, b, key);\n }\n if (direction === 'desc') cmp = -cmp;\n if (cmp !== 0) return cmp;\n }\n return 0;\n });\n return sorted;\n }\n}\n\nfunction defaultCompare(a: any, b: any, key: string): number {\n const aVal = a[key];\n const bVal = b[key];\n if (aVal == null && bVal == null) return 0;\n if (aVal == null) return -1;\n if (bVal == null) return 1;\n if (typeof aVal === 'string' && typeof bVal === 'string') {\n return aVal.localeCompare(bVal);\n }\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n}\n"],"
|
|
1
|
+
{"version":3,"file":"Sorting.js","names":[],"sources":["../src/Sorting.ts"],"sourcesContent":["import { Trackable } from './Trackable';\n\n/** Describes a single sort column with key and direction. */\nexport interface SortDescriptor {\n key: string;\n direction: 'asc' | 'desc';\n}\n\n/**\n * Multi-column sort state manager with a comparator pipeline.\n * Maintains an ordered list of sort descriptors and applies them to arrays.\n * Subscribable — auto-tracked when used as a ViewModel property.\n */\nexport class Sorting<T = any> extends Trackable {\n private _sorts: readonly SortDescriptor[];\n\n constructor(options?: { sorts?: SortDescriptor[] }) {\n super();\n this._sorts = Object.freeze(options?.sorts?.map(s => ({ ...s })) ?? []);\n }\n\n // ── Readable state ──\n\n /** Current list of active sort descriptors, in priority order. */\n get sorts(): readonly SortDescriptor[] {\n return this._sorts;\n }\n\n /** Primary sort key (first descriptor), or null when empty. */\n get key(): string | null {\n return this._sorts.length > 0 ? this._sorts[0].key : null;\n }\n\n /** Primary sort direction. Defaults to 'asc' when empty. */\n get direction(): 'asc' | 'desc' {\n return this._sorts.length > 0 ? this._sorts[0].direction : 'asc';\n }\n\n // ── Query ──\n\n /** Whether the given key is currently sorted. */\n isSorted(key: string): boolean {\n return this._sorts.some(s => s.key === key);\n }\n\n /** Returns the sort direction for a key, or null if not sorted. */\n directionOf(key: string): 'asc' | 'desc' | null {\n const found = this._sorts.find(s => s.key === key);\n return found ? found.direction : null;\n }\n\n /** Returns the priority index of a sorted key, or -1 if not sorted. */\n indexOf(key: string): number {\n return this._sorts.findIndex(s => s.key === key);\n }\n\n // ── Actions ──\n\n /** 3-click cycle: not sorted → asc → desc → removed. */\n toggle(key: string): void {\n const idx = this.indexOf(key);\n if (idx === -1) {\n // Add as asc\n this._sorts = Object.freeze([...this._sorts, { key, direction: 'asc' as const }]);\n } else if (this._sorts[idx].direction === 'asc') {\n // Flip to desc\n const next = this._sorts.map((s, i) =>\n i === idx ? { key: s.key, direction: 'desc' as const } : s\n );\n this._sorts = Object.freeze(next);\n } else {\n // Remove\n this._sorts = Object.freeze(this._sorts.filter((_, i) => i !== idx));\n }\n this.notify();\n }\n\n /** Replace all with a single sort. */\n setSort(key: string, direction: 'asc' | 'desc'): void {\n this._sorts = Object.freeze([{ key, direction }]);\n this.notify();\n }\n\n /** Replace all sorts. */\n setSorts(sorts: SortDescriptor[]): void {\n this._sorts = Object.freeze(sorts.map(s => ({ ...s })));\n this.notify();\n }\n\n /** Clear all sort descriptors. */\n reset(): void {\n this._sorts = Object.freeze([]);\n this.notify();\n }\n\n // ── Pipeline ──\n\n /** Sort an array using the current descriptors. Returns a new sorted array. */\n apply(\n items: T[],\n compareFn?: (a: T, b: T, key: string, dir: 'asc' | 'desc') => number,\n ): T[] {\n if (this._sorts.length === 0) return items;\n const sorted = items.slice();\n const sorts = this._sorts;\n sorted.sort((a, b) => {\n for (const { key, direction } of sorts) {\n let cmp: number;\n if (compareFn) {\n cmp = compareFn(a, b, key, direction);\n } else {\n cmp = defaultCompare(a, b, key);\n }\n if (direction === 'desc') cmp = -cmp;\n if (cmp !== 0) return cmp;\n }\n return 0;\n });\n return sorted;\n }\n}\n\nfunction defaultCompare(a: any, b: any, key: string): number {\n const aVal = a[key];\n const bVal = b[key];\n if (aVal == null && bVal == null) return 0;\n if (aVal == null) return -1;\n if (bVal == null) return 1;\n if (typeof aVal === 'string' && typeof bVal === 'string') {\n return aVal.localeCompare(bVal);\n }\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n}\n"],"mappings":";;;;;;;AAaA,IAAa,UAAb,cAAsC,UAAU;CAC9C;CAEA,YAAY,SAAwC;AAClD,SAAO;AACP,OAAK,SAAS,OAAO,OAAO,SAAS,OAAO,KAAI,OAAM,EAAE,GAAG,GAAG,EAAE,IAAI,EAAE,CAAC;;;CAMzE,IAAI,QAAmC;AACrC,SAAO,KAAK;;;CAId,IAAI,MAAqB;AACvB,SAAO,KAAK,OAAO,SAAS,IAAI,KAAK,OAAO,GAAG,MAAM;;;CAIvD,IAAI,YAA4B;AAC9B,SAAO,KAAK,OAAO,SAAS,IAAI,KAAK,OAAO,GAAG,YAAY;;;CAM7D,SAAS,KAAsB;AAC7B,SAAO,KAAK,OAAO,MAAK,MAAK,EAAE,QAAQ,IAAI;;;CAI7C,YAAY,KAAoC;EAC9C,MAAM,QAAQ,KAAK,OAAO,MAAK,MAAK,EAAE,QAAQ,IAAI;AAClD,SAAO,QAAQ,MAAM,YAAY;;;CAInC,QAAQ,KAAqB;AAC3B,SAAO,KAAK,OAAO,WAAU,MAAK,EAAE,QAAQ,IAAI;;;CAMlD,OAAO,KAAmB;EACxB,MAAM,MAAM,KAAK,QAAQ,IAAI;AAC7B,MAAI,QAAQ,GAEV,MAAK,SAAS,OAAO,OAAO,CAAC,GAAG,KAAK,QAAQ;GAAE;GAAK,WAAW;GAAgB,CAAC,CAAC;WACxE,KAAK,OAAO,KAAK,cAAc,OAAO;GAE/C,MAAM,OAAO,KAAK,OAAO,KAAK,GAAG,MAC/B,MAAM,MAAM;IAAE,KAAK,EAAE;IAAK,WAAW;IAAiB,GAAG,EAC1D;AACD,QAAK,SAAS,OAAO,OAAO,KAAK;QAGjC,MAAK,SAAS,OAAO,OAAO,KAAK,OAAO,QAAQ,GAAG,MAAM,MAAM,IAAI,CAAC;AAEtE,OAAK,QAAQ;;;CAIf,QAAQ,KAAa,WAAiC;AACpD,OAAK,SAAS,OAAO,OAAO,CAAC;GAAE;GAAK;GAAW,CAAC,CAAC;AACjD,OAAK,QAAQ;;;CAIf,SAAS,OAA+B;AACtC,OAAK,SAAS,OAAO,OAAO,MAAM,KAAI,OAAM,EAAE,GAAG,GAAG,EAAE,CAAC;AACvD,OAAK,QAAQ;;;CAIf,QAAc;AACZ,OAAK,SAAS,OAAO,OAAO,EAAE,CAAC;AAC/B,OAAK,QAAQ;;;CAMf,MACE,OACA,WACK;AACL,MAAI,KAAK,OAAO,WAAW,EAAG,QAAO;EACrC,MAAM,SAAS,MAAM,OAAO;EAC5B,MAAM,QAAQ,KAAK;AACnB,SAAO,MAAM,GAAG,MAAM;AACpB,QAAK,MAAM,EAAE,KAAK,eAAe,OAAO;IACtC,IAAI;AACJ,QAAI,UACF,OAAM,UAAU,GAAG,GAAG,KAAK,UAAU;QAErC,OAAM,eAAe,GAAG,GAAG,IAAI;AAEjC,QAAI,cAAc,OAAQ,OAAM,CAAC;AACjC,QAAI,QAAQ,EAAG,QAAO;;AAExB,UAAO;IACP;AACF,SAAO;;;AAIX,SAAS,eAAe,GAAQ,GAAQ,KAAqB;CAC3D,MAAM,OAAO,EAAE;CACf,MAAM,OAAO,EAAE;AACf,KAAI,QAAQ,QAAQ,QAAQ,KAAM,QAAO;AACzC,KAAI,QAAQ,KAAM,QAAO;AACzB,KAAI,QAAQ,KAAM,QAAO;AACzB,KAAI,OAAO,SAAS,YAAY,OAAO,SAAS,SAC9C,QAAO,KAAK,cAAc,KAAK;AAEjC,KAAI,OAAO,KAAM,QAAO;AACxB,KAAI,OAAO,KAAM,QAAO;AACxB,QAAO"}
|