sibujs 1.2.0 → 1.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (89) hide show
  1. package/README.md +29 -25
  2. package/dist/browser.cjs +804 -2
  3. package/dist/browser.d.cts +591 -1
  4. package/dist/browser.d.ts +591 -1
  5. package/dist/browser.js +50 -8
  6. package/dist/build.cjs +654 -144
  7. package/dist/build.js +14 -12
  8. package/dist/cdn.global.js +188 -7
  9. package/dist/chunk-2BYQDGN3.js +742 -0
  10. package/dist/chunk-32DY64NT.js +282 -0
  11. package/dist/chunk-3AIRKM3B.js +1263 -0
  12. package/dist/chunk-3X2YG6YM.js +505 -0
  13. package/dist/chunk-5X6PP2UK.js +28 -0
  14. package/dist/chunk-77L6NL3X.js +1097 -0
  15. package/dist/chunk-BGN5ZMP4.js +26 -0
  16. package/dist/chunk-BTU3TJDS.js +365 -0
  17. package/dist/chunk-CHF5OHIA.js +61 -0
  18. package/dist/chunk-CMBFNA7L.js +27 -0
  19. package/dist/chunk-DAHRH4ON.js +331 -0
  20. package/dist/chunk-EBGIRKQY.js +616 -0
  21. package/dist/chunk-EUZND3CB.js +27 -0
  22. package/dist/chunk-F3FA4F32.js +292 -0
  23. package/dist/chunk-JAKHTMQU.js +1000 -0
  24. package/dist/chunk-JCI5M6U6.js +956 -0
  25. package/dist/chunk-KQPDEVVS.js +398 -0
  26. package/dist/chunk-NEKUBFPT.js +60 -0
  27. package/dist/chunk-NYVAC6P5.js +37 -0
  28. package/dist/chunk-PTQJDMRT.js +146 -0
  29. package/dist/chunk-QWZG56ET.js +2744 -0
  30. package/dist/chunk-TSOKIX5Z.js +654 -0
  31. package/dist/chunk-VRW3FULF.js +725 -0
  32. package/dist/chunk-WZSPOOER.js +84 -0
  33. package/dist/chunk-YT6HQ6AM.js +14 -0
  34. package/dist/chunk-ZD6OAMTH.js +277 -0
  35. package/dist/contracts-DDrwxvJ-.d.cts +245 -0
  36. package/dist/contracts-DDrwxvJ-.d.ts +245 -0
  37. package/dist/data.cjs +35 -2
  38. package/dist/data.d.cts +7 -0
  39. package/dist/data.d.ts +7 -0
  40. package/dist/data.js +9 -8
  41. package/dist/devtools.cjs +122 -0
  42. package/dist/devtools.d.cts +69 -461
  43. package/dist/devtools.d.ts +69 -461
  44. package/dist/devtools.js +127 -6
  45. package/dist/ecosystem.cjs +23 -6
  46. package/dist/ecosystem.d.cts +1 -1
  47. package/dist/ecosystem.d.ts +1 -1
  48. package/dist/ecosystem.js +10 -9
  49. package/dist/extras.cjs +1207 -65
  50. package/dist/extras.d.cts +5 -5
  51. package/dist/extras.d.ts +5 -5
  52. package/dist/extras.js +69 -24
  53. package/dist/index.cjs +663 -144
  54. package/dist/index.d.cts +397 -17
  55. package/dist/index.d.ts +397 -17
  56. package/dist/index.js +39 -17
  57. package/dist/introspect-BumjnBKr.d.cts +477 -0
  58. package/dist/introspect-CZrlcaYy.d.ts +477 -0
  59. package/dist/introspect-Cb0zgpi2.d.cts +477 -0
  60. package/dist/introspect-Y2xNXGSf.d.ts +477 -0
  61. package/dist/motion.js +4 -4
  62. package/dist/patterns.cjs +51 -2
  63. package/dist/patterns.d.cts +18 -8
  64. package/dist/patterns.d.ts +18 -8
  65. package/dist/patterns.js +7 -7
  66. package/dist/performance.js +4 -4
  67. package/dist/plugins.cjs +428 -81
  68. package/dist/plugins.d.cts +27 -4
  69. package/dist/plugins.d.ts +27 -4
  70. package/dist/plugins.js +156 -37
  71. package/dist/ssr-4PBXAOO3.js +40 -0
  72. package/dist/ssr-Do_SiVoL.d.cts +201 -0
  73. package/dist/ssr-Do_SiVoL.d.ts +201 -0
  74. package/dist/ssr.cjs +312 -60
  75. package/dist/ssr.d.cts +10 -1
  76. package/dist/ssr.d.ts +10 -1
  77. package/dist/ssr.js +13 -10
  78. package/dist/tagFactory-DaJ0YWX6.d.cts +47 -0
  79. package/dist/tagFactory-DaJ0YWX6.d.ts +47 -0
  80. package/dist/testing.cjs +233 -2
  81. package/dist/testing.d.cts +42 -1
  82. package/dist/testing.d.ts +42 -1
  83. package/dist/testing.js +129 -2
  84. package/dist/ui.cjs +374 -3
  85. package/dist/ui.d.cts +252 -2
  86. package/dist/ui.d.ts +252 -2
  87. package/dist/ui.js +328 -8
  88. package/dist/widgets.js +7 -7
  89. package/package.json +1 -1
@@ -0,0 +1,1263 @@
1
+ import {
2
+ effect
3
+ } from "./chunk-CHF5OHIA.js";
4
+ import {
5
+ batch,
6
+ signal
7
+ } from "./chunk-WZSPOOER.js";
8
+
9
+ // src/browser/media.ts
10
+ function media(query) {
11
+ if (typeof window === "undefined" || typeof window.matchMedia !== "function") {
12
+ const [matches2] = signal(false);
13
+ return { matches: matches2, dispose: () => {
14
+ } };
15
+ }
16
+ const mql = window.matchMedia(query);
17
+ const [matches, setMatches] = signal(mql.matches);
18
+ const handler = (event) => {
19
+ setMatches(event.matches);
20
+ };
21
+ mql.addEventListener("change", handler);
22
+ function dispose() {
23
+ mql.removeEventListener("change", handler);
24
+ }
25
+ return { matches, dispose };
26
+ }
27
+
28
+ // src/browser/resize.ts
29
+ function resolveTarget(target) {
30
+ return typeof target === "function" ? target : () => target.current;
31
+ }
32
+ function resize(target) {
33
+ const [width, setWidth] = signal(0);
34
+ const [height, setHeight] = signal(0);
35
+ let observer = null;
36
+ if (typeof window === "undefined" || typeof ResizeObserver === "undefined") {
37
+ return { width, height, dispose: () => {
38
+ } };
39
+ }
40
+ const getter = resolveTarget(target);
41
+ const cleanup = effect(() => {
42
+ const el = getter();
43
+ if (observer) {
44
+ observer.disconnect();
45
+ observer = null;
46
+ }
47
+ if (!el) return;
48
+ observer = new ResizeObserver((entries) => {
49
+ const entry = entries[0];
50
+ if (entry) {
51
+ batch(() => {
52
+ setWidth(entry.contentRect.width);
53
+ setHeight(entry.contentRect.height);
54
+ });
55
+ }
56
+ });
57
+ observer.observe(el);
58
+ });
59
+ function dispose() {
60
+ cleanup();
61
+ if (observer) {
62
+ observer.disconnect();
63
+ observer = null;
64
+ }
65
+ }
66
+ return { width, height, dispose };
67
+ }
68
+
69
+ // src/browser/scroll.ts
70
+ function scroll(target) {
71
+ const [x, setX] = signal(0);
72
+ const [y, setY] = signal(0);
73
+ const [isScrolling, setIsScrolling] = signal(false);
74
+ let scrollTimer = null;
75
+ if (typeof window === "undefined") {
76
+ return { x, y, isScrolling, dispose: () => {
77
+ } };
78
+ }
79
+ const handler = () => {
80
+ const el = target ? target() : null;
81
+ batch(() => {
82
+ if (el) {
83
+ setX(el.scrollLeft);
84
+ setY(el.scrollTop);
85
+ } else {
86
+ setX(window.scrollX ?? window.pageXOffset ?? 0);
87
+ setY(window.scrollY ?? window.pageYOffset ?? 0);
88
+ }
89
+ setIsScrolling(true);
90
+ });
91
+ if (scrollTimer !== null) clearTimeout(scrollTimer);
92
+ scrollTimer = setTimeout(() => {
93
+ setIsScrolling(false);
94
+ scrollTimer = null;
95
+ }, 150);
96
+ };
97
+ const scrollTarget = target ? target() : null;
98
+ const eventTarget = scrollTarget || window;
99
+ eventTarget.addEventListener("scroll", handler, { passive: true });
100
+ function dispose() {
101
+ eventTarget.removeEventListener("scroll", handler);
102
+ if (scrollTimer !== null) {
103
+ clearTimeout(scrollTimer);
104
+ scrollTimer = null;
105
+ }
106
+ }
107
+ return { x, y, isScrolling, dispose };
108
+ }
109
+
110
+ // src/browser/online.ts
111
+ function online() {
112
+ if (typeof window === "undefined" || typeof navigator === "undefined") {
113
+ const [online3] = signal(true);
114
+ return { online: online3, dispose: () => {
115
+ } };
116
+ }
117
+ const [online2, setOnline] = signal(navigator.onLine);
118
+ const onOnline = () => setOnline(true);
119
+ const onOffline = () => setOnline(false);
120
+ window.addEventListener("online", onOnline);
121
+ window.addEventListener("offline", onOffline);
122
+ function dispose() {
123
+ window.removeEventListener("online", onOnline);
124
+ window.removeEventListener("offline", onOffline);
125
+ }
126
+ return { online: online2, dispose };
127
+ }
128
+
129
+ // src/browser/geo.ts
130
+ function geo(options) {
131
+ const [latitude, setLatitude] = signal(null);
132
+ const [longitude, setLongitude] = signal(null);
133
+ const [accuracy, setAccuracy] = signal(null);
134
+ const [error, setError] = signal(null);
135
+ let watchId = null;
136
+ if (typeof navigator !== "undefined" && navigator.geolocation) {
137
+ watchId = navigator.geolocation.watchPosition(
138
+ (position) => {
139
+ batch(() => {
140
+ setLatitude(position.coords.latitude);
141
+ setLongitude(position.coords.longitude);
142
+ setAccuracy(position.coords.accuracy);
143
+ setError(null);
144
+ });
145
+ },
146
+ (err) => {
147
+ setError(err);
148
+ },
149
+ options
150
+ );
151
+ }
152
+ function dispose() {
153
+ if (watchId !== null && typeof navigator !== "undefined" && navigator.geolocation) {
154
+ navigator.geolocation.clearWatch(watchId);
155
+ watchId = null;
156
+ }
157
+ }
158
+ return { latitude, longitude, accuracy, error, dispose };
159
+ }
160
+
161
+ // src/browser/battery.ts
162
+ function battery() {
163
+ const [level, setLevel] = signal(null);
164
+ const [charging, setCharging] = signal(null);
165
+ const [chargingTime, setChargingTime] = signal(null);
166
+ const [dischargingTime, setDischargingTime] = signal(null);
167
+ const [supported, setSupported] = signal(false);
168
+ let battery2 = null;
169
+ let onLevelChange = null;
170
+ let onChargingChange = null;
171
+ let onChargingTimeChange = null;
172
+ let onDischargingTimeChange = null;
173
+ let disposed = false;
174
+ if (typeof navigator !== "undefined" && "getBattery" in navigator) {
175
+ setSupported(true);
176
+ navigator.getBattery().then((bm) => {
177
+ if (disposed) return;
178
+ battery2 = bm;
179
+ batch(() => {
180
+ setLevel(bm.level);
181
+ setCharging(bm.charging);
182
+ setChargingTime(bm.chargingTime);
183
+ setDischargingTime(bm.dischargingTime);
184
+ });
185
+ onLevelChange = () => setLevel(bm.level);
186
+ onChargingChange = () => setCharging(bm.charging);
187
+ onChargingTimeChange = () => setChargingTime(bm.chargingTime);
188
+ onDischargingTimeChange = () => setDischargingTime(bm.dischargingTime);
189
+ bm.addEventListener("levelchange", onLevelChange);
190
+ bm.addEventListener("chargingchange", onChargingChange);
191
+ bm.addEventListener("chargingtimechange", onChargingTimeChange);
192
+ bm.addEventListener("dischargingtimechange", onDischargingTimeChange);
193
+ });
194
+ }
195
+ function dispose() {
196
+ disposed = true;
197
+ if (battery2) {
198
+ if (onLevelChange) battery2.removeEventListener("levelchange", onLevelChange);
199
+ if (onChargingChange) battery2.removeEventListener("chargingchange", onChargingChange);
200
+ if (onChargingTimeChange) battery2.removeEventListener("chargingtimechange", onChargingTimeChange);
201
+ if (onDischargingTimeChange) battery2.removeEventListener("dischargingtimechange", onDischargingTimeChange);
202
+ battery2 = null;
203
+ }
204
+ }
205
+ return { level, charging, chargingTime, dischargingTime, supported, dispose };
206
+ }
207
+
208
+ // src/browser/idle.ts
209
+ var ACTIVITY_EVENTS = ["mousemove", "mousedown", "keydown", "touchstart", "scroll"];
210
+ function idle(timeout = 6e4) {
211
+ const [idle2, setIdle] = signal(false);
212
+ if (typeof window === "undefined" || typeof document === "undefined") {
213
+ return { idle: idle2, dispose: () => {
214
+ } };
215
+ }
216
+ let timer = null;
217
+ function resetTimer() {
218
+ setIdle(false);
219
+ if (timer !== null) clearTimeout(timer);
220
+ timer = setTimeout(() => {
221
+ setIdle(true);
222
+ }, timeout);
223
+ }
224
+ for (const event of ACTIVITY_EVENTS) {
225
+ document.addEventListener(event, resetTimer, { passive: true });
226
+ }
227
+ resetTimer();
228
+ function dispose() {
229
+ if (timer !== null) {
230
+ clearTimeout(timer);
231
+ timer = null;
232
+ }
233
+ for (const event of ACTIVITY_EVENTS) {
234
+ document.removeEventListener(event, resetTimer);
235
+ }
236
+ }
237
+ return { idle: idle2, dispose };
238
+ }
239
+
240
+ // src/browser/permissions.ts
241
+ function permissions(name) {
242
+ const [state, setState] = signal("prompt");
243
+ let permissionStatus = null;
244
+ let onChange = null;
245
+ let disposed = false;
246
+ if (typeof navigator === "undefined" || !navigator.permissions) {
247
+ setState("unsupported");
248
+ return { state, dispose: () => {
249
+ } };
250
+ }
251
+ navigator.permissions.query({ name }).then((status) => {
252
+ if (disposed) return;
253
+ permissionStatus = status;
254
+ setState(status.state);
255
+ onChange = () => {
256
+ setState(status.state);
257
+ };
258
+ status.addEventListener("change", onChange);
259
+ }).catch(() => {
260
+ setState("unsupported");
261
+ });
262
+ function dispose() {
263
+ disposed = true;
264
+ if (permissionStatus && onChange) {
265
+ permissionStatus.removeEventListener("change", onChange);
266
+ permissionStatus = null;
267
+ onChange = null;
268
+ }
269
+ }
270
+ return { state, dispose };
271
+ }
272
+
273
+ // src/browser/clipboard.ts
274
+ function clipboard() {
275
+ const [text, setText] = signal("");
276
+ const [copied, setCopied] = signal(false);
277
+ let copiedTimer = null;
278
+ async function copy(value) {
279
+ if (typeof navigator === "undefined" || !navigator.clipboard) {
280
+ return;
281
+ }
282
+ await navigator.clipboard.writeText(value);
283
+ setText(value);
284
+ setCopied(true);
285
+ if (copiedTimer !== null) clearTimeout(copiedTimer);
286
+ copiedTimer = setTimeout(() => {
287
+ setCopied(false);
288
+ copiedTimer = null;
289
+ }, 2e3);
290
+ }
291
+ function dispose() {
292
+ if (copiedTimer !== null) {
293
+ clearTimeout(copiedTimer);
294
+ copiedTimer = null;
295
+ }
296
+ }
297
+ return { text, copy, copied, dispose };
298
+ }
299
+
300
+ // src/browser/dragDrop.ts
301
+ function resolveTarget2(target) {
302
+ return typeof target === "function" ? target : () => target.current;
303
+ }
304
+ function draggable(element, data) {
305
+ const [isDragging, setIsDragging] = signal(false);
306
+ if (typeof window === "undefined") {
307
+ return { isDragging, dispose: () => {
308
+ } };
309
+ }
310
+ let currentEl = null;
311
+ let onDragStart = null;
312
+ let onDragEnd = null;
313
+ const getter = resolveTarget2(element);
314
+ const cleanup = effect(() => {
315
+ if (currentEl && onDragStart && onDragEnd) {
316
+ currentEl.removeEventListener("dragstart", onDragStart);
317
+ currentEl.removeEventListener("dragend", onDragEnd);
318
+ }
319
+ const el = getter();
320
+ currentEl = el;
321
+ if (!el) return;
322
+ el.draggable = true;
323
+ onDragStart = (e) => {
324
+ setIsDragging(true);
325
+ if (e.dataTransfer && data !== void 0) {
326
+ e.dataTransfer.setData("application/json", JSON.stringify(data));
327
+ }
328
+ };
329
+ onDragEnd = () => {
330
+ setIsDragging(false);
331
+ };
332
+ el.addEventListener("dragstart", onDragStart);
333
+ el.addEventListener("dragend", onDragEnd);
334
+ });
335
+ function dispose() {
336
+ cleanup();
337
+ if (currentEl && onDragStart && onDragEnd) {
338
+ currentEl.removeEventListener("dragstart", onDragStart);
339
+ currentEl.removeEventListener("dragend", onDragEnd);
340
+ currentEl = null;
341
+ }
342
+ }
343
+ return { isDragging, dispose };
344
+ }
345
+ function dropZone(element, options) {
346
+ const [isOver, setIsOver] = signal(false);
347
+ if (typeof window === "undefined") {
348
+ return { isOver, dispose: () => {
349
+ } };
350
+ }
351
+ let currentEl = null;
352
+ let onDragOver = null;
353
+ let onDragEnter = null;
354
+ let onDragLeave = null;
355
+ let onDrop = null;
356
+ const getter = resolveTarget2(element);
357
+ const cleanup = effect(() => {
358
+ if (currentEl && onDragOver && onDragEnter && onDragLeave && onDrop) {
359
+ currentEl.removeEventListener("dragover", onDragOver);
360
+ currentEl.removeEventListener("dragenter", onDragEnter);
361
+ currentEl.removeEventListener("dragleave", onDragLeave);
362
+ currentEl.removeEventListener("drop", onDrop);
363
+ }
364
+ const el = getter();
365
+ currentEl = el;
366
+ if (!el) return;
367
+ onDragOver = (e) => {
368
+ e.preventDefault();
369
+ };
370
+ onDragEnter = (e) => {
371
+ e.preventDefault();
372
+ setIsOver(true);
373
+ };
374
+ onDragLeave = () => {
375
+ setIsOver(false);
376
+ };
377
+ onDrop = (e) => {
378
+ e.preventDefault();
379
+ setIsOver(false);
380
+ let transferData = null;
381
+ if (e.dataTransfer) {
382
+ const raw = e.dataTransfer.getData("application/json");
383
+ if (raw) {
384
+ try {
385
+ transferData = JSON.parse(raw);
386
+ } catch {
387
+ transferData = raw;
388
+ }
389
+ }
390
+ }
391
+ options.onDrop(transferData, e);
392
+ };
393
+ el.addEventListener("dragover", onDragOver);
394
+ el.addEventListener("dragenter", onDragEnter);
395
+ el.addEventListener("dragleave", onDragLeave);
396
+ el.addEventListener("drop", onDrop);
397
+ });
398
+ function dispose() {
399
+ cleanup();
400
+ if (currentEl && onDragOver && onDragEnter && onDragLeave && onDrop) {
401
+ currentEl.removeEventListener("dragover", onDragOver);
402
+ currentEl.removeEventListener("dragenter", onDragEnter);
403
+ currentEl.removeEventListener("dragleave", onDragLeave);
404
+ currentEl.removeEventListener("drop", onDrop);
405
+ currentEl = null;
406
+ }
407
+ }
408
+ return { isOver, dispose };
409
+ }
410
+
411
+ // src/browser/title.ts
412
+ function title(value) {
413
+ if (typeof document === "undefined") {
414
+ return () => {
415
+ };
416
+ }
417
+ const previousTitle = document.title;
418
+ if (typeof value === "function") {
419
+ const cleanup = effect(() => {
420
+ document.title = value();
421
+ });
422
+ return () => {
423
+ cleanup();
424
+ document.title = previousTitle;
425
+ };
426
+ }
427
+ document.title = value;
428
+ return () => {
429
+ document.title = previousTitle;
430
+ };
431
+ }
432
+
433
+ // src/browser/colorScheme.ts
434
+ function colorScheme() {
435
+ if (typeof window === "undefined" || typeof window.matchMedia !== "function") {
436
+ const [scheme2] = signal("light");
437
+ return { scheme: scheme2, dispose: () => {
438
+ } };
439
+ }
440
+ const mql = window.matchMedia("(prefers-color-scheme: dark)");
441
+ const [scheme, setScheme] = signal(mql.matches ? "dark" : "light");
442
+ const handler = (event) => {
443
+ setScheme(event.matches ? "dark" : "light");
444
+ };
445
+ mql.addEventListener("change", handler);
446
+ function dispose() {
447
+ mql.removeEventListener("change", handler);
448
+ }
449
+ return { scheme, dispose };
450
+ }
451
+
452
+ // src/browser/format.ts
453
+ function formatNumber(value, options) {
454
+ const { locale, ...formatOptions } = options ?? {};
455
+ return new Intl.NumberFormat(locale, formatOptions).format(value);
456
+ }
457
+ function formatCurrency(value, currency, options) {
458
+ const { locale, ...formatOptions } = options ?? {};
459
+ return new Intl.NumberFormat(locale, {
460
+ style: "currency",
461
+ currency,
462
+ ...formatOptions
463
+ }).format(value);
464
+ }
465
+
466
+ // src/browser/visibility.ts
467
+ function visibility() {
468
+ if (typeof document === "undefined") {
469
+ const [visible2] = signal(true);
470
+ return { visible: visible2, dispose: () => {
471
+ } };
472
+ }
473
+ const [visible, setVisible] = signal(!document.hidden);
474
+ const handler = () => setVisible(!document.hidden);
475
+ document.addEventListener("visibilitychange", handler);
476
+ function dispose() {
477
+ document.removeEventListener("visibilitychange", handler);
478
+ }
479
+ return { visible, dispose };
480
+ }
481
+
482
+ // src/browser/network.ts
483
+ function network() {
484
+ const connection = typeof navigator !== "undefined" ? navigator.connection ?? navigator.mozConnection ?? navigator.webkitConnection : void 0;
485
+ const [effectiveType, setEffectiveType] = signal(connection?.effectiveType ?? "unknown");
486
+ const [downlink, setDownlink] = signal(connection?.downlink ?? 0);
487
+ const [rtt, setRtt] = signal(connection?.rtt ?? 0);
488
+ const [saveData, setSaveData] = signal(connection?.saveData ?? false);
489
+ if (!connection) {
490
+ return { effectiveType, downlink, rtt, saveData, dispose: () => {
491
+ } };
492
+ }
493
+ const update = () => {
494
+ setEffectiveType(connection.effectiveType ?? "unknown");
495
+ setDownlink(connection.downlink ?? 0);
496
+ setRtt(connection.rtt ?? 0);
497
+ setSaveData(connection.saveData ?? false);
498
+ };
499
+ connection.addEventListener("change", update);
500
+ function dispose() {
501
+ connection?.removeEventListener("change", update);
502
+ }
503
+ return { effectiveType, downlink, rtt, saveData, dispose };
504
+ }
505
+
506
+ // src/browser/mouse.ts
507
+ function mouse(options = {}) {
508
+ const [x, setX] = signal(0);
509
+ const [y, setY] = signal(0);
510
+ if (typeof window === "undefined") {
511
+ return { x, y, dispose: () => {
512
+ } };
513
+ }
514
+ const target = options.target ?? window;
515
+ const trackTouch = options.touch ?? true;
516
+ const onMove = (e) => {
517
+ setX(e.clientX);
518
+ setY(e.clientY);
519
+ };
520
+ const onTouchMove = (e) => {
521
+ if (e.touches.length === 0) return;
522
+ setX(e.touches[0].clientX);
523
+ setY(e.touches[0].clientY);
524
+ };
525
+ target.addEventListener("mousemove", onMove, { passive: true });
526
+ if (trackTouch) {
527
+ target.addEventListener("touchmove", onTouchMove, { passive: true });
528
+ }
529
+ function dispose() {
530
+ target.removeEventListener("mousemove", onMove);
531
+ if (trackTouch) {
532
+ target.removeEventListener("touchmove", onTouchMove);
533
+ }
534
+ }
535
+ return { x, y, dispose };
536
+ }
537
+
538
+ // src/browser/swipe.ts
539
+ function swipe(target, options = {}) {
540
+ const threshold = options.threshold ?? 50;
541
+ const [direction, setDirection] = signal(null);
542
+ if (typeof window === "undefined") {
543
+ return { direction, dispose: () => {
544
+ } };
545
+ }
546
+ let startX = 0;
547
+ let startY = 0;
548
+ let tracking = false;
549
+ const onStart = (e) => {
550
+ if (e.touches.length === 0) return;
551
+ startX = e.touches[0].clientX;
552
+ startY = e.touches[0].clientY;
553
+ tracking = true;
554
+ };
555
+ const onEnd = (e) => {
556
+ if (!tracking) return;
557
+ tracking = false;
558
+ const touch = e.changedTouches[0];
559
+ if (!touch) return;
560
+ const dx = touch.clientX - startX;
561
+ const dy = touch.clientY - startY;
562
+ const absX = Math.abs(dx);
563
+ const absY = Math.abs(dy);
564
+ if (Math.max(absX, absY) < threshold) return;
565
+ let dir;
566
+ if (absX > absY) {
567
+ dir = dx > 0 ? "right" : "left";
568
+ } else {
569
+ dir = dy > 0 ? "down" : "up";
570
+ }
571
+ setDirection(dir);
572
+ options.onSwipe?.(dir, Math.max(absX, absY));
573
+ };
574
+ target.addEventListener("touchstart", onStart, { passive: true });
575
+ target.addEventListener("touchend", onEnd, { passive: true });
576
+ function dispose() {
577
+ target.removeEventListener("touchstart", onStart);
578
+ target.removeEventListener("touchend", onEnd);
579
+ }
580
+ return { direction, dispose };
581
+ }
582
+
583
+ // src/browser/windowSize.ts
584
+ function windowSize() {
585
+ if (typeof window === "undefined") {
586
+ const [width2] = signal(0);
587
+ const [height2] = signal(0);
588
+ return { width: width2, height: height2, dispose: () => {
589
+ } };
590
+ }
591
+ const [width, setWidth] = signal(window.innerWidth);
592
+ const [height, setHeight] = signal(window.innerHeight);
593
+ const handler = () => {
594
+ setWidth(window.innerWidth);
595
+ setHeight(window.innerHeight);
596
+ };
597
+ window.addEventListener("resize", handler, { passive: true });
598
+ function dispose() {
599
+ window.removeEventListener("resize", handler);
600
+ }
601
+ return { width, height, dispose };
602
+ }
603
+
604
+ // src/browser/urlState.ts
605
+ function urlState() {
606
+ if (typeof window === "undefined") {
607
+ const [params2] = signal(new URLSearchParams());
608
+ const [hash2] = signal("");
609
+ return {
610
+ params: params2,
611
+ hash: hash2,
612
+ setParams: () => {
613
+ },
614
+ setHash: () => {
615
+ },
616
+ dispose: () => {
617
+ }
618
+ };
619
+ }
620
+ const [params, setParamsSignal] = signal(new URLSearchParams(window.location.search));
621
+ const [hash, setHashSignal] = signal(window.location.hash);
622
+ const syncFromLocation = () => {
623
+ setParamsSignal(new URLSearchParams(window.location.search));
624
+ setHashSignal(window.location.hash);
625
+ };
626
+ const onPopState = () => syncFromLocation();
627
+ window.addEventListener("popstate", onPopState);
628
+ function setParams(next, opts = {}) {
629
+ const p = next instanceof URLSearchParams ? next : new URLSearchParams(next);
630
+ const query = p.toString();
631
+ const newUrl = `${window.location.pathname}${query ? `?${query}` : ""}${window.location.hash}`;
632
+ if (opts.replace) window.history.replaceState(null, "", newUrl);
633
+ else window.history.pushState(null, "", newUrl);
634
+ setParamsSignal(new URLSearchParams(p));
635
+ }
636
+ function setHash(next, opts = {}) {
637
+ const normalized = next.startsWith("#") ? next : next ? `#${next}` : "";
638
+ const newUrl = `${window.location.pathname}${window.location.search}${normalized}`;
639
+ if (opts.replace) window.history.replaceState(null, "", newUrl);
640
+ else window.history.pushState(null, "", newUrl);
641
+ setHashSignal(normalized);
642
+ }
643
+ function dispose() {
644
+ window.removeEventListener("popstate", onPopState);
645
+ }
646
+ return { params, hash, setParams, setHash, dispose };
647
+ }
648
+
649
+ // src/browser/broadcast.ts
650
+ function broadcast(channelName) {
651
+ if (typeof BroadcastChannel === "undefined") {
652
+ const [last2] = signal(null);
653
+ return { last: last2, post: () => {
654
+ }, dispose: () => {
655
+ } };
656
+ }
657
+ const [last, setLast] = signal(null);
658
+ const channel = new BroadcastChannel(channelName);
659
+ const handler = (ev) => setLast(ev.data);
660
+ channel.addEventListener("message", handler);
661
+ function post(message) {
662
+ channel.postMessage(message);
663
+ }
664
+ function dispose() {
665
+ channel.removeEventListener("message", handler);
666
+ channel.close();
667
+ }
668
+ return { last, post, dispose };
669
+ }
670
+
671
+ // src/browser/fullscreen.ts
672
+ function fullscreen() {
673
+ if (typeof document === "undefined") {
674
+ const [isFullscreen2] = signal(false);
675
+ const [element2] = signal(null);
676
+ return {
677
+ isFullscreen: isFullscreen2,
678
+ element: element2,
679
+ enter: async () => {
680
+ },
681
+ exit: async () => {
682
+ },
683
+ toggle: async () => {
684
+ },
685
+ dispose: () => {
686
+ }
687
+ };
688
+ }
689
+ const [isFullscreen, setIsFullscreen] = signal(!!document.fullscreenElement);
690
+ const [element, setElement] = signal(document.fullscreenElement);
691
+ const handler = () => {
692
+ setIsFullscreen(!!document.fullscreenElement);
693
+ setElement(document.fullscreenElement);
694
+ };
695
+ document.addEventListener("fullscreenchange", handler);
696
+ async function enter(el) {
697
+ if (!document.fullscreenElement && el.requestFullscreen) {
698
+ await el.requestFullscreen();
699
+ }
700
+ }
701
+ async function exit() {
702
+ if (document.fullscreenElement && document.exitFullscreen) {
703
+ await document.exitFullscreen();
704
+ }
705
+ }
706
+ async function toggle(el) {
707
+ if (document.fullscreenElement) await exit();
708
+ else await enter(el);
709
+ }
710
+ function dispose() {
711
+ document.removeEventListener("fullscreenchange", handler);
712
+ }
713
+ return { isFullscreen, element, enter, exit, toggle, dispose };
714
+ }
715
+
716
+ // src/browser/wakeLock.ts
717
+ function wakeLock() {
718
+ const [active, setActive] = signal(false);
719
+ if (typeof navigator === "undefined" || !("wakeLock" in navigator) || typeof document === "undefined") {
720
+ return {
721
+ active,
722
+ request: async () => {
723
+ },
724
+ release: async () => {
725
+ },
726
+ dispose: () => {
727
+ }
728
+ };
729
+ }
730
+ const api = navigator.wakeLock;
731
+ let sentinel = null;
732
+ async function request() {
733
+ try {
734
+ sentinel = await api.request("screen");
735
+ setActive(true);
736
+ sentinel.addEventListener("release", () => {
737
+ setActive(false);
738
+ });
739
+ } catch {
740
+ setActive(false);
741
+ }
742
+ }
743
+ async function release() {
744
+ if (sentinel && !sentinel.released) {
745
+ await sentinel.release();
746
+ }
747
+ sentinel = null;
748
+ setActive(false);
749
+ }
750
+ const onVisibility = () => {
751
+ if (sentinel?.released && !document.hidden) {
752
+ void request();
753
+ }
754
+ };
755
+ document.addEventListener("visibilitychange", onVisibility);
756
+ function dispose() {
757
+ document.removeEventListener("visibilitychange", onVisibility);
758
+ void release();
759
+ }
760
+ return { active, request, release, dispose };
761
+ }
762
+
763
+ // src/browser/animationFrame.ts
764
+ function animationFrame(options = {}) {
765
+ const [delta, setDelta] = signal(0);
766
+ const [elapsed, setElapsed] = signal(0);
767
+ const [running, setRunning] = signal(false);
768
+ if (typeof requestAnimationFrame === "undefined") {
769
+ return {
770
+ delta,
771
+ elapsed,
772
+ running,
773
+ pause: () => {
774
+ },
775
+ resume: () => {
776
+ },
777
+ dispose: () => {
778
+ }
779
+ };
780
+ }
781
+ let id = null;
782
+ let prev = -1;
783
+ let start = -1;
784
+ const minFrameMs = options.fpsLimit ? 1e3 / options.fpsLimit : 0;
785
+ const step = (now) => {
786
+ if (start < 0) start = now;
787
+ const firstTick = prev < 0;
788
+ const dt = firstTick ? 0 : now - prev;
789
+ if (firstTick || dt >= minFrameMs) {
790
+ setDelta(dt);
791
+ setElapsed(now - start);
792
+ prev = now;
793
+ }
794
+ id = requestAnimationFrame(step);
795
+ };
796
+ function resume() {
797
+ if (id !== null) return;
798
+ setRunning(true);
799
+ id = requestAnimationFrame(step);
800
+ }
801
+ function pause() {
802
+ if (id !== null) {
803
+ cancelAnimationFrame(id);
804
+ id = null;
805
+ }
806
+ setRunning(false);
807
+ prev = -1;
808
+ start = -1;
809
+ }
810
+ function dispose() {
811
+ pause();
812
+ }
813
+ if (options.immediate !== false) resume();
814
+ return { delta, elapsed, running, pause, resume, dispose };
815
+ }
816
+
817
+ // src/browser/mutationObserver.ts
818
+ function mutationObserver(target, options = { childList: true, subtree: true }) {
819
+ const [records, setRecords] = signal([]);
820
+ if (typeof MutationObserver === "undefined") {
821
+ return { records, dispose: () => {
822
+ } };
823
+ }
824
+ const observer = new MutationObserver((batch2) => {
825
+ setRecords(batch2);
826
+ });
827
+ observer.observe(target, options);
828
+ function dispose() {
829
+ observer.disconnect();
830
+ }
831
+ return { records, dispose };
832
+ }
833
+
834
+ // src/browser/bounds.ts
835
+ var ZERO = {
836
+ x: 0,
837
+ y: 0,
838
+ width: 0,
839
+ height: 0,
840
+ top: 0,
841
+ left: 0,
842
+ right: 0,
843
+ bottom: 0
844
+ };
845
+ function readRect(el) {
846
+ const r = el.getBoundingClientRect();
847
+ return {
848
+ x: r.x,
849
+ y: r.y,
850
+ width: r.width,
851
+ height: r.height,
852
+ top: r.top,
853
+ left: r.left,
854
+ right: r.right,
855
+ bottom: r.bottom
856
+ };
857
+ }
858
+ function bounds(target) {
859
+ const [rect, setRect] = signal(ZERO);
860
+ if (typeof window === "undefined" || !target) {
861
+ return {
862
+ rect,
863
+ refresh: () => {
864
+ },
865
+ dispose: () => {
866
+ }
867
+ };
868
+ }
869
+ function refresh() {
870
+ setRect(readRect(target));
871
+ }
872
+ refresh();
873
+ let resizeObserver = null;
874
+ if (typeof ResizeObserver !== "undefined") {
875
+ resizeObserver = new ResizeObserver(refresh);
876
+ resizeObserver.observe(target);
877
+ }
878
+ const onScroll = () => refresh();
879
+ window.addEventListener("scroll", onScroll, { passive: true, capture: true });
880
+ function dispose() {
881
+ resizeObserver?.disconnect();
882
+ window.removeEventListener("scroll", onScroll, { capture: true });
883
+ }
884
+ return { rect, refresh, dispose };
885
+ }
886
+
887
+ // src/browser/keyboard.ts
888
+ function keyboard(options = {}) {
889
+ const [pressed, setPressed] = signal(/* @__PURE__ */ new Set());
890
+ if (typeof window === "undefined") {
891
+ return {
892
+ pressed,
893
+ isPressed: () => false,
894
+ dispose: () => {
895
+ }
896
+ };
897
+ }
898
+ const target = options.target ?? window;
899
+ const filter = options.keys ? new Set(options.keys) : null;
900
+ const onDown = (e) => {
901
+ if (filter && !filter.has(e.key)) return;
902
+ setPressed((prev) => {
903
+ if (prev.has(e.key)) return prev;
904
+ const next = new Set(prev);
905
+ next.add(e.key);
906
+ return next;
907
+ });
908
+ };
909
+ const onUp = (e) => {
910
+ if (filter && !filter.has(e.key)) return;
911
+ setPressed((prev) => {
912
+ if (!prev.has(e.key)) return prev;
913
+ const next = new Set(prev);
914
+ next.delete(e.key);
915
+ return next;
916
+ });
917
+ };
918
+ const onBlur = () => setPressed(/* @__PURE__ */ new Set());
919
+ target.addEventListener("keydown", onDown);
920
+ target.addEventListener("keyup", onUp);
921
+ window.addEventListener("blur", onBlur);
922
+ function isPressed(key) {
923
+ return pressed().has(key);
924
+ }
925
+ function dispose() {
926
+ target.removeEventListener("keydown", onDown);
927
+ target.removeEventListener("keyup", onUp);
928
+ window.removeEventListener("blur", onBlur);
929
+ }
930
+ return { pressed, isPressed, dispose };
931
+ }
932
+
933
+ // src/browser/speech.ts
934
+ function speech() {
935
+ const [speaking, setSpeaking] = signal(false);
936
+ const [paused, setPaused] = signal(false);
937
+ if (typeof window === "undefined" || typeof window.speechSynthesis === "undefined") {
938
+ return {
939
+ speaking,
940
+ paused,
941
+ speak: () => {
942
+ },
943
+ pause: () => {
944
+ },
945
+ resume: () => {
946
+ },
947
+ cancel: () => {
948
+ },
949
+ dispose: () => {
950
+ }
951
+ };
952
+ }
953
+ const synth = window.speechSynthesis;
954
+ const interval = setInterval(() => {
955
+ setSpeaking(synth.speaking);
956
+ setPaused(synth.paused);
957
+ }, 200);
958
+ function speak(text, options = {}) {
959
+ const u = new SpeechSynthesisUtterance(text);
960
+ if (options.lang) u.lang = options.lang;
961
+ if (options.rate != null) u.rate = options.rate;
962
+ if (options.pitch != null) u.pitch = options.pitch;
963
+ if (options.volume != null) u.volume = options.volume;
964
+ if (options.voice) {
965
+ const voices = synth.getVoices();
966
+ const match = voices.find((v) => v.name === options.voice);
967
+ if (match) u.voice = match;
968
+ }
969
+ u.addEventListener("start", () => setSpeaking(true));
970
+ u.addEventListener("end", () => {
971
+ setSpeaking(false);
972
+ setPaused(false);
973
+ });
974
+ u.addEventListener("error", () => {
975
+ setSpeaking(false);
976
+ setPaused(false);
977
+ });
978
+ synth.speak(u);
979
+ }
980
+ function dispose() {
981
+ clearInterval(interval);
982
+ synth.cancel();
983
+ }
984
+ return {
985
+ speaking,
986
+ paused,
987
+ speak,
988
+ pause: () => synth.pause(),
989
+ resume: () => synth.resume(),
990
+ cancel: () => synth.cancel(),
991
+ dispose
992
+ };
993
+ }
994
+
995
+ // src/browser/gamepad.ts
996
+ function gamepad() {
997
+ const [pads, setPads] = signal([]);
998
+ if (typeof window === "undefined" || typeof navigator === "undefined" || typeof navigator.getGamepads !== "function") {
999
+ return { pads, dispose: () => {
1000
+ } };
1001
+ }
1002
+ let rafId = null;
1003
+ function snapshot(pad) {
1004
+ return {
1005
+ index: pad.index,
1006
+ id: pad.id,
1007
+ connected: pad.connected,
1008
+ buttons: pad.buttons.map((b) => ({ pressed: b.pressed, value: b.value })),
1009
+ axes: [...pad.axes]
1010
+ };
1011
+ }
1012
+ function equal(a, b) {
1013
+ if (a.length !== b.length) return false;
1014
+ for (let i = 0; i < a.length; i++) {
1015
+ const pa = a[i];
1016
+ const pb = b[i];
1017
+ if (pa.index !== pb.index || pa.connected !== pb.connected) return false;
1018
+ if (pa.buttons.length !== pb.buttons.length) return false;
1019
+ for (let j = 0; j < pa.buttons.length; j++) {
1020
+ if (pa.buttons[j].pressed !== pb.buttons[j].pressed) return false;
1021
+ if (pa.buttons[j].value !== pb.buttons[j].value) return false;
1022
+ }
1023
+ if (pa.axes.length !== pb.axes.length) return false;
1024
+ for (let j = 0; j < pa.axes.length; j++) {
1025
+ if (pa.axes[j] !== pb.axes[j]) return false;
1026
+ }
1027
+ }
1028
+ return true;
1029
+ }
1030
+ function poll() {
1031
+ const raw = navigator.getGamepads();
1032
+ const snap = Array.from(raw).filter((g) => g !== null).map(snapshot);
1033
+ const current = pads();
1034
+ if (!equal(current, snap)) setPads(snap);
1035
+ rafId = requestAnimationFrame(poll);
1036
+ }
1037
+ function startPolling() {
1038
+ if (rafId === null) poll();
1039
+ }
1040
+ function stopPolling() {
1041
+ if (rafId !== null) {
1042
+ cancelAnimationFrame(rafId);
1043
+ rafId = null;
1044
+ }
1045
+ }
1046
+ const onConnect = () => startPolling();
1047
+ const onDisconnect = () => {
1048
+ const raw = navigator.getGamepads();
1049
+ const hasAny = Array.from(raw).some((g) => g !== null);
1050
+ if (!hasAny) stopPolling();
1051
+ };
1052
+ window.addEventListener("gamepadconnected", onConnect);
1053
+ window.addEventListener("gamepaddisconnected", onDisconnect);
1054
+ const initial = Array.from(navigator.getGamepads()).some((g) => g !== null);
1055
+ if (initial) startPolling();
1056
+ function dispose() {
1057
+ stopPolling();
1058
+ window.removeEventListener("gamepadconnected", onConnect);
1059
+ window.removeEventListener("gamepaddisconnected", onDisconnect);
1060
+ }
1061
+ return { pads, dispose };
1062
+ }
1063
+
1064
+ // src/browser/pointerLock.ts
1065
+ function pointerLock() {
1066
+ const [locked, setLocked] = signal(false);
1067
+ if (typeof document === "undefined") {
1068
+ return {
1069
+ locked,
1070
+ request: () => {
1071
+ },
1072
+ exit: () => {
1073
+ },
1074
+ dispose: () => {
1075
+ }
1076
+ };
1077
+ }
1078
+ const handler = () => {
1079
+ setLocked(!!document.pointerLockElement);
1080
+ };
1081
+ document.addEventListener("pointerlockchange", handler);
1082
+ function request(element) {
1083
+ if (typeof element.requestPointerLock === "function") {
1084
+ element.requestPointerLock();
1085
+ }
1086
+ }
1087
+ function exit() {
1088
+ if (typeof document.exitPointerLock === "function") {
1089
+ document.exitPointerLock();
1090
+ }
1091
+ }
1092
+ function dispose() {
1093
+ document.removeEventListener("pointerlockchange", handler);
1094
+ }
1095
+ return { locked, request, exit, dispose };
1096
+ }
1097
+
1098
+ // src/browser/vibrate.ts
1099
+ function vibrate(pattern) {
1100
+ if (typeof navigator === "undefined" || typeof navigator.vibrate !== "function") {
1101
+ return false;
1102
+ }
1103
+ return navigator.vibrate(pattern);
1104
+ }
1105
+
1106
+ // src/browser/favicon.ts
1107
+ function favicon(url) {
1108
+ if (typeof document === "undefined") return;
1109
+ let link = document.querySelector("link[rel='icon']");
1110
+ if (!link) {
1111
+ link = document.createElement("link");
1112
+ link.rel = "icon";
1113
+ document.head.appendChild(link);
1114
+ }
1115
+ link.href = url;
1116
+ }
1117
+ function svgFavicon(svg) {
1118
+ return `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`;
1119
+ }
1120
+
1121
+ // src/browser/textSelection.ts
1122
+ function textSelection() {
1123
+ const [text, setText] = signal("");
1124
+ const [rect, setRect] = signal(null);
1125
+ if (typeof document === "undefined") {
1126
+ return {
1127
+ text,
1128
+ rect,
1129
+ hasSelection: () => false,
1130
+ clear: () => {
1131
+ },
1132
+ dispose: () => {
1133
+ }
1134
+ };
1135
+ }
1136
+ const handler = () => {
1137
+ const sel = document.getSelection();
1138
+ if (!sel || sel.rangeCount === 0 || sel.isCollapsed) {
1139
+ setText("");
1140
+ setRect(null);
1141
+ return;
1142
+ }
1143
+ setText(sel.toString());
1144
+ try {
1145
+ const r = sel.getRangeAt(0).getBoundingClientRect();
1146
+ setRect(r.width > 0 || r.height > 0 ? r : null);
1147
+ } catch {
1148
+ setRect(null);
1149
+ }
1150
+ };
1151
+ document.addEventListener("selectionchange", handler);
1152
+ function clear() {
1153
+ const sel = document.getSelection();
1154
+ sel?.removeAllRanges();
1155
+ setText("");
1156
+ setRect(null);
1157
+ }
1158
+ function dispose() {
1159
+ document.removeEventListener("selectionchange", handler);
1160
+ }
1161
+ return {
1162
+ text,
1163
+ rect,
1164
+ hasSelection: () => text().length > 0,
1165
+ clear,
1166
+ dispose
1167
+ };
1168
+ }
1169
+
1170
+ // src/browser/imageLoader.ts
1171
+ function imageLoader(src) {
1172
+ const [status, setStatus] = signal("pending");
1173
+ const [image, setImage] = signal(null);
1174
+ const [width, setWidth] = signal(0);
1175
+ const [height, setHeight] = signal(0);
1176
+ if (typeof Image === "undefined") {
1177
+ return {
1178
+ status,
1179
+ image,
1180
+ width,
1181
+ height,
1182
+ dispose: () => {
1183
+ }
1184
+ };
1185
+ }
1186
+ let current = null;
1187
+ let disposed = false;
1188
+ function start(url) {
1189
+ if (current) {
1190
+ current.onload = null;
1191
+ current.onerror = null;
1192
+ }
1193
+ setStatus("pending");
1194
+ setImage(null);
1195
+ const img = new Image();
1196
+ current = img;
1197
+ img.onload = () => {
1198
+ if (disposed || current !== img) return;
1199
+ setImage(img);
1200
+ setWidth(img.naturalWidth);
1201
+ setHeight(img.naturalHeight);
1202
+ setStatus("loaded");
1203
+ };
1204
+ img.onerror = () => {
1205
+ if (disposed || current !== img) return;
1206
+ setStatus("error");
1207
+ };
1208
+ img.src = url;
1209
+ }
1210
+ if (typeof src === "function") {
1211
+ start(src());
1212
+ } else {
1213
+ start(src);
1214
+ }
1215
+ function dispose() {
1216
+ disposed = true;
1217
+ if (current) {
1218
+ current.onload = null;
1219
+ current.onerror = null;
1220
+ current = null;
1221
+ }
1222
+ }
1223
+ return { status, image, width, height, dispose };
1224
+ }
1225
+
1226
+ export {
1227
+ media,
1228
+ resize,
1229
+ scroll,
1230
+ online,
1231
+ geo,
1232
+ battery,
1233
+ idle,
1234
+ permissions,
1235
+ clipboard,
1236
+ draggable,
1237
+ dropZone,
1238
+ title,
1239
+ colorScheme,
1240
+ formatNumber,
1241
+ formatCurrency,
1242
+ visibility,
1243
+ network,
1244
+ mouse,
1245
+ swipe,
1246
+ windowSize,
1247
+ urlState,
1248
+ broadcast,
1249
+ fullscreen,
1250
+ wakeLock,
1251
+ animationFrame,
1252
+ mutationObserver,
1253
+ bounds,
1254
+ keyboard,
1255
+ speech,
1256
+ gamepad,
1257
+ pointerLock,
1258
+ vibrate,
1259
+ favicon,
1260
+ svgFavicon,
1261
+ textSelection,
1262
+ imageLoader
1263
+ };