@plumile/router 0.1.11 → 0.1.13

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.
Files changed (127) hide show
  1. package/README.md +755 -1
  2. package/lib/esm/builder.d.ts.map +1 -1
  3. package/lib/esm/builder.js +10 -3
  4. package/lib/esm/eslint-rules/index.d.ts +2 -0
  5. package/lib/esm/eslint-rules/index.d.ts.map +1 -0
  6. package/lib/esm/eslint-rules/index.js +2 -0
  7. package/lib/esm/eslint-rules/no-direct-window-location-search.d.ts +4 -0
  8. package/lib/esm/eslint-rules/no-direct-window-location-search.d.ts.map +1 -0
  9. package/lib/esm/eslint-rules/no-direct-window-location-search.js +48 -0
  10. package/lib/esm/history/BrowserHistory.d.ts.map +1 -1
  11. package/lib/esm/history/BrowserHistory.js +4 -2
  12. package/lib/esm/index.d.ts +5 -0
  13. package/lib/esm/index.d.ts.map +1 -1
  14. package/lib/esm/index.js +6 -1
  15. package/lib/esm/routing/Link.d.ts +1 -0
  16. package/lib/esm/routing/Link.d.ts.map +1 -1
  17. package/lib/esm/routing/Link.js +35 -4
  18. package/lib/esm/routing/RouteComponentWrapper.d.ts.map +1 -1
  19. package/lib/esm/routing/RouteComponentWrapper.js +7 -2
  20. package/lib/esm/routing/createRouter.d.ts +8 -1
  21. package/lib/esm/routing/createRouter.d.ts.map +1 -1
  22. package/lib/esm/routing/createRouter.js +540 -17
  23. package/lib/esm/routing/devtools.d.ts +20 -0
  24. package/lib/esm/routing/devtools.d.ts.map +1 -0
  25. package/lib/esm/routing/devtools.js +678 -0
  26. package/lib/esm/routing/filters.d.ts +97 -0
  27. package/lib/esm/routing/filters.d.ts.map +1 -0
  28. package/lib/esm/routing/filters.js +557 -0
  29. package/lib/esm/routing/index.d.ts +10 -0
  30. package/lib/esm/routing/index.d.ts.map +1 -1
  31. package/lib/esm/routing/index.js +11 -1
  32. package/lib/esm/routing/useActiveFilters.d.ts +9 -0
  33. package/lib/esm/routing/useActiveFilters.d.ts.map +1 -0
  34. package/lib/esm/routing/useActiveFilters.js +38 -0
  35. package/lib/esm/routing/useFilterState.d.ts +10 -0
  36. package/lib/esm/routing/useFilterState.d.ts.map +1 -0
  37. package/lib/esm/routing/useFilterState.js +14 -0
  38. package/lib/esm/routing/useNavigate.d.ts +13 -0
  39. package/lib/esm/routing/useNavigate.d.ts.map +1 -0
  40. package/lib/esm/routing/useNavigate.js +11 -0
  41. package/lib/esm/routing/useNavigateWithQuery.d.ts +15 -0
  42. package/lib/esm/routing/useNavigateWithQuery.d.ts.map +1 -0
  43. package/lib/esm/routing/useNavigateWithQuery.js +95 -0
  44. package/lib/esm/routing/useQuery.d.ts +2 -0
  45. package/lib/esm/routing/useQuery.d.ts.map +1 -0
  46. package/lib/esm/routing/useQuery.js +9 -0
  47. package/lib/esm/routing/useQueryObject.d.ts +18 -0
  48. package/lib/esm/routing/useQueryObject.d.ts.map +1 -0
  49. package/lib/esm/routing/useQueryObject.js +107 -0
  50. package/lib/esm/routing/useQueryState.d.ts +13 -0
  51. package/lib/esm/routing/useQueryState.d.ts.map +1 -0
  52. package/lib/esm/routing/useQueryState.js +80 -0
  53. package/lib/esm/routing/useStableRefEquality.d.ts +5 -0
  54. package/lib/esm/routing/useStableRefEquality.d.ts.map +1 -0
  55. package/lib/esm/routing/useStableRefEquality.js +47 -0
  56. package/lib/esm/routing/useTypedQuery.d.ts +2 -0
  57. package/lib/esm/routing/useTypedQuery.d.ts.map +1 -0
  58. package/lib/esm/routing/useTypedQuery.js +36 -0
  59. package/lib/esm/tools/buildSearch.d.ts +12 -0
  60. package/lib/esm/tools/buildSearch.d.ts.map +1 -0
  61. package/lib/esm/tools/buildSearch.js +264 -0
  62. package/lib/esm/tools/query-dsl.d.ts +28 -0
  63. package/lib/esm/tools/query-dsl.d.ts.map +1 -0
  64. package/lib/esm/tools/query-dsl.js +250 -0
  65. package/lib/esm/tools/query.d.ts +2 -0
  66. package/lib/esm/tools/query.d.ts.map +1 -0
  67. package/lib/esm/tools/query.js +43 -0
  68. package/lib/esm/tools.d.ts +2 -2
  69. package/lib/esm/tools.d.ts.map +1 -1
  70. package/lib/esm/tools.js +3 -2
  71. package/lib/esm/type-tests/query-infer.test-d.d.ts +2 -0
  72. package/lib/esm/type-tests/query-infer.test-d.d.ts.map +1 -0
  73. package/lib/esm/type-tests/query-infer.test-d.js +49 -0
  74. package/lib/esm/types.d.ts +47 -4
  75. package/lib/esm/types.d.ts.map +1 -1
  76. package/lib/esm/types.js +1 -1
  77. package/lib/tsconfig.esm.tsbuildinfo +1 -1
  78. package/lib/types/builder.d.ts.map +1 -1
  79. package/lib/types/eslint-rules/index.d.ts +2 -0
  80. package/lib/types/eslint-rules/index.d.ts.map +1 -0
  81. package/lib/types/eslint-rules/no-direct-window-location-search.d.ts +4 -0
  82. package/lib/types/eslint-rules/no-direct-window-location-search.d.ts.map +1 -0
  83. package/lib/types/history/BrowserHistory.d.ts.map +1 -1
  84. package/lib/types/index.d.ts +5 -0
  85. package/lib/types/index.d.ts.map +1 -1
  86. package/lib/types/routing/Link.d.ts +1 -0
  87. package/lib/types/routing/Link.d.ts.map +1 -1
  88. package/lib/types/routing/RouteComponentWrapper.d.ts.map +1 -1
  89. package/lib/types/routing/createRouter.d.ts +8 -1
  90. package/lib/types/routing/createRouter.d.ts.map +1 -1
  91. package/lib/types/routing/devtools.d.ts +20 -0
  92. package/lib/types/routing/devtools.d.ts.map +1 -0
  93. package/lib/types/routing/filters.d.ts +97 -0
  94. package/lib/types/routing/filters.d.ts.map +1 -0
  95. package/lib/types/routing/index.d.ts +10 -0
  96. package/lib/types/routing/index.d.ts.map +1 -1
  97. package/lib/types/routing/useActiveFilters.d.ts +9 -0
  98. package/lib/types/routing/useActiveFilters.d.ts.map +1 -0
  99. package/lib/types/routing/useFilterState.d.ts +10 -0
  100. package/lib/types/routing/useFilterState.d.ts.map +1 -0
  101. package/lib/types/routing/useNavigate.d.ts +13 -0
  102. package/lib/types/routing/useNavigate.d.ts.map +1 -0
  103. package/lib/types/routing/useNavigateWithQuery.d.ts +15 -0
  104. package/lib/types/routing/useNavigateWithQuery.d.ts.map +1 -0
  105. package/lib/types/routing/useQuery.d.ts +2 -0
  106. package/lib/types/routing/useQuery.d.ts.map +1 -0
  107. package/lib/types/routing/useQueryObject.d.ts +18 -0
  108. package/lib/types/routing/useQueryObject.d.ts.map +1 -0
  109. package/lib/types/routing/useQueryState.d.ts +13 -0
  110. package/lib/types/routing/useQueryState.d.ts.map +1 -0
  111. package/lib/types/routing/useStableRefEquality.d.ts +5 -0
  112. package/lib/types/routing/useStableRefEquality.d.ts.map +1 -0
  113. package/lib/types/routing/useTypedQuery.d.ts +2 -0
  114. package/lib/types/routing/useTypedQuery.d.ts.map +1 -0
  115. package/lib/types/tools/buildSearch.d.ts +12 -0
  116. package/lib/types/tools/buildSearch.d.ts.map +1 -0
  117. package/lib/types/tools/query-dsl.d.ts +28 -0
  118. package/lib/types/tools/query-dsl.d.ts.map +1 -0
  119. package/lib/types/tools/query.d.ts +2 -0
  120. package/lib/types/tools/query.d.ts.map +1 -0
  121. package/lib/types/tools.d.ts +2 -2
  122. package/lib/types/tools.d.ts.map +1 -1
  123. package/lib/types/type-tests/query-infer.test-d.d.ts +2 -0
  124. package/lib/types/type-tests/query-infer.test-d.d.ts.map +1 -0
  125. package/lib/types/types.d.ts +47 -4
  126. package/lib/types/types.d.ts.map +1 -1
  127. package/package.json +7 -7
@@ -0,0 +1,557 @@
1
+ import { useCallback, useContext, useMemo, useRef } from 'react';
2
+ import RoutingContext from './RoutingContext.js';
3
+ export function numberFilter(def = {}) {
4
+ return { __filter: true, kind: 'number', ...def };
5
+ }
6
+ export function stringFilter(def = {}) {
7
+ return { __filter: true, kind: 'string', ...def };
8
+ }
9
+ export function defineFilters(schema) {
10
+ return { __defined: true, schema };
11
+ }
12
+ const NUMERIC_DEFAULT_OPS = [
13
+ 'eq',
14
+ 'neq',
15
+ 'gt',
16
+ 'gte',
17
+ 'lt',
18
+ 'lte',
19
+ 'in',
20
+ 'nin',
21
+ 'between',
22
+ 'exists',
23
+ ];
24
+ const STRING_DEFAULT_OPS = [
25
+ 'eq',
26
+ 'neq',
27
+ 'in',
28
+ 'nin',
29
+ 'between',
30
+ 'exists',
31
+ 'contains',
32
+ 'starts',
33
+ 'ends',
34
+ 'regex',
35
+ ];
36
+ function allowedOperators(def) {
37
+ if (def.operators != null) {
38
+ return def.operators;
39
+ }
40
+ if (def.kind === 'number') {
41
+ return NUMERIC_DEFAULT_OPS;
42
+ }
43
+ return STRING_DEFAULT_OPS;
44
+ }
45
+ function parsePrimitive(kind, raw) {
46
+ if (kind === 'string') {
47
+ return raw;
48
+ }
49
+ const n = Number(raw);
50
+ if (Number.isNaN(n)) {
51
+ return undefined;
52
+ }
53
+ return n;
54
+ }
55
+ function serializePrimitive(v) {
56
+ if (typeof v === 'boolean') {
57
+ if (v) {
58
+ return '1';
59
+ }
60
+ return '0';
61
+ }
62
+ return String(v);
63
+ }
64
+ const PARSE_CACHE = new Map();
65
+ const SCHEMA_IDS = new WeakMap();
66
+ let NEXT_SCHEMA_ID = 0;
67
+ function stableFiltersSignature(obj, schemaId) {
68
+ const parts = ['S', String(schemaId)];
69
+ const fields = Object.keys(obj).sort();
70
+ for (const f of fields) {
71
+ parts.push('F', f);
72
+ const fieldObj = obj[f];
73
+ const ops = Object.keys(fieldObj ?? {}).sort();
74
+ for (const op of ops) {
75
+ const desc = Object.getOwnPropertyDescriptor(fieldObj, op);
76
+ if (desc == null) {
77
+ continue;
78
+ }
79
+ const val = desc.value;
80
+ parts.push('O', op);
81
+ if (typeof val === 'boolean') {
82
+ if (val) {
83
+ parts.push('T');
84
+ }
85
+ else {
86
+ parts.push('F');
87
+ }
88
+ continue;
89
+ }
90
+ if (Array.isArray(val)) {
91
+ if (val.length > 0 && Array.isArray(val[0])) {
92
+ parts.push('B');
93
+ for (const t of val) {
94
+ if (Array.isArray(t)) {
95
+ parts.push(String(t[0]), String(t[1]));
96
+ }
97
+ }
98
+ }
99
+ else {
100
+ parts.push('A');
101
+ for (const v of val) {
102
+ parts.push(String(v));
103
+ }
104
+ }
105
+ continue;
106
+ }
107
+ if (val === undefined) {
108
+ parts.push('U');
109
+ continue;
110
+ }
111
+ if (typeof val === 'string' || typeof val === 'number') {
112
+ parts.push('V', String(val));
113
+ continue;
114
+ }
115
+ parts.push('V', '[obj]');
116
+ }
117
+ }
118
+ return parts.join('\u001f');
119
+ }
120
+ export function parseFilters(defined, rawQuery) {
121
+ const schemaId = (() => {
122
+ const wm = SCHEMA_IDS;
123
+ let id = wm.get(defined);
124
+ if (id == null) {
125
+ id = NEXT_SCHEMA_ID += 1;
126
+ wm.set(defined, id);
127
+ }
128
+ return id;
129
+ })();
130
+ const { schema } = defined;
131
+ const out = {};
132
+ for (const entry of Object.entries(schema)) {
133
+ const field = entry[0];
134
+ const def = entry[1];
135
+ const fieldState = {};
136
+ const ops = allowedOperators(def);
137
+ for (const op of ops) {
138
+ const key = `${field}.${op}`;
139
+ const rawVal = rawQuery[key];
140
+ if (rawVal === undefined) {
141
+ }
142
+ else {
143
+ let values;
144
+ if (Array.isArray(rawVal)) {
145
+ values = rawVal;
146
+ }
147
+ else {
148
+ values = [rawVal];
149
+ }
150
+ if (op === 'exists') {
151
+ const last = values[values.length - 1];
152
+ if (last !== undefined && last !== '') {
153
+ const norm = last.toLowerCase();
154
+ if (norm === '0' || norm === 'false' || norm === 'no') {
155
+ fieldState.exists = false;
156
+ }
157
+ else {
158
+ fieldState.exists = true;
159
+ }
160
+ }
161
+ }
162
+ else if (op === 'between') {
163
+ const tuples = [];
164
+ let i = 0;
165
+ while (i + 1 < values.length) {
166
+ const rawA = values[i];
167
+ const rawB = values[i + 1];
168
+ if (rawA === undefined || rawB === undefined) {
169
+ break;
170
+ }
171
+ const a = parsePrimitive(def.kind, rawA);
172
+ const b = parsePrimitive(def.kind, rawB);
173
+ if (a !== undefined && b !== undefined) {
174
+ let x = a;
175
+ let y = b;
176
+ if (x > y) {
177
+ const tmp = x;
178
+ x = y;
179
+ y = tmp;
180
+ }
181
+ tuples.push([x, y]);
182
+ }
183
+ i += 2;
184
+ }
185
+ if (tuples.length > 0) {
186
+ tuples.sort((a, b) => {
187
+ const ax = a[0];
188
+ const bx = b[0];
189
+ if (ax < bx)
190
+ return -1;
191
+ if (ax > bx)
192
+ return 1;
193
+ const ay = a[1];
194
+ const by = b[1];
195
+ if (ay < by)
196
+ return -1;
197
+ if (ay > by)
198
+ return 1;
199
+ return 0;
200
+ });
201
+ fieldState.between = tuples;
202
+ }
203
+ }
204
+ else {
205
+ const arr = [];
206
+ for (const rawItem of values) {
207
+ let value = rawItem;
208
+ if (def.kind === 'string' && def.normalize != null) {
209
+ try {
210
+ value = def.normalize(rawItem);
211
+ }
212
+ catch {
213
+ value = rawItem;
214
+ }
215
+ if (value === '') {
216
+ continue;
217
+ }
218
+ }
219
+ const parsed = parsePrimitive(def.kind, value);
220
+ if (parsed !== undefined && parsed !== '') {
221
+ arr.push(parsed);
222
+ }
223
+ }
224
+ if (arr.length > 0) {
225
+ if (def.kind === 'number') {
226
+ const nums = arr;
227
+ nums.sort((a, b) => {
228
+ return a - b;
229
+ });
230
+ if (def.dedupe) {
231
+ const ded = [];
232
+ let last;
233
+ for (const n of nums) {
234
+ if (n !== last) {
235
+ ded.push(n);
236
+ last = n;
237
+ }
238
+ }
239
+ fieldState[op] = ded;
240
+ }
241
+ else {
242
+ fieldState[op] = nums;
243
+ }
244
+ }
245
+ else if (def.dedupe) {
246
+ const seen = new Set();
247
+ const res = [];
248
+ for (const s of arr) {
249
+ if (!seen.has(s)) {
250
+ seen.add(s);
251
+ res.push(s);
252
+ }
253
+ }
254
+ fieldState[op] = res;
255
+ }
256
+ else {
257
+ fieldState[op] = arr;
258
+ }
259
+ }
260
+ }
261
+ }
262
+ }
263
+ out[field] = fieldState;
264
+ }
265
+ const signature = stableFiltersSignature(out, schemaId);
266
+ const existing = PARSE_CACHE.get(signature);
267
+ if (existing !== undefined) {
268
+ return existing;
269
+ }
270
+ PARSE_CACHE.set(signature, out);
271
+ return out;
272
+ }
273
+ export const OPERATOR_PRIORITY = [
274
+ 'eq',
275
+ 'between',
276
+ 'gt',
277
+ 'gte',
278
+ 'lt',
279
+ 'lte',
280
+ 'in',
281
+ 'nin',
282
+ 'exists',
283
+ 'contains',
284
+ 'starts',
285
+ 'ends',
286
+ 'regex',
287
+ ];
288
+ function arraysEqual(a, b) {
289
+ if (a.length !== b.length) {
290
+ return false;
291
+ }
292
+ let i = 0;
293
+ while (i < a.length) {
294
+ const av = a[i];
295
+ const bv = b[i];
296
+ if (Array.isArray(av) && Array.isArray(bv)) {
297
+ if (!arraysEqual(av, bv)) {
298
+ return false;
299
+ }
300
+ }
301
+ else if (av !== bv) {
302
+ return false;
303
+ }
304
+ i += 1;
305
+ }
306
+ return true;
307
+ }
308
+ export function buildFiltersQuery(defined, currentQuery, next) {
309
+ const { schema } = defined;
310
+ const out = {};
311
+ const fields = Object.keys(schema);
312
+ for (const kv of Object.entries(currentQuery)) {
313
+ const k = kv[0];
314
+ const v = kv[1];
315
+ let belongs = false;
316
+ for (const f of fields) {
317
+ if (k.startsWith(`${f}.`)) {
318
+ belongs = true;
319
+ break;
320
+ }
321
+ }
322
+ if (!belongs) {
323
+ out[k] = v;
324
+ }
325
+ }
326
+ for (const entry of Object.entries(schema)) {
327
+ const field = entry[0];
328
+ const def = entry[1];
329
+ const opsUnsorted = allowedOperators(def);
330
+ const ops = [...opsUnsorted];
331
+ ops.sort((a, b) => {
332
+ const ia = OPERATOR_PRIORITY.indexOf(a);
333
+ const ib = OPERATOR_PRIORITY.indexOf(b);
334
+ let safeA = ia;
335
+ if (safeA === -1) {
336
+ safeA = 999;
337
+ }
338
+ let safeB = ib;
339
+ if (safeB === -1) {
340
+ safeB = 999;
341
+ }
342
+ if (safeA < safeB)
343
+ return -1;
344
+ if (safeA > safeB)
345
+ return 1;
346
+ if (a < b)
347
+ return -1;
348
+ if (a > b)
349
+ return 1;
350
+ return 0;
351
+ });
352
+ const recordNext = next;
353
+ const state = recordNext[field] ?? {};
354
+ for (const op of ops) {
355
+ if (Object.prototype.hasOwnProperty.call(state, op)) {
356
+ const value = state[op];
357
+ if (value != null) {
358
+ const defaults = def.defaults;
359
+ if (defaults != null &&
360
+ Object.prototype.hasOwnProperty.call(defaults, op)) {
361
+ const dv = defaults[op];
362
+ let equal = false;
363
+ if (Array.isArray(dv) && Array.isArray(value)) {
364
+ equal = arraysEqual(dv, value);
365
+ }
366
+ else if (typeof dv === 'boolean' && typeof value === 'boolean') {
367
+ equal = dv === value;
368
+ }
369
+ if (equal) {
370
+ continue;
371
+ }
372
+ }
373
+ const key = `${field}.${op}`;
374
+ if (op === 'exists' && typeof value === 'boolean') {
375
+ out[key] = serializePrimitive(value);
376
+ }
377
+ else if (op === 'between') {
378
+ const ranges = value;
379
+ for (const range of ranges) {
380
+ const aVal = range[0];
381
+ const bVal = range[1];
382
+ const pair = [
383
+ serializePrimitive(aVal),
384
+ serializePrimitive(bVal),
385
+ ];
386
+ const existing = out[key];
387
+ if (existing == null) {
388
+ out[key] = pair;
389
+ }
390
+ else if (Array.isArray(existing)) {
391
+ out[key] = [...existing, ...pair];
392
+ }
393
+ else {
394
+ out[key] = [existing, ...pair];
395
+ }
396
+ }
397
+ }
398
+ else {
399
+ const arr = value;
400
+ for (const val of arr) {
401
+ const sv = serializePrimitive(val);
402
+ const existing = out[key];
403
+ if (existing == null) {
404
+ out[key] = sv;
405
+ }
406
+ else if (Array.isArray(existing)) {
407
+ existing.push(sv);
408
+ }
409
+ else {
410
+ out[key] = [existing, sv];
411
+ }
412
+ }
413
+ }
414
+ }
415
+ }
416
+ }
417
+ }
418
+ return out;
419
+ }
420
+ export function useFilters(defined) {
421
+ const ctx = useContext(RoutingContext);
422
+ const entry = ctx?.get();
423
+ const activeDefined = useMemo(() => {
424
+ if (defined != null) {
425
+ return defined;
426
+ }
427
+ if (ctx == null) {
428
+ return undefined;
429
+ }
430
+ return ctx.currentMergedFilterSchema();
431
+ }, [defined, ctx]);
432
+ const filters = useMemo(() => {
433
+ if (activeDefined == null || entry == null) {
434
+ return {};
435
+ }
436
+ return parseFilters(activeDefined, entry.query);
437
+ }, [activeDefined, entry]);
438
+ const schemaRef = useRef(activeDefined);
439
+ schemaRef.current = activeDefined;
440
+ const navigateFilters = useCallback((nextFilters, opts) => {
441
+ if (ctx == null) {
442
+ return;
443
+ }
444
+ let schemaNow = schemaRef.current;
445
+ schemaNow ??= ctx.currentMergedFilterSchema();
446
+ if (schemaNow == null) {
447
+ return;
448
+ }
449
+ ctx.navigate({
450
+ filters: nextFilters,
451
+ filterSchema: schemaNow,
452
+ replace: opts?.replace,
453
+ batch: opts?.batch,
454
+ });
455
+ }, [ctx]);
456
+ const set = useCallback((next, navOpts) => {
457
+ const prev = filters;
458
+ let resolved = prev;
459
+ if (typeof next === 'function') {
460
+ try {
461
+ resolved = next(prev);
462
+ }
463
+ catch {
464
+ resolved = prev;
465
+ }
466
+ }
467
+ else {
468
+ resolved = next;
469
+ }
470
+ const internal = navOpts?.internalBatch !== false;
471
+ if (!internal) {
472
+ let replaceFlag;
473
+ let batchFlag;
474
+ if (navOpts.replace === true) {
475
+ replaceFlag = true;
476
+ }
477
+ if (navOpts.batch === true) {
478
+ batchFlag = true;
479
+ }
480
+ navigateFilters(resolved, { replace: replaceFlag, batch: batchFlag });
481
+ return;
482
+ }
483
+ const self = set;
484
+ self.__pending ??= { scheduled: false };
485
+ const pending = self.__pending;
486
+ pending.filters = resolved;
487
+ if (navOpts?.replace) {
488
+ pending.replace = true;
489
+ }
490
+ if (navOpts?.batch) {
491
+ pending.batch = true;
492
+ }
493
+ if (!pending.scheduled) {
494
+ pending.scheduled = true;
495
+ queueMicrotask(() => {
496
+ pending.scheduled = false;
497
+ if (pending.filters != null) {
498
+ navigateFilters(pending.filters, {
499
+ replace: pending.replace,
500
+ batch: pending.batch,
501
+ });
502
+ }
503
+ pending.filters = undefined;
504
+ pending.replace = undefined;
505
+ pending.batch = undefined;
506
+ });
507
+ }
508
+ }, [filters, navigateFilters]);
509
+ const patch = useCallback((delta, navOpts) => {
510
+ set((prev) => {
511
+ const merged = { ...prev };
512
+ for (const [field, ops] of Object.entries(delta)) {
513
+ let cur = merged[field];
514
+ if (cur == null || typeof cur !== 'object') {
515
+ cur = {};
516
+ }
517
+ else {
518
+ cur = { ...cur };
519
+ }
520
+ if (ops != null && typeof ops === 'object') {
521
+ for (const [op, val] of Object.entries(ops)) {
522
+ cur[op] = val;
523
+ }
524
+ }
525
+ merged[field] = cur;
526
+ }
527
+ return merged;
528
+ }, navOpts);
529
+ }, [set]);
530
+ const clear = useCallback((fields, navOpts) => {
531
+ if (fields == null) {
532
+ set((prev) => {
533
+ const next = {};
534
+ for (const k of Object.keys(prev)) {
535
+ next[k] = {};
536
+ }
537
+ return next;
538
+ }, navOpts);
539
+ }
540
+ else {
541
+ const emptyEntries = fields.map((f) => {
542
+ return [f, {}];
543
+ });
544
+ const delta = Object.fromEntries(emptyEntries);
545
+ patch(delta, navOpts);
546
+ }
547
+ }, [patch, set]);
548
+ const helpers = {
549
+ set,
550
+ patch,
551
+ clear,
552
+ };
553
+ return [filters, helpers];
554
+ }
555
+ export const useQueryFilters = useFilters;
556
+ export default useFilters;
557
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filters.js","sourceRoot":"","sources":["../../../src/routing/filters.ts"],"names":[],"mappings":"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","sourcesContent":["import { useCallback, useContext, useMemo, useRef } from 'react';\nimport RoutingContext from './RoutingContext.js';\n\n// ------------------------------------------------------\n// Types\n// ------------------------------------------------------\n\nexport type FilterOperator =\n  | 'eq'\n  | 'neq'\n  | 'gt'\n  | 'gte'\n  | 'lt'\n  | 'lte'\n  | 'in'\n  | 'nin'\n  | 'between'\n  | 'exists'\n  | 'contains'\n  | 'starts'\n  | 'ends'\n  | 'regex';\n\nexport type FilterValueKind = 'string' | 'number';\n\nexport interface FilterFieldDefinition<\n  Kind extends FilterValueKind = FilterValueKind,\n> {\n  __filter: true;\n  kind: Kind;\n  operators?: FilterOperator[];\n  dedupe?: boolean;\n  label?: string;\n  normalize?: (raw: string) => string; // string only\n  defaults?: Partial<\n    Kind extends 'number'\n      ? {\n          eq: number[];\n          neq: number[];\n          gt: number[];\n          gte: number[];\n          lt: number[];\n          lte: number[];\n          in: number[];\n          nin: number[];\n          between: [number, number][];\n          exists: boolean;\n        }\n      : {\n          eq: string[];\n          neq: string[];\n          in: string[];\n          nin: string[];\n          between: [string, string][];\n          exists: boolean;\n          contains: string[];\n          starts: string[];\n          ends: string[];\n          regex: string[];\n        }\n  >;\n}\n\nexport type FilterSchema = Record<string, FilterFieldDefinition>;\nexport interface DefinedFilterSchema<F extends FilterSchema> {\n  __defined: true;\n  schema: F;\n}\n\n/** Create a number filter field definition */\nexport function numberFilter(\n  def: Partial<Omit<FilterFieldDefinition<'number'>, '__filter' | 'kind'>> = {},\n): FilterFieldDefinition<'number'> {\n  return { __filter: true, kind: 'number', ...def };\n}\n/** Create a string filter field definition */\nexport function stringFilter(\n  def: Partial<Omit<FilterFieldDefinition<'string'>, '__filter' | 'kind'>> = {},\n): FilterFieldDefinition<'string'> {\n  return { __filter: true, kind: 'string', ...def };\n}\n/** Wrap a schema to mark it as defined for inference */\nexport function defineFilters<F extends FilterSchema>(\n  schema: F,\n): DefinedFilterSchema<F> {\n  return { __defined: true, schema };\n}\n\n// ------------------------------------------------------\n// Inference helpers\n// ------------------------------------------------------\nexport type OperatorValue<K extends FilterValueKind> = K extends 'number'\n  ? {\n      eq?: number[];\n      neq?: number[];\n      gt?: number[];\n      gte?: number[];\n      lt?: number[];\n      lte?: number[];\n      in?: number[];\n      nin?: number[];\n      between?: [number, number][];\n      exists?: boolean;\n    }\n  : {\n      eq?: string[];\n      neq?: string[];\n      in?: string[];\n      nin?: string[];\n      between?: [string, string][];\n      exists?: boolean;\n      contains?: string[];\n      starts?: string[];\n      ends?: string[];\n      regex?: string[];\n    };\nexport type InferFilters<F extends FilterSchema> = {\n  [K in keyof F]: OperatorValue<F[K]['kind']>;\n};\n\n// ------------------------------------------------------\n// Operator defaults & utilities\n// ------------------------------------------------------\nconst NUMERIC_DEFAULT_OPS: FilterOperator[] = [\n  'eq',\n  'neq',\n  'gt',\n  'gte',\n  'lt',\n  'lte',\n  'in',\n  'nin',\n  'between',\n  'exists',\n];\nconst STRING_DEFAULT_OPS: FilterOperator[] = [\n  'eq',\n  'neq',\n  'in',\n  'nin',\n  'between',\n  'exists',\n  'contains',\n  'starts',\n  'ends',\n  'regex',\n];\n\n/** Return allowed operators for a field (explicit or defaults) */\nfunction allowedOperators(def: FilterFieldDefinition): FilterOperator[] {\n  if (def.operators != null) {\n    return def.operators;\n  }\n  if (def.kind === 'number') {\n    return NUMERIC_DEFAULT_OPS;\n  }\n  return STRING_DEFAULT_OPS;\n}\n\n/** Parse a primitive according to kind */\nfunction parsePrimitive(\n  kind: FilterValueKind,\n  raw: string,\n): string | number | undefined {\n  if (kind === 'string') {\n    return raw;\n  }\n  const n = Number(raw);\n  if (Number.isNaN(n)) {\n    return undefined;\n  }\n  return n;\n}\n\n/** Serialize primitive or boolean to query string value */\nfunction serializePrimitive(v: string | number | boolean): string {\n  if (typeof v === 'boolean') {\n    if (v) {\n      return '1';\n    }\n    return '0';\n  }\n  return String(v);\n}\n\n// -------------------------- Cache utilities ------------------------------\n/** Global parse cache keyed by canonical signature */\nconst PARSE_CACHE = new Map<string, unknown>();\n/** Weak map assigning stable numeric ids per schema wrapper */\nconst SCHEMA_IDS = new WeakMap<object, number>();\nlet NEXT_SCHEMA_ID = 0;\n\n/** Build a stable canonical signature for parsed filters */\nfunction stableFiltersSignature(\n  obj: Record<string, Record<string, unknown>>,\n  schemaId: number,\n): string {\n  const parts: string[] = ['S', String(schemaId)];\n  const fields = Object.keys(obj).sort();\n  for (const f of fields) {\n    parts.push('F', f);\n    const fieldObj = obj[f];\n    const ops = Object.keys(fieldObj ?? {}).sort();\n    for (const op of ops) {\n      const desc = Object.getOwnPropertyDescriptor(fieldObj, op);\n      if (desc == null) {\n        // eslint-disable-next-line no-continue\n        continue;\n      }\n      const val: unknown = desc.value;\n      parts.push('O', op);\n      if (typeof val === 'boolean') {\n        if (val) {\n          parts.push('T');\n        } else {\n          parts.push('F');\n        }\n        // eslint-disable-next-line no-continue\n        continue;\n      }\n      if (Array.isArray(val)) {\n        if (val.length > 0 && Array.isArray(val[0])) {\n          parts.push('B');\n          for (const t of val as unknown[]) {\n            if (Array.isArray(t)) {\n              parts.push(String(t[0]), String(t[1]));\n            }\n          }\n        } else {\n          parts.push('A');\n          for (const v of val as unknown[]) {\n            parts.push(String(v));\n          }\n        }\n        // eslint-disable-next-line no-continue\n        continue;\n      }\n      if (val === undefined) {\n        parts.push('U');\n        // eslint-disable-next-line no-continue\n        continue;\n      }\n      if (typeof val === 'string' || typeof val === 'number') {\n        parts.push('V', String(val));\n        // eslint-disable-next-line no-continue\n        continue;\n      }\n      parts.push('V', '[obj]');\n    }\n  }\n  return parts.join('\\u001f');\n}\n\n// ------------------------------------------------------\n// Parsing\n// ------------------------------------------------------\n/** Parse raw query into structured filters (with identity cache) */\nexport function parseFilters<F extends DefinedFilterSchema<FilterSchema>>(\n  defined: F,\n  rawQuery: Record<string, string | string[]>,\n): InferFilters<F['schema']> {\n  // --- Caching Strategy --------------------------------------------------\n  // We rebuild the structured object every invocation (simplicity) then\n  // compute a stable canonical signature and return the previously cached\n  // object when identical. This satisfies identity expectations in tests\n  // (parseFilters(a) === parseFilters(a) for unchanged raw query) while\n  // keeping implementation straightforward. Memory usage is bounded by\n  // variation of canonical filter states encountered during app lifetime.\n  // -----------------------------------------------------------------------\n  const schemaId = (() => {\n    // Assign stable small numeric id per defined schema wrapper.\n    const wm = SCHEMA_IDS;\n    let id = wm.get(defined);\n    if (id == null) {\n      id = NEXT_SCHEMA_ID += 1;\n      wm.set(defined, id);\n    }\n    return id;\n  })();\n  // We build then hash; alternative pre-hash of raw values omitted for clarity.\n  const { schema } = defined;\n  const out: Record<string, Record<string, unknown>> = {};\n  for (const entry of Object.entries(schema)) {\n    const field = entry[0];\n    const def = entry[1];\n    const fieldState: Record<string, unknown> = {};\n    const ops = allowedOperators(def);\n    for (const op of ops) {\n      const key = `${field}.${op}`;\n      const rawVal = rawQuery[key];\n      if (rawVal === undefined) {\n        // no key\n      } else {\n        let values: string[];\n        if (Array.isArray(rawVal)) {\n          values = rawVal;\n        } else {\n          values = [rawVal];\n        }\n        if (op === 'exists') {\n          const last = values[values.length - 1];\n          if (last !== undefined && last !== '') {\n            const norm = last.toLowerCase();\n            if (norm === '0' || norm === 'false' || norm === 'no') {\n              fieldState.exists = false;\n            } else {\n              fieldState.exists = true;\n            }\n          }\n        } else if (op === 'between') {\n          const tuples: [unknown, unknown][] = [];\n          let i = 0;\n          while (i + 1 < values.length) {\n            const rawA = values[i];\n            const rawB = values[i + 1];\n            if (rawA === undefined || rawB === undefined) {\n              break;\n            }\n            const a = parsePrimitive(def.kind, rawA);\n            const b = parsePrimitive(def.kind, rawB);\n            if (a !== undefined && b !== undefined) {\n              let x = a;\n              let y = b;\n              if ((x as number) > (y as number)) {\n                const tmp = x;\n                x = y;\n                y = tmp;\n              }\n              tuples.push([x, y]);\n            }\n            i += 2;\n          }\n          if (tuples.length > 0) {\n            tuples.sort((a, b) => {\n              const ax = a[0] as number | string;\n              const bx = b[0] as number | string;\n              if (ax < bx) return -1;\n              if (ax > bx) return 1;\n              const ay = a[1] as number | string;\n              const by = b[1] as number | string;\n              if (ay < by) return -1;\n              if (ay > by) return 1;\n              return 0;\n            });\n            fieldState.between = tuples;\n          }\n        } else {\n          const arr: (string | number)[] = [];\n          for (const rawItem of values) {\n            let value = rawItem;\n            if (def.kind === 'string' && def.normalize != null) {\n              try {\n                value = def.normalize(rawItem);\n              } catch {\n                value = rawItem;\n              }\n              if (value === '') {\n                // skip empty\n                // skip empty\n                // eslint-disable-next-line no-continue\n                continue;\n              }\n            }\n            const parsed = parsePrimitive(def.kind, value);\n            if (parsed !== undefined && parsed !== '') {\n              arr.push(parsed);\n            }\n          }\n          if (arr.length > 0) {\n            if (def.kind === 'number') {\n              const nums = arr as number[];\n              nums.sort((a, b) => {\n                return a - b;\n              });\n              if (def.dedupe) {\n                const ded: number[] = [];\n                let last: number | undefined;\n                for (const n of nums) {\n                  if (n !== last) {\n                    ded.push(n);\n                    last = n;\n                  }\n                }\n                fieldState[op] = ded;\n              } else {\n                fieldState[op] = nums;\n              }\n            } else if (def.dedupe) {\n              const seen = new Set<string>();\n              const res: string[] = [];\n              for (const s of arr as string[]) {\n                if (!seen.has(s)) {\n                  seen.add(s);\n                  res.push(s);\n                }\n              }\n              fieldState[op] = res;\n            } else {\n              fieldState[op] = arr as string[];\n            }\n          }\n        }\n      }\n    }\n    out[field] = fieldState;\n  }\n  const signature = stableFiltersSignature(out, schemaId);\n  const existing = PARSE_CACHE.get(signature);\n  if (existing !== undefined) {\n    return existing as InferFilters<F['schema']>;\n  }\n  PARSE_CACHE.set(signature, out);\n  return out as InferFilters<F['schema']>;\n}\n\n// ------------------------------------------------------\n// Serialization\n// ------------------------------------------------------\nexport const OPERATOR_PRIORITY: FilterOperator[] = [\n  'eq',\n  'between',\n  'gt',\n  'gte',\n  'lt',\n  'lte',\n  'in',\n  'nin',\n  'exists',\n  'contains',\n  'starts',\n  'ends',\n  'regex',\n];\n\nexport type FiltersPatch<F> = { [K in keyof F]?: Partial<F[K]> };\n\n/** Deep (one-level) array equality with tuple recursion */\nfunction arraysEqual(a: unknown[], b: unknown[]): boolean {\n  if (a.length !== b.length) {\n    return false;\n  }\n  let i = 0;\n  while (i < a.length) {\n    const av = a[i];\n    const bv = b[i];\n    if (Array.isArray(av) && Array.isArray(bv)) {\n      if (!arraysEqual(av, bv)) {\n        return false;\n      }\n    } else if (av !== bv) {\n      return false;\n    }\n    i += 1;\n  }\n  return true;\n}\n\n/** Serialize structured filters back into flat query object */\nexport function buildFiltersQuery<F extends DefinedFilterSchema<FilterSchema>>(\n  defined: F,\n  currentQuery: Record<string, unknown>,\n  next: InferFilters<F['schema']>,\n): Record<string, unknown> {\n  const { schema } = defined;\n  const out: Record<string, unknown> = {};\n  const fields = Object.keys(schema);\n  for (const kv of Object.entries(currentQuery)) {\n    const k = kv[0];\n    const v = kv[1];\n    let belongs = false;\n    for (const f of fields) {\n      if (k.startsWith(`${f}.`)) {\n        belongs = true;\n        break;\n      }\n    }\n    if (!belongs) {\n      out[k] = v;\n    }\n  }\n  for (const entry of Object.entries(schema)) {\n    const field = entry[0];\n    const def = entry[1];\n    const opsUnsorted = allowedOperators(def);\n    const ops = [...opsUnsorted];\n    ops.sort((a, b) => {\n      const ia = OPERATOR_PRIORITY.indexOf(a);\n      const ib = OPERATOR_PRIORITY.indexOf(b);\n      let safeA = ia;\n      if (safeA === -1) {\n        safeA = 999;\n      }\n      let safeB = ib;\n      if (safeB === -1) {\n        safeB = 999;\n      }\n      if (safeA < safeB) return -1;\n      if (safeA > safeB) return 1;\n      if (a < b) return -1;\n      if (a > b) return 1;\n      return 0;\n    });\n    const recordNext = next as Record<string, Record<string, unknown>>;\n    const state = recordNext[field] ?? {};\n    for (const op of ops) {\n      if (Object.prototype.hasOwnProperty.call(state, op)) {\n        const value = state[op];\n        if (value != null) {\n          const defaults = def.defaults as Record<string, unknown> | undefined;\n          if (\n            defaults != null &&\n            Object.prototype.hasOwnProperty.call(defaults, op)\n          ) {\n            const dv = defaults[op];\n            let equal = false;\n            if (Array.isArray(dv) && Array.isArray(value)) {\n              equal = arraysEqual(dv, value as unknown[]);\n            } else if (typeof dv === 'boolean' && typeof value === 'boolean') {\n              equal = dv === value;\n            }\n            if (equal) {\n              // omit default-equivalent\n              // eslint-disable-next-line no-continue\n              continue; // omit default-equivalent\n            }\n          }\n          const key = `${field}.${op}`;\n          if (op === 'exists' && typeof value === 'boolean') {\n            out[key] = serializePrimitive(value);\n          } else if (op === 'between') {\n            const ranges = value as [unknown, unknown][];\n            for (const range of ranges) {\n              const aVal = range[0];\n              const bVal = range[1];\n              const pair: (string | number)[] = [\n                serializePrimitive(aVal as string | number | boolean),\n                serializePrimitive(bVal as string | number | boolean),\n              ];\n              const existing = out[key];\n              if (existing == null) {\n                out[key] = pair;\n              } else if (Array.isArray(existing)) {\n                out[key] = [...existing, ...pair];\n              } else {\n                out[key] = [existing, ...pair];\n              }\n            }\n          } else {\n            const arr = value as unknown[];\n            for (const val of arr) {\n              const sv = serializePrimitive(val as string | number | boolean);\n              const existing = out[key];\n              if (existing == null) {\n                out[key] = sv;\n              } else if (Array.isArray(existing)) {\n                existing.push(sv);\n              } else {\n                out[key] = [existing, sv];\n              }\n            }\n          }\n        }\n      }\n    }\n  }\n  return out;\n}\n\n// ------------------------------------------------------\n// Hook\n// ------------------------------------------------------\nexport interface UseFiltersHelpers<F> {\n  set: (\n    next: F | ((p: F) => F),\n    navOpts?: { replace?: boolean; batch?: boolean; internalBatch?: boolean },\n  ) => void;\n  patch: (\n    delta: FiltersPatch<F>,\n    navOpts?: { replace?: boolean; batch?: boolean; internalBatch?: boolean },\n  ) => void;\n  clear: (\n    fields?: (keyof F)[],\n    navOpts?: { replace?: boolean; batch?: boolean; internalBatch?: boolean },\n  ) => void;\n}\n\n/**\n * useFilters hook\n * - When a schema wrapper is provided, types remain inferred.\n * - When omitted, implicit merged route schema is used (helpers still work but record is generic).\n */\nexport function useFilters(\n  defined?: DefinedFilterSchema<FilterSchema>,\n): [\n  Record<string, Record<string, unknown>>,\n  UseFiltersHelpers<Record<string, Record<string, unknown>>>,\n] {\n  const ctx = useContext(RoutingContext);\n  const entry = ctx?.get();\n  const activeDefined = useMemo(() => {\n    if (defined != null) {\n      return defined;\n    }\n    if (ctx == null) {\n      return undefined;\n    }\n    return ctx.currentMergedFilterSchema();\n  }, [defined, ctx]);\n\n  const filters = useMemo(() => {\n    if (activeDefined == null || entry == null) {\n      return {} as Record<string, Record<string, unknown>>;\n    }\n    return parseFilters(activeDefined, entry.query);\n  }, [activeDefined, entry]);\n\n  type NavOpts = {\n    replace?: boolean;\n    batch?: boolean;\n    internalBatch?: boolean;\n  };\n  const schemaRef = useRef(activeDefined);\n  schemaRef.current = activeDefined;\n\n  const navigateFilters = useCallback(\n    (\n      nextFilters: Record<string, Record<string, unknown>>,\n      opts?: { replace?: boolean; batch?: boolean },\n    ) => {\n      if (ctx == null) {\n        return;\n      }\n      let schemaNow = schemaRef.current;\n      schemaNow ??= ctx.currentMergedFilterSchema();\n      if (schemaNow == null) {\n        return;\n      }\n      ctx.navigate({\n        filters: nextFilters,\n        filterSchema: schemaNow,\n        replace: opts?.replace,\n        batch: opts?.batch,\n      });\n    },\n    [ctx],\n  );\n\n  const set = useCallback(\n    (\n      next:\n        | Record<string, Record<string, unknown>>\n        | ((\n            p: Record<string, Record<string, unknown>>,\n          ) => Record<string, Record<string, unknown>>),\n      navOpts?: NavOpts,\n    ) => {\n      const prev = filters;\n      let resolved: Record<string, Record<string, unknown>> = prev;\n      if (typeof next === 'function') {\n        try {\n          resolved = next(prev);\n        } catch {\n          resolved = prev;\n        }\n      } else {\n        resolved = next;\n      }\n      const internal = navOpts?.internalBatch !== false;\n      if (!internal) {\n        let replaceFlag: boolean | undefined;\n        let batchFlag: boolean | undefined;\n        if (navOpts.replace === true) {\n          replaceFlag = true;\n        }\n        if (navOpts.batch === true) {\n          batchFlag = true;\n        }\n        navigateFilters(resolved, { replace: replaceFlag, batch: batchFlag });\n        return;\n      }\n      interface PendingState {\n        filters?: Record<string, Record<string, unknown>>;\n        replace?: boolean;\n        batch?: boolean;\n        scheduled: boolean;\n      }\n      const self = set as unknown as { __pending?: PendingState };\n      self.__pending ??= { scheduled: false };\n      const pending = self.__pending;\n      pending.filters = resolved;\n      if (navOpts?.replace) {\n        pending.replace = true;\n      }\n      if (navOpts?.batch) {\n        pending.batch = true;\n      }\n      if (!pending.scheduled) {\n        pending.scheduled = true;\n        queueMicrotask(() => {\n          pending.scheduled = false;\n          if (pending.filters != null) {\n            navigateFilters(pending.filters, {\n              replace: pending.replace,\n              batch: pending.batch,\n            });\n          }\n          pending.filters = undefined;\n          pending.replace = undefined;\n          pending.batch = undefined;\n        });\n      }\n    },\n    [filters, navigateFilters],\n  );\n\n  const patch = useCallback(\n    (\n      delta: FiltersPatch<Record<string, Record<string, unknown>>>,\n      navOpts?: NavOpts,\n    ) => {\n      set((prev) => {\n        const merged: Record<string, Record<string, unknown>> = { ...prev };\n        for (const [field, ops] of Object.entries(delta)) {\n          let cur: Record<string, unknown> | undefined = merged[field];\n          if (cur == null || typeof cur !== 'object') {\n            cur = {};\n          } else {\n            cur = { ...cur };\n          }\n          if (ops != null && typeof ops === 'object') {\n            for (const [op, val] of Object.entries(\n              ops as Record<string, unknown>,\n            )) {\n              cur[op] = val;\n            }\n          }\n          merged[field] = cur;\n        }\n        return merged;\n      }, navOpts);\n    },\n    [set],\n  );\n\n  const clear = useCallback(\n    (fields?: string[], navOpts?: NavOpts) => {\n      if (fields == null) {\n        set((prev) => {\n          const next: Record<string, Record<string, unknown>> = {};\n          for (const k of Object.keys(prev)) {\n            next[k] = {};\n          }\n          return next;\n        }, navOpts);\n      } else {\n        const emptyEntries = fields.map((f) => {\n          return [f, {}];\n        });\n        const delta = Object.fromEntries(emptyEntries) as FiltersPatch<\n          Record<string, Record<string, unknown>>\n        >;\n        patch(delta, navOpts);\n      }\n    },\n    [patch, set],\n  );\n\n  const helpers: UseFiltersHelpers<Record<string, Record<string, unknown>>> = {\n    set,\n    patch,\n    clear,\n  };\n  return [filters, helpers];\n}\n\nexport const useQueryFilters = useFilters;\nexport default useFilters;\n"]}