sibujs 1.5.0 → 2.1.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 (208) hide show
  1. package/dist/browser.cjs +332 -121
  2. package/dist/browser.d.cts +5 -0
  3. package/dist/browser.d.ts +5 -0
  4. package/dist/browser.js +6 -6
  5. package/dist/build.cjs +1049 -344
  6. package/dist/build.js +15 -13
  7. package/dist/cdn.global.js +17 -16
  8. package/dist/chunk-2RA7SHDA.js +65 -0
  9. package/dist/chunk-2UPRY23K.js +80 -0
  10. package/dist/{chunk-BMPL52BF.js → chunk-3DZP6OIT.js} +118 -66
  11. package/dist/chunk-3JHCYHWN.js +125 -0
  12. package/dist/{chunk-CZUGLNJS.js → chunk-45YP72ZQ.js} +3 -3
  13. package/dist/{chunk-JCDUJN2F.js → chunk-AMK2TYNW.js} +490 -153
  14. package/dist/{chunk-NHUC2QWH.js → chunk-CWBVQML6.js} +1 -1
  15. package/dist/{chunk-XHK6BDAJ.js → chunk-DRUZZAK4.js} +25 -8
  16. package/dist/{chunk-RJ46C3CS.js → chunk-GWWURC5M.js} +71 -20
  17. package/dist/{chunk-3X2YG6YM.js → chunk-JYD2PWXH.js} +59 -28
  18. package/dist/{chunk-2BYQDGN3.js → chunk-KGYT6UO6.js} +234 -63
  19. package/dist/{chunk-5X6PP2UK.js → chunk-LMLD24FC.js} +2 -2
  20. package/dist/{chunk-M4NLBH4I.js → chunk-LYTCUZ7H.js} +3 -2
  21. package/dist/{chunk-XUEEGU5O.js → chunk-NASX6ST2.js} +16 -4
  22. package/dist/{chunk-VQDZK23A.js → chunk-O6EFQ3KT.js} +181 -66
  23. package/dist/{chunk-BGN5ZMP4.js → chunk-OJ3P4ECI.js} +14 -2
  24. package/dist/chunk-ON5MMR2J.js +1327 -0
  25. package/dist/{chunk-SFKNRVCU.js → chunk-P2HSJDDN.js} +135 -79
  26. package/dist/chunk-QO3WC6FS.js +384 -0
  27. package/dist/{chunk-WZSPOOER.js → chunk-RDTDJCAB.js} +8 -5
  28. package/dist/{chunk-7GRNSCFT.js → chunk-TH2ILCYW.js} +312 -185
  29. package/dist/chunk-UCS6AMJ7.js +79 -0
  30. package/dist/{chunk-VAPYJN4X.js → chunk-V6C4FADE.js} +93 -23
  31. package/dist/{chunk-OUZZEE4S.js → chunk-WANSMF2L.js} +17 -11
  32. package/dist/{chunk-23VV7YD3.js → chunk-WIPZPFBQ.js} +25 -30
  33. package/dist/chunk-WZA53FXU.js +149 -0
  34. package/dist/{chunk-BGTHZHJ5.js → chunk-ZAQSMOED.js} +188 -44
  35. package/dist/{customElement-BL3Uo8dL.d.cts → customElement-CPfIrbvg.d.cts} +14 -10
  36. package/dist/{customElement-BL3Uo8dL.d.ts → customElement-CPfIrbvg.d.ts} +14 -10
  37. package/dist/data.cjs +536 -151
  38. package/dist/data.d.cts +20 -2
  39. package/dist/data.d.ts +20 -2
  40. package/dist/data.js +11 -9
  41. package/dist/devtools.cjs +613 -266
  42. package/dist/devtools.d.cts +1 -1
  43. package/dist/devtools.d.ts +1 -1
  44. package/dist/devtools.js +12 -6
  45. package/dist/ecosystem.cjs +602 -197
  46. package/dist/ecosystem.d.cts +9 -7
  47. package/dist/ecosystem.d.ts +9 -7
  48. package/dist/ecosystem.js +12 -11
  49. package/dist/extras.cjs +3500 -1608
  50. package/dist/extras.d.cts +9 -9
  51. package/dist/extras.d.ts +9 -9
  52. package/dist/extras.js +58 -45
  53. package/dist/index.cjs +1055 -344
  54. package/dist/index.d.cts +85 -8
  55. package/dist/index.d.ts +85 -8
  56. package/dist/index.js +32 -16
  57. package/dist/{introspect-BumjnBKr.d.cts → introspect-2TOlQ7oa.d.cts} +25 -3
  58. package/dist/{introspect-CZrlcaYy.d.ts → introspect-DnIpHQQz.d.ts} +25 -3
  59. package/dist/motion.cjs +122 -63
  60. package/dist/motion.js +4 -4
  61. package/dist/patterns.cjs +450 -110
  62. package/dist/patterns.d.cts +11 -12
  63. package/dist/patterns.d.ts +11 -12
  64. package/dist/patterns.js +7 -7
  65. package/dist/performance.cjs +373 -149
  66. package/dist/performance.d.cts +23 -16
  67. package/dist/performance.d.ts +23 -16
  68. package/dist/performance.js +13 -8
  69. package/dist/plugin-D30wlGW5.d.cts +71 -0
  70. package/dist/plugin-D30wlGW5.d.ts +71 -0
  71. package/dist/plugins.cjs +729 -301
  72. package/dist/plugins.d.cts +10 -3
  73. package/dist/plugins.d.ts +10 -3
  74. package/dist/plugins.js +106 -38
  75. package/dist/{ssr-Do_SiVoL.d.cts → ssr-CrVNy6Pa.d.cts} +9 -15
  76. package/dist/{ssr-Do_SiVoL.d.ts → ssr-CrVNy6Pa.d.ts} +9 -15
  77. package/dist/{ssr-4PBXAOO3.js → ssr-FXD2PPMC.js} +4 -3
  78. package/dist/ssr.cjs +736 -274
  79. package/dist/ssr.d.cts +26 -6
  80. package/dist/ssr.d.ts +26 -6
  81. package/dist/ssr.js +12 -11
  82. package/dist/{tagFactory-DaJ0YWX6.d.cts → tagFactory-S17H2qxu.d.cts} +9 -1
  83. package/dist/{tagFactory-DaJ0YWX6.d.ts → tagFactory-S17H2qxu.d.ts} +9 -1
  84. package/dist/testing.cjs +303 -76
  85. package/dist/testing.d.cts +17 -4
  86. package/dist/testing.d.ts +17 -4
  87. package/dist/testing.js +100 -44
  88. package/dist/ui.cjs +589 -178
  89. package/dist/ui.d.cts +1 -1
  90. package/dist/ui.d.ts +1 -1
  91. package/dist/ui.js +20 -17
  92. package/dist/widgets.cjs +1103 -146
  93. package/dist/widgets.d.cts +104 -2
  94. package/dist/widgets.d.ts +104 -2
  95. package/dist/widgets.js +9 -7
  96. package/package.json +8 -2
  97. package/dist/chunk-32DY64NT.js +0 -282
  98. package/dist/chunk-3AIRKM3B.js +0 -1263
  99. package/dist/chunk-3ARAQO7B.js +0 -398
  100. package/dist/chunk-3CRQALYP.js +0 -877
  101. package/dist/chunk-4EI4AG32.js +0 -482
  102. package/dist/chunk-4MYMUBRS.js +0 -21
  103. package/dist/chunk-5ZYQ6KDD.js +0 -154
  104. package/dist/chunk-6BMPXPUW.js +0 -26
  105. package/dist/chunk-6HLLIF3K.js +0 -398
  106. package/dist/chunk-6LSNVCS2.js +0 -937
  107. package/dist/chunk-6SA3QQES.js +0 -61
  108. package/dist/chunk-77L6NL3X.js +0 -1097
  109. package/dist/chunk-7BF6TK55.js +0 -1097
  110. package/dist/chunk-7TQKR4PP.js +0 -294
  111. package/dist/chunk-7V26P53V.js +0 -712
  112. package/dist/chunk-AZ3ISID5.js +0 -298
  113. package/dist/chunk-B7SWRFUT.js +0 -332
  114. package/dist/chunk-BTU3TJDS.js +0 -365
  115. package/dist/chunk-BW3WT46K.js +0 -937
  116. package/dist/chunk-C6KFWOFV.js +0 -616
  117. package/dist/chunk-CHF5OHIA.js +0 -61
  118. package/dist/chunk-CHJ27IGK.js +0 -26
  119. package/dist/chunk-CMBFNA7L.js +0 -27
  120. package/dist/chunk-DAHRH4ON.js +0 -331
  121. package/dist/chunk-DKOHBI74.js +0 -924
  122. package/dist/chunk-DTCOOBMX.js +0 -725
  123. package/dist/chunk-EBGIRKQY.js +0 -616
  124. package/dist/chunk-EUZND3CB.js +0 -27
  125. package/dist/chunk-EVCZO745.js +0 -365
  126. package/dist/chunk-EWFVA3TJ.js +0 -282
  127. package/dist/chunk-F3FA4F32.js +0 -292
  128. package/dist/chunk-FGOEVHY3.js +0 -60
  129. package/dist/chunk-G3BOQPVO.js +0 -365
  130. package/dist/chunk-GCOK2LC3.js +0 -282
  131. package/dist/chunk-GJPXRJ45.js +0 -37
  132. package/dist/chunk-HGMJFBC7.js +0 -654
  133. package/dist/chunk-JAKHTMQU.js +0 -1000
  134. package/dist/chunk-JCI5M6U6.js +0 -956
  135. package/dist/chunk-K4G4ZQNR.js +0 -286
  136. package/dist/chunk-K5ZUMYVS.js +0 -89
  137. package/dist/chunk-KQPDEVVS.js +0 -398
  138. package/dist/chunk-L6JRBDNS.js +0 -60
  139. package/dist/chunk-LA6KQEDU.js +0 -712
  140. package/dist/chunk-MB6QFH3I.js +0 -2776
  141. package/dist/chunk-MDVXJWFN.js +0 -304
  142. package/dist/chunk-MEZVEBPN.js +0 -2008
  143. package/dist/chunk-MK4ERFYL.js +0 -2249
  144. package/dist/chunk-MLKGABMK.js +0 -9
  145. package/dist/chunk-MQ5GOYPH.js +0 -2249
  146. package/dist/chunk-MYRV7VDM.js +0 -742
  147. package/dist/chunk-N6IZB6KJ.js +0 -567
  148. package/dist/chunk-NEKUBFPT.js +0 -60
  149. package/dist/chunk-NMRUZALC.js +0 -1097
  150. package/dist/chunk-NYVAC6P5.js +0 -37
  151. package/dist/chunk-NZIIMDWI.js +0 -84
  152. package/dist/chunk-OF7UZIVB.js +0 -725
  153. package/dist/chunk-P3XWXJZU.js +0 -282
  154. package/dist/chunk-P6W3STU4.js +0 -2249
  155. package/dist/chunk-PBHF5WKN.js +0 -616
  156. package/dist/chunk-PDZQY43A.js +0 -616
  157. package/dist/chunk-PTQJDMRT.js +0 -146
  158. package/dist/chunk-PZEGYCF5.js +0 -61
  159. package/dist/chunk-QBMDLBU2.js +0 -975
  160. package/dist/chunk-QWZG56ET.js +0 -2744
  161. package/dist/chunk-RQGQSLQK.js +0 -725
  162. package/dist/chunk-SDLZDHKP.js +0 -107
  163. package/dist/chunk-TDGZL5CU.js +0 -365
  164. package/dist/chunk-TNQWPPE6.js +0 -37
  165. package/dist/chunk-TSOKIX5Z.js +0 -654
  166. package/dist/chunk-UHNL42EF.js +0 -2730
  167. package/dist/chunk-UNXCEF6S.js +0 -21
  168. package/dist/chunk-V2XTI523.js +0 -347
  169. package/dist/chunk-VAU366PN.js +0 -2241
  170. package/dist/chunk-VMVDTCXB.js +0 -712
  171. package/dist/chunk-VQNQZCWJ.js +0 -61
  172. package/dist/chunk-VRW3FULF.js +0 -725
  173. package/dist/chunk-WADYRCO2.js +0 -304
  174. package/dist/chunk-WILQZRO4.js +0 -282
  175. package/dist/chunk-WR5D4EGH.js +0 -26
  176. package/dist/chunk-WUHJISPP.js +0 -298
  177. package/dist/chunk-XYU6TZOW.js +0 -182
  178. package/dist/chunk-Y6GP4QGG.js +0 -276
  179. package/dist/chunk-YECR7UIA.js +0 -347
  180. package/dist/chunk-YUTWTI4B.js +0 -654
  181. package/dist/chunk-Z65KYU7I.js +0 -26
  182. package/dist/chunk-Z6POF5YC.js +0 -975
  183. package/dist/chunk-ZBJP6WFL.js +0 -482
  184. package/dist/chunk-ZD6OAMTH.js +0 -277
  185. package/dist/chunk-ZWKZCBO6.js +0 -317
  186. package/dist/contracts-DDrwxvJ-.d.cts +0 -245
  187. package/dist/contracts-DDrwxvJ-.d.ts +0 -245
  188. package/dist/contracts-DOrhwbke.d.cts +0 -245
  189. package/dist/contracts-DOrhwbke.d.ts +0 -245
  190. package/dist/contracts-xo5ckdRP.d.cts +0 -240
  191. package/dist/contracts-xo5ckdRP.d.ts +0 -240
  192. package/dist/customElement-BKQfbSZQ.d.cts +0 -262
  193. package/dist/customElement-BKQfbSZQ.d.ts +0 -262
  194. package/dist/customElement-D2DJp_xn.d.cts +0 -313
  195. package/dist/customElement-D2DJp_xn.d.ts +0 -313
  196. package/dist/customElement-yz8uyk-0.d.cts +0 -308
  197. package/dist/customElement-yz8uyk-0.d.ts +0 -308
  198. package/dist/introspect-Cb0zgpi2.d.cts +0 -477
  199. package/dist/introspect-Y2xNXGSf.d.ts +0 -477
  200. package/dist/plugin-Bek4RhJY.d.cts +0 -43
  201. package/dist/plugin-Bek4RhJY.d.ts +0 -43
  202. package/dist/ssr-3RXHP5ES.js +0 -38
  203. package/dist/ssr-6GIMY5MX.js +0 -38
  204. package/dist/ssr-BA6sxxUd.d.cts +0 -135
  205. package/dist/ssr-BA6sxxUd.d.ts +0 -135
  206. package/dist/ssr-WKUPVSSK.js +0 -36
  207. package/dist/tagFactory-Dl8QCLga.d.cts +0 -23
  208. package/dist/tagFactory-Dl8QCLga.d.ts +0 -23
@@ -1,1263 +0,0 @@
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
- };