@plumile/router 0.1.12 → 0.1.14
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 +426 -13
- package/lib/esm/index.d.ts +4 -0
- package/lib/esm/index.d.ts.map +1 -1
- package/lib/esm/index.js +5 -1
- package/lib/esm/routing/createRouter.d.ts +1 -0
- package/lib/esm/routing/createRouter.d.ts.map +1 -1
- package/lib/esm/routing/createRouter.js +447 -360
- package/lib/esm/routing/devtools.d.ts +21 -0
- package/lib/esm/routing/devtools.d.ts.map +1 -0
- package/lib/esm/routing/devtools.js +690 -0
- package/lib/esm/routing/filters.d.ts +97 -0
- package/lib/esm/routing/filters.d.ts.map +1 -0
- package/lib/esm/routing/filters.js +557 -0
- package/lib/esm/routing/index.d.ts +6 -0
- package/lib/esm/routing/index.d.ts.map +1 -1
- package/lib/esm/routing/index.js +7 -1
- package/lib/esm/routing/useActiveFilters.d.ts +9 -0
- package/lib/esm/routing/useActiveFilters.d.ts.map +1 -0
- package/lib/esm/routing/useActiveFilters.js +38 -0
- package/lib/esm/routing/useFilterState.d.ts +10 -0
- package/lib/esm/routing/useFilterState.d.ts.map +1 -0
- package/lib/esm/routing/useFilterState.js +14 -0
- package/lib/esm/routing/useNavigate.d.ts +7 -0
- package/lib/esm/routing/useNavigate.d.ts.map +1 -1
- package/lib/esm/routing/useNavigate.js +1 -1
- package/lib/esm/routing/useNavigateWithQuery.d.ts +15 -0
- package/lib/esm/routing/useNavigateWithQuery.d.ts.map +1 -0
- package/lib/esm/routing/useNavigateWithQuery.js +95 -0
- package/lib/esm/routing/useQueryObject.d.ts +18 -0
- package/lib/esm/routing/useQueryObject.d.ts.map +1 -0
- package/lib/esm/routing/useQueryObject.js +107 -0
- package/lib/esm/routing/useStableRefEquality.d.ts +5 -0
- package/lib/esm/routing/useStableRefEquality.d.ts.map +1 -0
- package/lib/esm/routing/useStableRefEquality.js +47 -0
- package/lib/esm/tools/buildSearch.d.ts +8 -2
- package/lib/esm/tools/buildSearch.d.ts.map +1 -1
- package/lib/esm/tools/buildSearch.js +216 -12
- package/lib/esm/types.d.ts +19 -0
- package/lib/esm/types.d.ts.map +1 -1
- package/lib/esm/types.js +1 -1
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/lib/types/index.d.ts +4 -0
- package/lib/types/index.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +1 -0
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/routing/devtools.d.ts +21 -0
- package/lib/types/routing/devtools.d.ts.map +1 -0
- package/lib/types/routing/filters.d.ts +97 -0
- package/lib/types/routing/filters.d.ts.map +1 -0
- package/lib/types/routing/index.d.ts +6 -0
- package/lib/types/routing/index.d.ts.map +1 -1
- package/lib/types/routing/useActiveFilters.d.ts +9 -0
- package/lib/types/routing/useActiveFilters.d.ts.map +1 -0
- package/lib/types/routing/useFilterState.d.ts +10 -0
- package/lib/types/routing/useFilterState.d.ts.map +1 -0
- package/lib/types/routing/useNavigate.d.ts +7 -0
- package/lib/types/routing/useNavigate.d.ts.map +1 -1
- package/lib/types/routing/useNavigateWithQuery.d.ts +15 -0
- package/lib/types/routing/useNavigateWithQuery.d.ts.map +1 -0
- package/lib/types/routing/useQueryObject.d.ts +18 -0
- package/lib/types/routing/useQueryObject.d.ts.map +1 -0
- package/lib/types/routing/useStableRefEquality.d.ts +5 -0
- package/lib/types/routing/useStableRefEquality.d.ts.map +1 -0
- package/lib/types/tools/buildSearch.d.ts +8 -2
- package/lib/types/tools/buildSearch.d.ts.map +1 -1
- package/lib/types/types.d.ts +19 -0
- package/lib/types/types.d.ts.map +1 -1
- package/package.json +3 -3
|
@@ -0,0 +1,690 @@
|
|
|
1
|
+
import { parseFilters } from './filters.js';
|
|
2
|
+
function diffQueries(prev, next) {
|
|
3
|
+
const added = [];
|
|
4
|
+
const removed = [];
|
|
5
|
+
const changed = [];
|
|
6
|
+
const unchanged = [];
|
|
7
|
+
const prevObj = prev ?? {};
|
|
8
|
+
const prevKeys = new Set(Object.keys(prevObj));
|
|
9
|
+
for (const k of Object.keys(next)) {
|
|
10
|
+
if (!prevKeys.has(k)) {
|
|
11
|
+
added.push(k);
|
|
12
|
+
}
|
|
13
|
+
else {
|
|
14
|
+
const pv = prevObj[k];
|
|
15
|
+
const nv = next[k];
|
|
16
|
+
let same = false;
|
|
17
|
+
if (Array.isArray(pv) && Array.isArray(nv) && pv.length === nv.length) {
|
|
18
|
+
same = pv.every((v, i) => {
|
|
19
|
+
return v === nv[i];
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
else {
|
|
23
|
+
same = pv === nv;
|
|
24
|
+
}
|
|
25
|
+
if (same) {
|
|
26
|
+
unchanged.push(k);
|
|
27
|
+
}
|
|
28
|
+
else {
|
|
29
|
+
changed.push(k);
|
|
30
|
+
}
|
|
31
|
+
prevKeys.delete(k);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
for (const k of prevKeys) {
|
|
35
|
+
removed.push(k);
|
|
36
|
+
}
|
|
37
|
+
return { added, removed, changed, unchanged };
|
|
38
|
+
}
|
|
39
|
+
function styleMiniButton(btn, color) {
|
|
40
|
+
const { style } = btn;
|
|
41
|
+
style.cursor = 'pointer';
|
|
42
|
+
style.background = '#fff';
|
|
43
|
+
style.border = '1px solid #ccc';
|
|
44
|
+
style.borderRadius = '4px';
|
|
45
|
+
style.fontSize = '10px';
|
|
46
|
+
style.lineHeight = '1';
|
|
47
|
+
style.padding = '2px 6px';
|
|
48
|
+
style.color = color;
|
|
49
|
+
}
|
|
50
|
+
function stylePillClose(btn) {
|
|
51
|
+
const { style } = btn;
|
|
52
|
+
style.cursor = 'pointer';
|
|
53
|
+
style.background = 'transparent';
|
|
54
|
+
style.border = 'none';
|
|
55
|
+
style.fontSize = '10px';
|
|
56
|
+
style.lineHeight = '1';
|
|
57
|
+
style.padding = '0 2px';
|
|
58
|
+
style.color = '#444';
|
|
59
|
+
}
|
|
60
|
+
function createQueryDiffPill(label, kind) {
|
|
61
|
+
const span = document.createElement('span');
|
|
62
|
+
span.textContent = label;
|
|
63
|
+
span.style.padding = '1px 4px';
|
|
64
|
+
span.style.borderRadius = '10px';
|
|
65
|
+
span.style.fontSize = '10px';
|
|
66
|
+
span.style.border = '1px solid';
|
|
67
|
+
if (kind === 'added') {
|
|
68
|
+
span.style.background = '#e6ffed';
|
|
69
|
+
span.style.borderColor = '#2cbe4e';
|
|
70
|
+
span.style.color = '#22863a';
|
|
71
|
+
}
|
|
72
|
+
else if (kind === 'removed') {
|
|
73
|
+
span.style.background = '#ffeef0';
|
|
74
|
+
span.style.borderColor = '#d73a49';
|
|
75
|
+
span.style.color = '#cb2431';
|
|
76
|
+
}
|
|
77
|
+
else if (kind === 'changed') {
|
|
78
|
+
span.style.background = '#fff5b1';
|
|
79
|
+
span.style.borderColor = '#ffd33d';
|
|
80
|
+
span.style.color = '#735c0f';
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
span.style.background = '#f1f8ff';
|
|
84
|
+
span.style.borderColor = '#79b8ff';
|
|
85
|
+
span.style.color = '#0366d6';
|
|
86
|
+
}
|
|
87
|
+
return span;
|
|
88
|
+
}
|
|
89
|
+
function mountDevtoolsPanel(params) {
|
|
90
|
+
try {
|
|
91
|
+
if (typeof window === 'undefined' || typeof document === 'undefined') {
|
|
92
|
+
return function noopRecorder() {
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
if (document.querySelector('[data-plumile-router-devtools-panel]') != null) {
|
|
96
|
+
return function alreadyMountedRecorder() {
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
const { context, shortcut, verbose } = params;
|
|
100
|
+
const host = document.createElement('div');
|
|
101
|
+
host.setAttribute('data-plumile-router-devtools-panel', '');
|
|
102
|
+
const hostStyle = host.style;
|
|
103
|
+
hostStyle.position = 'fixed';
|
|
104
|
+
hostStyle.zIndex = '2147483646';
|
|
105
|
+
hostStyle.bottom = '8px';
|
|
106
|
+
hostStyle.right = '8px';
|
|
107
|
+
hostStyle.fontFamily = [
|
|
108
|
+
'ui-monospace',
|
|
109
|
+
'SFMono-Regular',
|
|
110
|
+
'Menlo',
|
|
111
|
+
'Monaco',
|
|
112
|
+
'Consolas',
|
|
113
|
+
'"Liberation Mono"',
|
|
114
|
+
'monospace',
|
|
115
|
+
].join(', ');
|
|
116
|
+
hostStyle.fontSize = '12px';
|
|
117
|
+
hostStyle.lineHeight = '1.3';
|
|
118
|
+
hostStyle.color = '#111';
|
|
119
|
+
hostStyle.background = 'transparent';
|
|
120
|
+
hostStyle.pointerEvents = 'none';
|
|
121
|
+
const shadow = host.attachShadow({ mode: 'open' });
|
|
122
|
+
const panel = document.createElement('div');
|
|
123
|
+
const panelStyle = panel.style;
|
|
124
|
+
panelStyle.pointerEvents = 'auto';
|
|
125
|
+
panelStyle.minWidth = '260px';
|
|
126
|
+
panelStyle.maxWidth = '420px';
|
|
127
|
+
panelStyle.maxHeight = '50vh';
|
|
128
|
+
panelStyle.overflow = 'auto';
|
|
129
|
+
panelStyle.background = 'rgba(255,255,255,0.95)';
|
|
130
|
+
panelStyle.border = '1px solid #ccc';
|
|
131
|
+
panelStyle.borderRadius = '6px';
|
|
132
|
+
panelStyle.boxShadow = '0 2px 8px rgba(0,0,0,0.15)';
|
|
133
|
+
panelStyle.padding = '6px 8px 8px';
|
|
134
|
+
panelStyle.backdropFilter = 'blur(4px)';
|
|
135
|
+
panelStyle.display = 'flex';
|
|
136
|
+
panelStyle.flexDirection = 'column';
|
|
137
|
+
panelStyle.rowGap = '4px';
|
|
138
|
+
const header = document.createElement('div');
|
|
139
|
+
const headerStyle = header.style;
|
|
140
|
+
headerStyle.display = 'flex';
|
|
141
|
+
headerStyle.alignItems = 'center';
|
|
142
|
+
headerStyle.justifyContent = 'space-between';
|
|
143
|
+
headerStyle.gap = '8px';
|
|
144
|
+
const title = document.createElement('span');
|
|
145
|
+
title.textContent = 'Plumile Router';
|
|
146
|
+
title.style.fontWeight = '600';
|
|
147
|
+
const headerBtns = document.createElement('div');
|
|
148
|
+
const headerBtnsStyle = headerBtns.style;
|
|
149
|
+
headerBtnsStyle.display = 'flex';
|
|
150
|
+
headerBtnsStyle.gap = '4px';
|
|
151
|
+
headerBtnsStyle.alignItems = 'center';
|
|
152
|
+
const tabsWrap = document.createElement('div');
|
|
153
|
+
const tabsWrapStyle = tabsWrap.style;
|
|
154
|
+
tabsWrapStyle.display = 'flex';
|
|
155
|
+
tabsWrapStyle.gap = '4px';
|
|
156
|
+
tabsWrapStyle.marginRight = '6px';
|
|
157
|
+
const routeTabButton = document.createElement('button');
|
|
158
|
+
routeTabButton.textContent = 'Route';
|
|
159
|
+
const filtersTabButton = document.createElement('button');
|
|
160
|
+
filtersTabButton.textContent = 'Filters';
|
|
161
|
+
const tabButtons = [routeTabButton, filtersTabButton];
|
|
162
|
+
for (const tb of tabButtons) {
|
|
163
|
+
const s = tb.style;
|
|
164
|
+
s.cursor = 'pointer';
|
|
165
|
+
s.background = '#f1f1f1';
|
|
166
|
+
s.border = '1px solid #ccc';
|
|
167
|
+
s.borderRadius = '4px';
|
|
168
|
+
s.fontSize = '11px';
|
|
169
|
+
s.padding = '2px 6px';
|
|
170
|
+
}
|
|
171
|
+
routeTabButton.style.fontWeight = '600';
|
|
172
|
+
routeTabButton.dataset.active = 'true';
|
|
173
|
+
tabsWrap.appendChild(routeTabButton);
|
|
174
|
+
tabsWrap.appendChild(filtersTabButton);
|
|
175
|
+
const collapseBtn = document.createElement('button');
|
|
176
|
+
collapseBtn.textContent = '−';
|
|
177
|
+
const closeBtn = document.createElement('button');
|
|
178
|
+
closeBtn.textContent = '×';
|
|
179
|
+
const headerActionButtons = [collapseBtn, closeBtn];
|
|
180
|
+
for (const b of headerActionButtons) {
|
|
181
|
+
const st = b.style;
|
|
182
|
+
st.cursor = 'pointer';
|
|
183
|
+
st.background = '#eee';
|
|
184
|
+
st.border = '1px solid #ccc';
|
|
185
|
+
st.borderRadius = '4px';
|
|
186
|
+
st.fontSize = '11px';
|
|
187
|
+
st.lineHeight = '1';
|
|
188
|
+
st.padding = '2px 5px';
|
|
189
|
+
}
|
|
190
|
+
header.appendChild(title);
|
|
191
|
+
headerBtns.appendChild(tabsWrap);
|
|
192
|
+
headerBtns.appendChild(collapseBtn);
|
|
193
|
+
headerBtns.appendChild(closeBtn);
|
|
194
|
+
header.appendChild(headerBtns);
|
|
195
|
+
const pathDisplay = document.createElement('div');
|
|
196
|
+
pathDisplay.style.whiteSpace = 'nowrap';
|
|
197
|
+
const varsPre = document.createElement('pre');
|
|
198
|
+
const rawQueryPre = document.createElement('pre');
|
|
199
|
+
const typedQueryPre = document.createElement('pre');
|
|
200
|
+
const preBlocks = [varsPre, rawQueryPre, typedQueryPre];
|
|
201
|
+
for (const pb of preBlocks) {
|
|
202
|
+
const stp = pb.style;
|
|
203
|
+
stp.margin = '0';
|
|
204
|
+
stp.padding = '4px';
|
|
205
|
+
stp.background = '#f6f8fa';
|
|
206
|
+
stp.border = '1px solid #e2e2e2';
|
|
207
|
+
stp.borderRadius = '4px';
|
|
208
|
+
stp.overflowX = 'auto';
|
|
209
|
+
stp.maxHeight = '160px';
|
|
210
|
+
}
|
|
211
|
+
const timelineWrap = document.createElement('div');
|
|
212
|
+
const timelineWrapStyle = timelineWrap.style;
|
|
213
|
+
timelineWrapStyle.display = 'flex';
|
|
214
|
+
timelineWrapStyle.flexDirection = 'column';
|
|
215
|
+
timelineWrapStyle.gap = '2px';
|
|
216
|
+
timelineWrapStyle.marginTop = '4px';
|
|
217
|
+
const timelineHeader = document.createElement('div');
|
|
218
|
+
timelineHeader.textContent = 'Timeline (query changes)';
|
|
219
|
+
timelineHeader.style.fontWeight = '600';
|
|
220
|
+
timelineHeader.style.fontSize = '11px';
|
|
221
|
+
const timeline = document.createElement('div');
|
|
222
|
+
const timelineStyle = timeline.style;
|
|
223
|
+
timelineStyle.display = 'flex';
|
|
224
|
+
timelineStyle.flexDirection = 'column-reverse';
|
|
225
|
+
timelineStyle.maxHeight = '140px';
|
|
226
|
+
timelineStyle.overflowY = 'auto';
|
|
227
|
+
timelineStyle.border = '1px solid #e2e2e2';
|
|
228
|
+
timelineStyle.borderRadius = '4px';
|
|
229
|
+
timelineStyle.background = '#fafbfc';
|
|
230
|
+
timelineStyle.fontFamily = 'inherit';
|
|
231
|
+
timelineStyle.fontSize = '11px';
|
|
232
|
+
timelineWrap.appendChild(timelineHeader);
|
|
233
|
+
timelineWrap.appendChild(timeline);
|
|
234
|
+
const routeTabContent = document.createElement('div');
|
|
235
|
+
const routeTabContentStyle = routeTabContent.style;
|
|
236
|
+
routeTabContentStyle.display = 'flex';
|
|
237
|
+
routeTabContentStyle.flexDirection = 'column';
|
|
238
|
+
routeTabContentStyle.gap = '4px';
|
|
239
|
+
routeTabContent.appendChild(pathDisplay);
|
|
240
|
+
routeTabContent.appendChild(varsPre);
|
|
241
|
+
routeTabContent.appendChild(rawQueryPre);
|
|
242
|
+
routeTabContent.appendChild(typedQueryPre);
|
|
243
|
+
routeTabContent.appendChild(timelineWrap);
|
|
244
|
+
const filtersTabContent = document.createElement('div');
|
|
245
|
+
const filtersTabContentStyle = filtersTabContent.style;
|
|
246
|
+
filtersTabContentStyle.display = 'none';
|
|
247
|
+
filtersTabContentStyle.flexDirection = 'column';
|
|
248
|
+
filtersTabContentStyle.gap = '4px';
|
|
249
|
+
const filterActionsWrap = document.createElement('div');
|
|
250
|
+
const filterActionsWrapStyle = filterActionsWrap.style;
|
|
251
|
+
filterActionsWrapStyle.display = 'flex';
|
|
252
|
+
filterActionsWrapStyle.flexWrap = 'wrap';
|
|
253
|
+
filterActionsWrapStyle.gap = '4px';
|
|
254
|
+
const filtersBody = document.createElement('div');
|
|
255
|
+
const filtersBodyStyle = filtersBody.style;
|
|
256
|
+
filtersBodyStyle.display = 'flex';
|
|
257
|
+
filtersBodyStyle.flexDirection = 'column';
|
|
258
|
+
filtersBodyStyle.gap = '4px';
|
|
259
|
+
filtersTabContent.appendChild(filterActionsWrap);
|
|
260
|
+
filtersTabContent.appendChild(filtersBody);
|
|
261
|
+
const shortcutHint = document.createElement('div');
|
|
262
|
+
shortcutHint.textContent = `Shortcut: ${shortcut}`;
|
|
263
|
+
shortcutHint.style.fontSize = '10px';
|
|
264
|
+
shortcutHint.style.opacity = '0.7';
|
|
265
|
+
panel.appendChild(header);
|
|
266
|
+
panel.appendChild(routeTabContent);
|
|
267
|
+
panel.appendChild(filtersTabContent);
|
|
268
|
+
panel.appendChild(shortcutHint);
|
|
269
|
+
shadow.appendChild(panel);
|
|
270
|
+
document.body.appendChild(host);
|
|
271
|
+
let collapsed = false;
|
|
272
|
+
function onCollapseToggle() {
|
|
273
|
+
collapsed = !collapsed;
|
|
274
|
+
if (collapsed) {
|
|
275
|
+
panel.style.maxHeight = '24px';
|
|
276
|
+
panel.style.overflow = 'hidden';
|
|
277
|
+
collapseBtn.textContent = '+';
|
|
278
|
+
}
|
|
279
|
+
else {
|
|
280
|
+
panel.style.maxHeight = '50vh';
|
|
281
|
+
panel.style.overflow = 'auto';
|
|
282
|
+
collapseBtn.textContent = '−';
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
collapseBtn.addEventListener('click', onCollapseToggle);
|
|
286
|
+
let unsubscribe;
|
|
287
|
+
function onClose() {
|
|
288
|
+
host.remove();
|
|
289
|
+
if (typeof unsubscribe === 'function') {
|
|
290
|
+
unsubscribe();
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
closeBtn.addEventListener('click', onClose);
|
|
294
|
+
const navEvents = [];
|
|
295
|
+
function recordNavEvent(meta) {
|
|
296
|
+
navEvents.push(meta);
|
|
297
|
+
if (navEvents.length > 60)
|
|
298
|
+
navEvents.shift();
|
|
299
|
+
}
|
|
300
|
+
let lastQuery;
|
|
301
|
+
let lastSearch;
|
|
302
|
+
function render() {
|
|
303
|
+
try {
|
|
304
|
+
const entry = context.get();
|
|
305
|
+
pathDisplay.textContent = `${entry.location.pathname}${entry.location.search}`;
|
|
306
|
+
varsPre.textContent = `vars: ${JSON.stringify(entry.route?.params ?? {}, null, 2)}`;
|
|
307
|
+
rawQueryPre.textContent = `raw: ${JSON.stringify(entry.query, null, 2)}`;
|
|
308
|
+
typedQueryPre.textContent = `typed: ${JSON.stringify(entry.typedQuery, null, 2)}`;
|
|
309
|
+
const rawChanged = entry.rawSearch !== lastSearch;
|
|
310
|
+
if (rawChanged || verbose) {
|
|
311
|
+
const currentQuery = entry.query;
|
|
312
|
+
const diff = diffQueries(lastQuery, currentQuery);
|
|
313
|
+
const row = document.createElement('div');
|
|
314
|
+
const rowStyle = row.style;
|
|
315
|
+
rowStyle.display = 'flex';
|
|
316
|
+
rowStyle.flexDirection = 'column';
|
|
317
|
+
rowStyle.borderBottom = '1px solid #e9e9e9';
|
|
318
|
+
rowStyle.padding = '3px 4px';
|
|
319
|
+
const head = document.createElement('div');
|
|
320
|
+
const headStyle = head.style;
|
|
321
|
+
headStyle.display = 'flex';
|
|
322
|
+
headStyle.justifyContent = 'space-between';
|
|
323
|
+
const timeStr = new Date().toLocaleTimeString();
|
|
324
|
+
let searchDisplay = '(empty)';
|
|
325
|
+
if (entry.location.search !== '') {
|
|
326
|
+
searchDisplay = entry.location.search;
|
|
327
|
+
}
|
|
328
|
+
let origin = 'system';
|
|
329
|
+
for (let i = navEvents.length - 1; i >= 0; i -= 1) {
|
|
330
|
+
const ev = navEvents[i];
|
|
331
|
+
if (ev != null && ev.search === entry.rawSearch) {
|
|
332
|
+
origin = `${ev.kind}:${ev.mode}`;
|
|
333
|
+
if (ev.kind === 'batch-flush') {
|
|
334
|
+
origin = `${origin}(${ev.count})`;
|
|
335
|
+
}
|
|
336
|
+
break;
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
head.innerHTML = [
|
|
340
|
+
`<strong>${timeStr}</strong>`,
|
|
341
|
+
`<code style="background:#eee;padding:1px 3px;border-radius:3px;">${searchDisplay}</code>`,
|
|
342
|
+
`<span style="font-size:10px;opacity:0.7;">${origin}</span>`,
|
|
343
|
+
].join(' ');
|
|
344
|
+
const body = document.createElement('div');
|
|
345
|
+
const bodyStyle = body.style;
|
|
346
|
+
bodyStyle.display = 'flex';
|
|
347
|
+
bodyStyle.flexWrap = 'wrap';
|
|
348
|
+
bodyStyle.gap = '4px';
|
|
349
|
+
if (diff.added.length > 0) {
|
|
350
|
+
for (const k of diff.added) {
|
|
351
|
+
body.appendChild(createQueryDiffPill(`+${k}`, 'added'));
|
|
352
|
+
}
|
|
353
|
+
}
|
|
354
|
+
if (diff.changed.length > 0) {
|
|
355
|
+
for (const k of diff.changed) {
|
|
356
|
+
body.appendChild(createQueryDiffPill(`~${k}`, 'changed'));
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
if (diff.removed.length > 0) {
|
|
360
|
+
for (const k of diff.removed) {
|
|
361
|
+
body.appendChild(createQueryDiffPill(`-${k}`, 'removed'));
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
if (rawChanged) {
|
|
365
|
+
if (diff.added.length === 0 &&
|
|
366
|
+
diff.changed.length === 0 &&
|
|
367
|
+
diff.removed.length === 0) {
|
|
368
|
+
body.appendChild(createQueryDiffPill('no-change', 'unchanged'));
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
else if (verbose) {
|
|
372
|
+
body.appendChild(createQueryDiffPill('same-search', 'unchanged'));
|
|
373
|
+
}
|
|
374
|
+
row.appendChild(head);
|
|
375
|
+
row.appendChild(body);
|
|
376
|
+
timeline.appendChild(row);
|
|
377
|
+
if (timeline.childElementCount > 50) {
|
|
378
|
+
const first = timeline.firstElementChild;
|
|
379
|
+
if (first != null) {
|
|
380
|
+
timeline.removeChild(first);
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
if (rawChanged) {
|
|
384
|
+
lastQuery = currentQuery;
|
|
385
|
+
lastSearch = entry.rawSearch;
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
try {
|
|
389
|
+
const merged = context.currentMergedFilterSchema();
|
|
390
|
+
while (filterActionsWrap.firstChild != null) {
|
|
391
|
+
filterActionsWrap.removeChild(filterActionsWrap.firstChild);
|
|
392
|
+
}
|
|
393
|
+
while (filtersBody.firstChild != null) {
|
|
394
|
+
filtersBody.removeChild(filtersBody.firstChild);
|
|
395
|
+
}
|
|
396
|
+
if (merged == null) {
|
|
397
|
+
const emptyMsg = document.createElement('div');
|
|
398
|
+
emptyMsg.textContent = 'No filter schema for current route';
|
|
399
|
+
emptyMsg.style.fontStyle = 'italic';
|
|
400
|
+
filtersBody.appendChild(emptyMsg);
|
|
401
|
+
}
|
|
402
|
+
else {
|
|
403
|
+
const schemaObj = merged.schema;
|
|
404
|
+
let parsedFilters = {};
|
|
405
|
+
try {
|
|
406
|
+
parsedFilters = parseFilters(merged, entry.query);
|
|
407
|
+
}
|
|
408
|
+
catch {
|
|
409
|
+
}
|
|
410
|
+
const fieldNames = Object.keys(schemaObj).sort();
|
|
411
|
+
let anyActive = false;
|
|
412
|
+
for (const f of fieldNames) {
|
|
413
|
+
const state = parsedFilters[f];
|
|
414
|
+
if (state != null && Object.keys(state).length > 0) {
|
|
415
|
+
anyActive = true;
|
|
416
|
+
break;
|
|
417
|
+
}
|
|
418
|
+
}
|
|
419
|
+
if (anyActive) {
|
|
420
|
+
const clearAllBtn = document.createElement('button');
|
|
421
|
+
clearAllBtn.textContent = 'Clear All Filters';
|
|
422
|
+
styleMiniButton(clearAllBtn, '#d73a49');
|
|
423
|
+
function onClearAllFilters() {
|
|
424
|
+
try {
|
|
425
|
+
const blank = {};
|
|
426
|
+
for (const fname of fieldNames) {
|
|
427
|
+
blank[fname] = {};
|
|
428
|
+
}
|
|
429
|
+
context.navigate({
|
|
430
|
+
filters: blank,
|
|
431
|
+
filterSchema: merged,
|
|
432
|
+
batch: true,
|
|
433
|
+
});
|
|
434
|
+
}
|
|
435
|
+
catch {
|
|
436
|
+
}
|
|
437
|
+
}
|
|
438
|
+
clearAllBtn.addEventListener('click', onClearAllFilters);
|
|
439
|
+
filterActionsWrap.appendChild(clearAllBtn);
|
|
440
|
+
}
|
|
441
|
+
for (const field of fieldNames) {
|
|
442
|
+
const defLike = schemaObj[field];
|
|
443
|
+
const fieldBox = document.createElement('div');
|
|
444
|
+
const fbStyle = fieldBox.style;
|
|
445
|
+
fbStyle.border = '1px solid #e2e2e2';
|
|
446
|
+
fbStyle.borderRadius = '4px';
|
|
447
|
+
fbStyle.padding = '4px 6px';
|
|
448
|
+
fbStyle.background = '#f9fafb';
|
|
449
|
+
fbStyle.display = 'flex';
|
|
450
|
+
fbStyle.flexDirection = 'column';
|
|
451
|
+
fbStyle.gap = '3px';
|
|
452
|
+
const fieldHead = document.createElement('div');
|
|
453
|
+
const headStyle2 = fieldHead.style;
|
|
454
|
+
headStyle2.display = 'flex';
|
|
455
|
+
headStyle2.justifyContent = 'space-between';
|
|
456
|
+
headStyle2.alignItems = 'center';
|
|
457
|
+
const labelEl = document.createElement('strong');
|
|
458
|
+
let labelText = field;
|
|
459
|
+
if (defLike?.label != null) {
|
|
460
|
+
labelText = String(defLike.label);
|
|
461
|
+
}
|
|
462
|
+
labelEl.textContent = labelText;
|
|
463
|
+
labelEl.style.fontSize = '11px';
|
|
464
|
+
const headActions = document.createElement('div');
|
|
465
|
+
headActions.style.display = 'flex';
|
|
466
|
+
headActions.style.gap = '4px';
|
|
467
|
+
const clearFieldBtn = document.createElement('button');
|
|
468
|
+
clearFieldBtn.textContent = 'clear';
|
|
469
|
+
styleMiniButton(clearFieldBtn, '#6a737d');
|
|
470
|
+
function onClearField() {
|
|
471
|
+
try {
|
|
472
|
+
const next = { ...parsedFilters };
|
|
473
|
+
next[field] = {};
|
|
474
|
+
context.navigate({
|
|
475
|
+
filters: next,
|
|
476
|
+
filterSchema: merged,
|
|
477
|
+
batch: true,
|
|
478
|
+
});
|
|
479
|
+
}
|
|
480
|
+
catch {
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
clearFieldBtn.addEventListener('click', onClearField);
|
|
484
|
+
headActions.appendChild(clearFieldBtn);
|
|
485
|
+
fieldHead.appendChild(labelEl);
|
|
486
|
+
fieldHead.appendChild(headActions);
|
|
487
|
+
fieldBox.appendChild(fieldHead);
|
|
488
|
+
const opsWrap = document.createElement('div');
|
|
489
|
+
const opsWrapStyle = opsWrap.style;
|
|
490
|
+
opsWrapStyle.display = 'flex';
|
|
491
|
+
opsWrapStyle.flexWrap = 'wrap';
|
|
492
|
+
opsWrapStyle.gap = '4px';
|
|
493
|
+
const present = parsedFilters[field] ?? {};
|
|
494
|
+
const defaults = defLike?.defaults ?? {};
|
|
495
|
+
let allowed = [];
|
|
496
|
+
if (Array.isArray(defLike?.operators)) {
|
|
497
|
+
allowed = defLike.operators.slice();
|
|
498
|
+
}
|
|
499
|
+
else if (defLike?.kind === 'number') {
|
|
500
|
+
allowed = [
|
|
501
|
+
'eq',
|
|
502
|
+
'neq',
|
|
503
|
+
'gt',
|
|
504
|
+
'gte',
|
|
505
|
+
'lt',
|
|
506
|
+
'lte',
|
|
507
|
+
'in',
|
|
508
|
+
'nin',
|
|
509
|
+
'between',
|
|
510
|
+
'exists',
|
|
511
|
+
];
|
|
512
|
+
}
|
|
513
|
+
else {
|
|
514
|
+
allowed = [
|
|
515
|
+
'eq',
|
|
516
|
+
'neq',
|
|
517
|
+
'in',
|
|
518
|
+
'nin',
|
|
519
|
+
'between',
|
|
520
|
+
'exists',
|
|
521
|
+
'contains',
|
|
522
|
+
'starts',
|
|
523
|
+
'ends',
|
|
524
|
+
'regex',
|
|
525
|
+
];
|
|
526
|
+
}
|
|
527
|
+
for (const op of allowed) {
|
|
528
|
+
const span = document.createElement('span');
|
|
529
|
+
const spanStyle = span.style;
|
|
530
|
+
spanStyle.display = 'inline-flex';
|
|
531
|
+
spanStyle.alignItems = 'center';
|
|
532
|
+
spanStyle.gap = '2px';
|
|
533
|
+
spanStyle.padding = '1px 5px';
|
|
534
|
+
spanStyle.borderRadius = '12px';
|
|
535
|
+
spanStyle.border = '1px solid #d0d7de';
|
|
536
|
+
spanStyle.fontSize = '10px';
|
|
537
|
+
const activeVal = present[op];
|
|
538
|
+
if (activeVal != null) {
|
|
539
|
+
spanStyle.background = '#e6ffed';
|
|
540
|
+
spanStyle.borderColor = '#34d058';
|
|
541
|
+
const opText = document.createElement('span');
|
|
542
|
+
opText.textContent = op;
|
|
543
|
+
span.appendChild(opText);
|
|
544
|
+
const closeBtnOp = document.createElement('button');
|
|
545
|
+
closeBtnOp.textContent = '×';
|
|
546
|
+
stylePillClose(closeBtnOp);
|
|
547
|
+
function onRemoveOperator() {
|
|
548
|
+
try {
|
|
549
|
+
const next = { ...parsedFilters };
|
|
550
|
+
const existing = next[field] ?? {};
|
|
551
|
+
const copyField = {};
|
|
552
|
+
for (const k of Object.keys(existing)) {
|
|
553
|
+
if (k !== op) {
|
|
554
|
+
copyField[k] = existing[k];
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
next[field] = copyField;
|
|
558
|
+
context.navigate({
|
|
559
|
+
filters: next,
|
|
560
|
+
filterSchema: merged,
|
|
561
|
+
batch: true,
|
|
562
|
+
});
|
|
563
|
+
}
|
|
564
|
+
catch {
|
|
565
|
+
}
|
|
566
|
+
}
|
|
567
|
+
closeBtnOp.addEventListener('click', onRemoveOperator);
|
|
568
|
+
span.appendChild(closeBtnOp);
|
|
569
|
+
}
|
|
570
|
+
else {
|
|
571
|
+
const isDefault = Object.prototype.hasOwnProperty.call(defaults, op);
|
|
572
|
+
if (isDefault) {
|
|
573
|
+
span.textContent = `${op} (default)`;
|
|
574
|
+
spanStyle.fontStyle = 'italic';
|
|
575
|
+
span.title = 'Default value omitted in URL';
|
|
576
|
+
}
|
|
577
|
+
else {
|
|
578
|
+
span.textContent = op;
|
|
579
|
+
}
|
|
580
|
+
spanStyle.opacity = '0.55';
|
|
581
|
+
}
|
|
582
|
+
opsWrap.appendChild(span);
|
|
583
|
+
}
|
|
584
|
+
fieldBox.appendChild(opsWrap);
|
|
585
|
+
filtersBody.appendChild(fieldBox);
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
}
|
|
589
|
+
catch {
|
|
590
|
+
}
|
|
591
|
+
}
|
|
592
|
+
catch {
|
|
593
|
+
}
|
|
594
|
+
}
|
|
595
|
+
render();
|
|
596
|
+
unsubscribe = context.subscribe(render);
|
|
597
|
+
function activateTab(name) {
|
|
598
|
+
if (name === 'route') {
|
|
599
|
+
routeTabContent.style.display = 'flex';
|
|
600
|
+
filtersTabContent.style.display = 'none';
|
|
601
|
+
routeTabButton.dataset.active = 'true';
|
|
602
|
+
filtersTabButton.dataset.active = 'false';
|
|
603
|
+
routeTabButton.style.fontWeight = '600';
|
|
604
|
+
filtersTabButton.style.fontWeight = 'normal';
|
|
605
|
+
}
|
|
606
|
+
else {
|
|
607
|
+
routeTabContent.style.display = 'none';
|
|
608
|
+
filtersTabContent.style.display = 'flex';
|
|
609
|
+
filtersTabButton.dataset.active = 'true';
|
|
610
|
+
routeTabButton.dataset.active = 'false';
|
|
611
|
+
filtersTabButton.style.fontWeight = '600';
|
|
612
|
+
routeTabButton.style.fontWeight = 'normal';
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
function onRouteTabClick() {
|
|
616
|
+
activateTab('route');
|
|
617
|
+
}
|
|
618
|
+
function onFiltersTabClick() {
|
|
619
|
+
activateTab('filters');
|
|
620
|
+
}
|
|
621
|
+
routeTabButton.addEventListener('click', onRouteTabClick);
|
|
622
|
+
filtersTabButton.addEventListener('click', onFiltersTabClick);
|
|
623
|
+
function onKeydown(ev) {
|
|
624
|
+
try {
|
|
625
|
+
const parts = [];
|
|
626
|
+
if (ev.altKey)
|
|
627
|
+
parts.push('Alt');
|
|
628
|
+
if (ev.shiftKey)
|
|
629
|
+
parts.push('Shift');
|
|
630
|
+
if (ev.ctrlKey)
|
|
631
|
+
parts.push('Ctrl');
|
|
632
|
+
let keyPart = ev.key;
|
|
633
|
+
if (ev.key.length === 1) {
|
|
634
|
+
keyPart = ev.key.toUpperCase();
|
|
635
|
+
}
|
|
636
|
+
parts.push(keyPart);
|
|
637
|
+
const combo = parts.join('+');
|
|
638
|
+
const normShortcut = shortcut.replace(/\s+/g, '');
|
|
639
|
+
if (combo === normShortcut) {
|
|
640
|
+
if (host.style.display === 'none') {
|
|
641
|
+
host.style.display = 'block';
|
|
642
|
+
}
|
|
643
|
+
else {
|
|
644
|
+
host.style.display = 'none';
|
|
645
|
+
}
|
|
646
|
+
}
|
|
647
|
+
}
|
|
648
|
+
catch {
|
|
649
|
+
}
|
|
650
|
+
}
|
|
651
|
+
window.addEventListener('keydown', onKeydown);
|
|
652
|
+
return recordNavEvent;
|
|
653
|
+
}
|
|
654
|
+
catch {
|
|
655
|
+
return function silentRecorder() {
|
|
656
|
+
};
|
|
657
|
+
}
|
|
658
|
+
}
|
|
659
|
+
export function initRouterDevtools(params) {
|
|
660
|
+
const { context, options } = params;
|
|
661
|
+
const { global, panel, shortcut } = options;
|
|
662
|
+
if (global) {
|
|
663
|
+
try {
|
|
664
|
+
if (typeof window !== 'undefined') {
|
|
665
|
+
window.__PLUMILE_ROUTER__ = {
|
|
666
|
+
get: context.get,
|
|
667
|
+
subscribe: context.subscribe,
|
|
668
|
+
};
|
|
669
|
+
}
|
|
670
|
+
}
|
|
671
|
+
catch {
|
|
672
|
+
}
|
|
673
|
+
}
|
|
674
|
+
function noopNavRecorder() {
|
|
675
|
+
}
|
|
676
|
+
let navRecorder = noopNavRecorder;
|
|
677
|
+
if (panel) {
|
|
678
|
+
navRecorder = mountDevtoolsPanel({
|
|
679
|
+
context,
|
|
680
|
+
shortcut,
|
|
681
|
+
verbose: options.verbose,
|
|
682
|
+
});
|
|
683
|
+
}
|
|
684
|
+
function handleNavEvent(meta) {
|
|
685
|
+
navRecorder(meta);
|
|
686
|
+
}
|
|
687
|
+
return handleNavEvent;
|
|
688
|
+
}
|
|
689
|
+
export default initRouterDevtools;
|
|
690
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"devtools.js","sourceRoot":"","sources":["../../../src/routing/devtools.ts"],"names":[],"mappings":"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","sourcesContent":["/* Router Devtools extracted module (development-only).\n * Objectif refactor: lisibilité améliorée (noms explicites, helpers externalisés).\n */\nimport type { RoutingContextType } from '../types.js';\nimport { parseFilters } from './filters.js';\n\nexport interface InitRouterDevtoolsOptions {\n  global: boolean;\n  panel: boolean;\n  shortcut: string;\n  verbose?: boolean;\n}\nexport interface NavEventMetaPublic {\n  kind: string;\n  mode: string;\n  count: number;\n  pathname: string;\n  search: string;\n  timestamp: number;\n}\n\n/** Init devtools and return nav event callback (see exported init). */\n/**\n * Compute a diff between two query objects (flat key-value maps).\n * Clones are not performed: keys are inspected directly and categorized.\n * @param prev Previous query object (can be undefined on first render).\n * @param next Current query object.\n * @returns Buckets of keys: added, removed, changed, unchanged.\n */\nfunction diffQueries(\n  prev: Record<string, unknown> | undefined,\n  next: Record<string, unknown>,\n) {\n  const added: string[] = [];\n  const removed: string[] = [];\n  const changed: string[] = [];\n  const unchanged: string[] = [];\n  const prevObj = prev ?? {};\n  const prevKeys = new Set(Object.keys(prevObj));\n  for (const k of Object.keys(next)) {\n    if (!prevKeys.has(k)) {\n      added.push(k);\n    } else {\n      const pv = prevObj[k];\n      const nv = next[k];\n      let same = false;\n      if (Array.isArray(pv) && Array.isArray(nv) && pv.length === nv.length) {\n        same = pv.every((v, i) => {\n          return v === nv[i];\n        });\n      } else {\n        same = pv === nv;\n      }\n      if (same) {\n        unchanged.push(k);\n      } else {\n        changed.push(k);\n      }\n      prevKeys.delete(k);\n    }\n  }\n  for (const k of prevKeys) {\n    removed.push(k);\n  }\n  return { added, removed, changed, unchanged };\n}\n/**\n * Apply minimal button styling used for small action buttons in the devtools panel.\n * @param btn Target button element.\n * @param color Text color to apply.\n */\nfunction styleMiniButton(btn: HTMLButtonElement, color: string): void {\n  const { style } = btn;\n  style.cursor = 'pointer';\n  style.background = '#fff';\n  style.border = '1px solid #ccc';\n  style.borderRadius = '4px';\n  style.fontSize = '10px';\n  style.lineHeight = '1';\n  style.padding = '2px 6px';\n  style.color = color;\n}\n/**\n * Style the close (x) button that appears inside query / operator pills.\n * @param btn Target button element.\n */\nfunction stylePillClose(btn: HTMLButtonElement): void {\n  const { style } = btn;\n  style.cursor = 'pointer';\n  style.background = 'transparent';\n  style.border = 'none';\n  style.fontSize = '10px';\n  style.lineHeight = '1';\n  style.padding = '0 2px';\n  style.color = '#444';\n}\n/**\n * Create a colored pill element representing one key in a query diff.\n * Color scheme communicates semantic (added / removed / changed / unchanged).\n * @param label Display text for the pill.\n * @param kind Diff category.\n * @returns Span element ready to attach.\n */\nfunction createQueryDiffPill(\n  label: string,\n  kind: 'added' | 'removed' | 'changed' | 'unchanged',\n): HTMLSpanElement {\n  const span = document.createElement('span');\n  span.textContent = label;\n  span.style.padding = '1px 4px';\n  span.style.borderRadius = '10px';\n  span.style.fontSize = '10px';\n  span.style.border = '1px solid';\n  if (kind === 'added') {\n    span.style.background = '#e6ffed';\n    span.style.borderColor = '#2cbe4e';\n    span.style.color = '#22863a';\n  } else if (kind === 'removed') {\n    span.style.background = '#ffeef0';\n    span.style.borderColor = '#d73a49';\n    span.style.color = '#cb2431';\n  } else if (kind === 'changed') {\n    span.style.background = '#fff5b1';\n    span.style.borderColor = '#ffd33d';\n    span.style.color = '#735c0f';\n  } else {\n    span.style.background = '#f1f8ff';\n    span.style.borderColor = '#79b8ff';\n    span.style.color = '#0366d6';\n  }\n  return span;\n}\n/**\n * Mount the floating devtools panel (Shadow DOM host + internal structure).\n * Guarded for non-DOM environments and idempotent (won't mount twice).\n * @param params.context Routing context (provides state + subscribe).\n * @param params.shortcut Keyboard shortcut (e.g. \"Ctrl+Shift+R\") to toggle visibility.\n * @returns Recorder function to capture navigation events for timeline origin metadata.\n */\nfunction mountDevtoolsPanel(params: {\n  context: RoutingContextType<any>;\n  shortcut: string;\n  verbose?: boolean;\n}): (meta: NavEventMetaPublic) => void {\n  try {\n    if (typeof window === 'undefined' || typeof document === 'undefined') {\n      return function noopRecorder() {\n        /* noop (non-DOM env) */\n      };\n    }\n    if (\n      document.querySelector('[data-plumile-router-devtools-panel]') != null\n    ) {\n      return function alreadyMountedRecorder() {\n        /* noop (already mounted) */\n      };\n    }\n    const { context, shortcut, verbose } = params;\n\n    // Host\n    const host = document.createElement('div');\n    host.setAttribute('data-plumile-router-devtools-panel', '');\n    const hostStyle = host.style;\n    hostStyle.position = 'fixed';\n    hostStyle.zIndex = '2147483646';\n    hostStyle.bottom = '8px';\n    hostStyle.right = '8px';\n    hostStyle.fontFamily = [\n      'ui-monospace',\n      'SFMono-Regular',\n      'Menlo',\n      'Monaco',\n      'Consolas',\n      '\"Liberation Mono\"',\n      'monospace',\n    ].join(', ');\n    hostStyle.fontSize = '12px';\n    hostStyle.lineHeight = '1.3';\n    hostStyle.color = '#111';\n    hostStyle.background = 'transparent';\n    hostStyle.pointerEvents = 'none';\n    const shadow = host.attachShadow({ mode: 'open' });\n\n    // Panel container\n    const panel = document.createElement('div');\n    const panelStyle = panel.style;\n    panelStyle.pointerEvents = 'auto';\n    panelStyle.minWidth = '260px';\n    panelStyle.maxWidth = '420px';\n    panelStyle.maxHeight = '50vh';\n    panelStyle.overflow = 'auto';\n    panelStyle.background = 'rgba(255,255,255,0.95)';\n    panelStyle.border = '1px solid #ccc';\n    panelStyle.borderRadius = '6px';\n    panelStyle.boxShadow = '0 2px 8px rgba(0,0,0,0.15)';\n    panelStyle.padding = '6px 8px 8px';\n    panelStyle.backdropFilter = 'blur(4px)';\n    panelStyle.display = 'flex';\n    panelStyle.flexDirection = 'column';\n    panelStyle.rowGap = '4px';\n\n    // Header\n    const header = document.createElement('div');\n    const headerStyle = header.style;\n    headerStyle.display = 'flex';\n    headerStyle.alignItems = 'center';\n    headerStyle.justifyContent = 'space-between';\n    headerStyle.gap = '8px';\n    const title = document.createElement('span');\n    title.textContent = 'Plumile Router';\n    title.style.fontWeight = '600';\n    const headerBtns = document.createElement('div');\n    const headerBtnsStyle = headerBtns.style;\n    headerBtnsStyle.display = 'flex';\n    headerBtnsStyle.gap = '4px';\n    headerBtnsStyle.alignItems = 'center';\n    const tabsWrap = document.createElement('div');\n    const tabsWrapStyle = tabsWrap.style;\n    tabsWrapStyle.display = 'flex';\n    tabsWrapStyle.gap = '4px';\n    tabsWrapStyle.marginRight = '6px';\n    const routeTabButton = document.createElement('button');\n    routeTabButton.textContent = 'Route';\n    const filtersTabButton = document.createElement('button');\n    filtersTabButton.textContent = 'Filters';\n    const tabButtons: HTMLButtonElement[] = [routeTabButton, filtersTabButton];\n    for (const tb of tabButtons) {\n      const s = tb.style;\n      s.cursor = 'pointer';\n      s.background = '#f1f1f1';\n      s.border = '1px solid #ccc';\n      s.borderRadius = '4px';\n      s.fontSize = '11px';\n      s.padding = '2px 6px';\n    }\n    routeTabButton.style.fontWeight = '600';\n    routeTabButton.dataset.active = 'true';\n    tabsWrap.appendChild(routeTabButton);\n    tabsWrap.appendChild(filtersTabButton);\n    const collapseBtn = document.createElement('button');\n    collapseBtn.textContent = '−';\n    const closeBtn = document.createElement('button');\n    closeBtn.textContent = '×';\n    const headerActionButtons: HTMLButtonElement[] = [collapseBtn, closeBtn];\n    for (const b of headerActionButtons) {\n      const st = b.style;\n      st.cursor = 'pointer';\n      st.background = '#eee';\n      st.border = '1px solid #ccc';\n      st.borderRadius = '4px';\n      st.fontSize = '11px';\n      st.lineHeight = '1';\n      st.padding = '2px 5px';\n    }\n    header.appendChild(title);\n    headerBtns.appendChild(tabsWrap);\n    headerBtns.appendChild(collapseBtn);\n    headerBtns.appendChild(closeBtn);\n    header.appendChild(headerBtns);\n\n    // Route tab elements\n    const pathDisplay = document.createElement('div');\n    pathDisplay.style.whiteSpace = 'nowrap';\n    const varsPre = document.createElement('pre');\n    const rawQueryPre = document.createElement('pre');\n    const typedQueryPre = document.createElement('pre');\n    const preBlocks: HTMLPreElement[] = [varsPre, rawQueryPre, typedQueryPre];\n    for (const pb of preBlocks) {\n      const stp = pb.style;\n      stp.margin = '0';\n      stp.padding = '4px';\n      stp.background = '#f6f8fa';\n      stp.border = '1px solid #e2e2e2';\n      stp.borderRadius = '4px';\n      stp.overflowX = 'auto';\n      stp.maxHeight = '160px';\n    }\n    const timelineWrap = document.createElement('div');\n    const timelineWrapStyle = timelineWrap.style;\n    timelineWrapStyle.display = 'flex';\n    timelineWrapStyle.flexDirection = 'column';\n    timelineWrapStyle.gap = '2px';\n    timelineWrapStyle.marginTop = '4px';\n    const timelineHeader = document.createElement('div');\n    timelineHeader.textContent = 'Timeline (query changes)';\n    timelineHeader.style.fontWeight = '600';\n    timelineHeader.style.fontSize = '11px';\n    const timeline = document.createElement('div');\n    const timelineStyle = timeline.style;\n    timelineStyle.display = 'flex';\n    timelineStyle.flexDirection = 'column-reverse';\n    timelineStyle.maxHeight = '140px';\n    timelineStyle.overflowY = 'auto';\n    timelineStyle.border = '1px solid #e2e2e2';\n    timelineStyle.borderRadius = '4px';\n    timelineStyle.background = '#fafbfc';\n    timelineStyle.fontFamily = 'inherit';\n    timelineStyle.fontSize = '11px';\n    timelineWrap.appendChild(timelineHeader);\n    timelineWrap.appendChild(timeline);\n    const routeTabContent = document.createElement('div');\n    const routeTabContentStyle = routeTabContent.style;\n    routeTabContentStyle.display = 'flex';\n    routeTabContentStyle.flexDirection = 'column';\n    routeTabContentStyle.gap = '4px';\n    routeTabContent.appendChild(pathDisplay);\n    routeTabContent.appendChild(varsPre);\n    routeTabContent.appendChild(rawQueryPre);\n    routeTabContent.appendChild(typedQueryPre);\n    routeTabContent.appendChild(timelineWrap);\n\n    // Filters tab skeleton\n    const filtersTabContent = document.createElement('div');\n    const filtersTabContentStyle = filtersTabContent.style;\n    filtersTabContentStyle.display = 'none';\n    filtersTabContentStyle.flexDirection = 'column';\n    filtersTabContentStyle.gap = '4px';\n    const filterActionsWrap = document.createElement('div');\n    const filterActionsWrapStyle = filterActionsWrap.style;\n    filterActionsWrapStyle.display = 'flex';\n    filterActionsWrapStyle.flexWrap = 'wrap';\n    filterActionsWrapStyle.gap = '4px';\n    const filtersBody = document.createElement('div');\n    const filtersBodyStyle = filtersBody.style;\n    filtersBodyStyle.display = 'flex';\n    filtersBodyStyle.flexDirection = 'column';\n    filtersBodyStyle.gap = '4px';\n    filtersTabContent.appendChild(filterActionsWrap);\n    filtersTabContent.appendChild(filtersBody);\n\n    const shortcutHint = document.createElement('div');\n    shortcutHint.textContent = `Shortcut: ${shortcut}`;\n    shortcutHint.style.fontSize = '10px';\n    shortcutHint.style.opacity = '0.7';\n\n    panel.appendChild(header);\n    panel.appendChild(routeTabContent);\n    panel.appendChild(filtersTabContent);\n    panel.appendChild(shortcutHint);\n    shadow.appendChild(panel);\n    document.body.appendChild(host);\n\n    // Collapse / close logic\n    let collapsed = false;\n    /** Toggle collapsed/expanded panel state. */\n    function onCollapseToggle(): void {\n      collapsed = !collapsed;\n      if (collapsed) {\n        panel.style.maxHeight = '24px';\n        panel.style.overflow = 'hidden';\n        collapseBtn.textContent = '+';\n      } else {\n        panel.style.maxHeight = '50vh';\n        panel.style.overflow = 'auto';\n        collapseBtn.textContent = '−';\n      }\n    }\n    collapseBtn.addEventListener('click', onCollapseToggle);\n    let unsubscribe: undefined | (() => void);\n    /** Close and unmount the panel, unsubscribing listener. */\n    function onClose(): void {\n      host.remove();\n      if (typeof unsubscribe === 'function') {\n        unsubscribe();\n      }\n    }\n    closeBtn.addEventListener('click', onClose);\n\n    // Nav event store\n    const navEvents: NavEventMetaPublic[] = [];\n    /**\n     * Record a navigation event for later attribution when rendering timeline rows.\n     * Stores only a bounded ring (FIFO trimming beyond 60 entries).\n     * @param meta Navigation meta event object.\n     */\n    /** Record a nav event (bounded history length). */\n    function recordNavEvent(meta: NavEventMetaPublic): void {\n      navEvents.push(meta);\n      if (navEvents.length > 60) navEvents.shift();\n    }\n\n    // Render state cache\n    let lastQuery: Record<string, unknown> | undefined;\n    let lastSearch: string | undefined;\n\n    /**\n     * Re-render the panel UI from current routing context & cached navigation events.\n     * Contains defensive try/catch blocks to avoid breaking the app if devtools fails.\n     */\n    /** Render panel contents from context state. */\n    function render(): void {\n      try {\n        const entry = context.get();\n        pathDisplay.textContent = `${entry.location.pathname}${entry.location.search}`;\n        varsPre.textContent = `vars: ${JSON.stringify(entry.route?.params ?? {}, null, 2)}`;\n        rawQueryPre.textContent = `raw:  ${JSON.stringify(entry.query, null, 2)}`;\n        typedQueryPre.textContent = `typed: ${JSON.stringify(entry.typedQuery, null, 2)}`;\n        const rawChanged = entry.rawSearch !== lastSearch;\n        if (rawChanged || verbose) {\n          const currentQuery = entry.query as Record<string, unknown>;\n          const diff = diffQueries(lastQuery, currentQuery);\n          const row = document.createElement('div');\n          const rowStyle = row.style;\n          rowStyle.display = 'flex';\n          rowStyle.flexDirection = 'column';\n          rowStyle.borderBottom = '1px solid #e9e9e9';\n          rowStyle.padding = '3px 4px';\n          const head = document.createElement('div');\n          const headStyle = head.style;\n          headStyle.display = 'flex';\n          headStyle.justifyContent = 'space-between';\n          const timeStr = new Date().toLocaleTimeString();\n          let searchDisplay = '(empty)';\n          if (entry.location.search !== '') {\n            searchDisplay = entry.location.search;\n          }\n          let origin = 'system';\n          for (let i = navEvents.length - 1; i >= 0; i -= 1) {\n            const ev = navEvents[i];\n            if (ev != null && ev.search === entry.rawSearch) {\n              origin = `${ev.kind}:${ev.mode}`;\n              if (ev.kind === 'batch-flush') {\n                origin = `${origin}(${ev.count})`;\n              }\n              break;\n            }\n          }\n          head.innerHTML = [\n            `<strong>${timeStr}</strong>`,\n            `<code style=\"background:#eee;padding:1px 3px;border-radius:3px;\">${searchDisplay}</code>`,\n            `<span style=\"font-size:10px;opacity:0.7;\">${origin}</span>`,\n          ].join(' ');\n          const body = document.createElement('div');\n          const bodyStyle = body.style;\n          bodyStyle.display = 'flex';\n          bodyStyle.flexWrap = 'wrap';\n          bodyStyle.gap = '4px';\n          if (diff.added.length > 0) {\n            for (const k of diff.added) {\n              body.appendChild(createQueryDiffPill(`+${k}`, 'added'));\n            }\n          }\n          if (diff.changed.length > 0) {\n            for (const k of diff.changed) {\n              body.appendChild(createQueryDiffPill(`~${k}`, 'changed'));\n            }\n          }\n          if (diff.removed.length > 0) {\n            for (const k of diff.removed) {\n              body.appendChild(createQueryDiffPill(`-${k}`, 'removed'));\n            }\n          }\n          if (rawChanged) {\n            if (\n              diff.added.length === 0 &&\n              diff.changed.length === 0 &&\n              diff.removed.length === 0\n            ) {\n              body.appendChild(createQueryDiffPill('no-change', 'unchanged'));\n            }\n          } else if (verbose) {\n            body.appendChild(createQueryDiffPill('same-search', 'unchanged'));\n          }\n          row.appendChild(head);\n          row.appendChild(body);\n          timeline.appendChild(row);\n          if (timeline.childElementCount > 50) {\n            const first = timeline.firstElementChild;\n            if (first != null) {\n              timeline.removeChild(first);\n            }\n          }\n          if (rawChanged) {\n            lastQuery = currentQuery;\n            lastSearch = entry.rawSearch;\n          }\n        }\n        // Filters tab\n        try {\n          const merged = context.currentMergedFilterSchema();\n          while (filterActionsWrap.firstChild != null) {\n            filterActionsWrap.removeChild(filterActionsWrap.firstChild);\n          }\n          while (filtersBody.firstChild != null) {\n            filtersBody.removeChild(filtersBody.firstChild);\n          }\n          if (merged == null) {\n            const emptyMsg = document.createElement('div');\n            emptyMsg.textContent = 'No filter schema for current route';\n            emptyMsg.style.fontStyle = 'italic';\n            filtersBody.appendChild(emptyMsg);\n          } else {\n            const schemaObj = merged.schema as Record<\n              string,\n              {\n                label?: string;\n                operators?: string[];\n                kind?: string;\n                defaults?: Record<string, unknown>;\n              }\n            >;\n            let parsedFilters: Record<string, Record<string, unknown>> = {};\n            try {\n              parsedFilters = parseFilters(\n                merged as any,\n                entry.query as any,\n              ) as Record<string, Record<string, unknown>>;\n            } catch {\n              /* ignore parse */\n            }\n            const fieldNames = Object.keys(schemaObj).sort();\n            let anyActive = false;\n            for (const f of fieldNames) {\n              const state = parsedFilters[f];\n              if (state != null && Object.keys(state).length > 0) {\n                anyActive = true;\n                break;\n              }\n            }\n            if (anyActive) {\n              const clearAllBtn = document.createElement('button');\n              clearAllBtn.textContent = 'Clear All Filters';\n              styleMiniButton(clearAllBtn, '#d73a49');\n              /** Clear all filters across every field. */\n              function onClearAllFilters(): void {\n                try {\n                  const blank: Record<string, Record<string, unknown>> = {};\n                  for (const fname of fieldNames) {\n                    blank[fname] = {};\n                  }\n                  context.navigate({\n                    filters: blank,\n                    filterSchema: merged,\n                    batch: true,\n                  });\n                } catch {\n                  /* ignore */\n                }\n              }\n              clearAllBtn.addEventListener('click', onClearAllFilters);\n              filterActionsWrap.appendChild(clearAllBtn);\n            }\n            for (const field of fieldNames) {\n              const defLike = schemaObj[field];\n              const fieldBox = document.createElement('div');\n              const fbStyle = fieldBox.style;\n              fbStyle.border = '1px solid #e2e2e2';\n              fbStyle.borderRadius = '4px';\n              fbStyle.padding = '4px 6px';\n              fbStyle.background = '#f9fafb';\n              fbStyle.display = 'flex';\n              fbStyle.flexDirection = 'column';\n              fbStyle.gap = '3px';\n              const fieldHead = document.createElement('div');\n              const headStyle2 = fieldHead.style;\n              headStyle2.display = 'flex';\n              headStyle2.justifyContent = 'space-between';\n              headStyle2.alignItems = 'center';\n              const labelEl = document.createElement('strong');\n              let labelText = field;\n              if (defLike?.label != null) {\n                labelText = String(defLike.label);\n              }\n              labelEl.textContent = labelText;\n              labelEl.style.fontSize = '11px';\n              const headActions = document.createElement('div');\n              headActions.style.display = 'flex';\n              headActions.style.gap = '4px';\n              const clearFieldBtn = document.createElement('button');\n              clearFieldBtn.textContent = 'clear';\n              styleMiniButton(clearFieldBtn, '#6a737d');\n              /** Clear only this field's filters. */\n              function onClearField(): void {\n                try {\n                  const next = { ...parsedFilters };\n                  next[field] = {};\n                  context.navigate({\n                    filters: next,\n                    filterSchema: merged,\n                    batch: true,\n                  });\n                } catch {\n                  /* ignore */\n                }\n              }\n              clearFieldBtn.addEventListener('click', onClearField);\n              headActions.appendChild(clearFieldBtn);\n              fieldHead.appendChild(labelEl);\n              fieldHead.appendChild(headActions);\n              fieldBox.appendChild(fieldHead);\n              const opsWrap = document.createElement('div');\n              const opsWrapStyle = opsWrap.style;\n              opsWrapStyle.display = 'flex';\n              opsWrapStyle.flexWrap = 'wrap';\n              opsWrapStyle.gap = '4px';\n              const present = parsedFilters[field] ?? {};\n              const defaults = defLike?.defaults ?? {};\n              let allowed: string[] = [];\n              if (Array.isArray(defLike?.operators)) {\n                allowed = defLike.operators.slice();\n              } else if (defLike?.kind === 'number') {\n                allowed = [\n                  'eq',\n                  'neq',\n                  'gt',\n                  'gte',\n                  'lt',\n                  'lte',\n                  'in',\n                  'nin',\n                  'between',\n                  'exists',\n                ];\n              } else {\n                allowed = [\n                  'eq',\n                  'neq',\n                  'in',\n                  'nin',\n                  'between',\n                  'exists',\n                  'contains',\n                  'starts',\n                  'ends',\n                  'regex',\n                ];\n              }\n              for (const op of allowed) {\n                const span = document.createElement('span');\n                const spanStyle = span.style;\n                spanStyle.display = 'inline-flex';\n                spanStyle.alignItems = 'center';\n                spanStyle.gap = '2px';\n                spanStyle.padding = '1px 5px';\n                spanStyle.borderRadius = '12px';\n                spanStyle.border = '1px solid #d0d7de';\n                spanStyle.fontSize = '10px';\n                const activeVal = present[op];\n                if (activeVal != null) {\n                  spanStyle.background = '#e6ffed';\n                  spanStyle.borderColor = '#34d058';\n                  const opText = document.createElement('span');\n                  opText.textContent = op;\n                  span.appendChild(opText);\n                  const closeBtnOp = document.createElement('button');\n                  closeBtnOp.textContent = '×';\n                  stylePillClose(closeBtnOp);\n                  /** Remove this operator from the field filter. */\n                  function onRemoveOperator(): void {\n                    try {\n                      const next = { ...parsedFilters } as Record<\n                        string,\n                        Record<string, unknown>\n                      >;\n                      const existing: Record<string, unknown> =\n                        next[field] ?? {};\n                      const copyField: Record<string, unknown> = {};\n                      for (const k of Object.keys(existing)) {\n                        if (k !== op) {\n                          copyField[k] = existing[k];\n                        }\n                      }\n                      next[field] = copyField;\n                      context.navigate({\n                        filters: next,\n                        filterSchema: merged,\n                        batch: true,\n                      });\n                    } catch {\n                      /* ignore */\n                    }\n                  }\n                  closeBtnOp.addEventListener('click', onRemoveOperator);\n                  span.appendChild(closeBtnOp);\n                } else {\n                  const isDefault = Object.prototype.hasOwnProperty.call(\n                    defaults,\n                    op,\n                  );\n                  if (isDefault) {\n                    span.textContent = `${op} (default)`;\n                    spanStyle.fontStyle = 'italic';\n                    span.title = 'Default value omitted in URL';\n                  } else {\n                    span.textContent = op;\n                  }\n                  spanStyle.opacity = '0.55';\n                }\n                opsWrap.appendChild(span);\n              }\n              fieldBox.appendChild(opsWrap);\n              filtersBody.appendChild(fieldBox);\n            }\n          }\n        } catch {\n          /* ignore filters tab */\n        }\n      } catch {\n        /* ignore render */\n      }\n    }\n\n    render();\n    unsubscribe = context.subscribe(render);\n    /**\n     * Switch visible tab between Route and Filters views.\n     * @param name Target tab identifier.\n     */\n    function activateTab(name: 'route' | 'filters'): void {\n      if (name === 'route') {\n        routeTabContent.style.display = 'flex';\n        filtersTabContent.style.display = 'none';\n        routeTabButton.dataset.active = 'true';\n        filtersTabButton.dataset.active = 'false';\n        routeTabButton.style.fontWeight = '600';\n        filtersTabButton.style.fontWeight = 'normal';\n      } else {\n        routeTabContent.style.display = 'none';\n        filtersTabContent.style.display = 'flex';\n        filtersTabButton.dataset.active = 'true';\n        routeTabButton.dataset.active = 'false';\n        filtersTabButton.style.fontWeight = '600';\n        routeTabButton.style.fontWeight = 'normal';\n      }\n    }\n    /** Activate route tab. */\n    function onRouteTabClick(): void {\n      activateTab('route');\n    }\n    /** Activate filters tab. */\n    function onFiltersTabClick(): void {\n      activateTab('filters');\n    }\n    routeTabButton.addEventListener('click', onRouteTabClick);\n    filtersTabButton.addEventListener('click', onFiltersTabClick);\n    /** Keydown handler for shortcut toggle. */\n    function onKeydown(ev: KeyboardEvent): void {\n      try {\n        const parts: string[] = [];\n        if (ev.altKey) parts.push('Alt');\n        if (ev.shiftKey) parts.push('Shift');\n        if (ev.ctrlKey) parts.push('Ctrl');\n        let keyPart = ev.key;\n        if (ev.key.length === 1) {\n          keyPart = ev.key.toUpperCase();\n        }\n        parts.push(keyPart);\n        const combo = parts.join('+');\n        const normShortcut = shortcut.replace(/\\s+/g, '');\n        if (combo === normShortcut) {\n          if (host.style.display === 'none') {\n            host.style.display = 'block';\n          } else {\n            host.style.display = 'none';\n          }\n        }\n      } catch {\n        /* ignore shortcut */\n      }\n    }\n    window.addEventListener('keydown', onKeydown);\n    return recordNavEvent;\n  } catch {\n    return function silentRecorder() {\n      /* ignore errors mounting */\n    };\n  }\n}\n\n// ---------------------------------------------------------------------------\n// Public API\n// ---------------------------------------------------------------------------\n\n/**\n * Initialize router devtools based on provided options.\n * Optionally exposes context globally, mounts UI panel, and returns a nav event handler.\n * The returned function should be invoked by router code whenever a navigation meta event occurs.\n * @param params.context The routing context instance.\n * @param params.options Devtools configuration options.\n * @returns Function receiving a NavEventMetaPublic to feed the timeline (noop if panel disabled).\n */\nexport function initRouterDevtools(params: {\n  context: RoutingContextType<any>;\n  options: InitRouterDevtoolsOptions;\n}): (meta: NavEventMetaPublic) => void {\n  const { context, options } = params;\n  const { global, panel, shortcut } = options;\n\n  // Global window exposure\n  if (global) {\n    try {\n      if (typeof window !== 'undefined') {\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n        (window as any).__PLUMILE_ROUTER__ = {\n          get: context.get,\n          subscribe: context.subscribe,\n        };\n      }\n    } catch {\n      /* ignore */\n    }\n  }\n\n  /** No-op recorder used when panel disabled. */\n  function noopNavRecorder(): void {\n    /* noop */\n  }\n  let navRecorder: (m: NavEventMetaPublic) => void = noopNavRecorder;\n  if (panel) {\n    navRecorder = mountDevtoolsPanel({\n      context,\n      shortcut,\n      verbose: options.verbose,\n    });\n  }\n  /** Forward navigation meta event to active recorder. */\n  function handleNavEvent(meta: NavEventMetaPublic): void {\n    navRecorder(meta);\n  }\n  return handleNavEvent;\n}\n\nexport default initRouterDevtools;\n"]}
|