sibujs 1.1.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.
- package/README.md +29 -25
- package/dist/browser.cjs +804 -2
- package/dist/browser.d.cts +591 -1
- package/dist/browser.d.ts +591 -1
- package/dist/browser.js +50 -8
- package/dist/build.cjs +706 -161
- package/dist/build.js +21 -12
- package/dist/cdn.global.js +188 -7
- package/dist/chunk-2BYQDGN3.js +742 -0
- package/dist/chunk-32DY64NT.js +282 -0
- package/dist/chunk-3AIRKM3B.js +1263 -0
- package/dist/chunk-3X2YG6YM.js +505 -0
- package/dist/chunk-5X6PP2UK.js +28 -0
- package/dist/chunk-77L6NL3X.js +1097 -0
- package/dist/chunk-B7SWRFUT.js +332 -0
- package/dist/chunk-BGN5ZMP4.js +26 -0
- package/dist/chunk-BTU3TJDS.js +365 -0
- package/dist/chunk-CHF5OHIA.js +61 -0
- package/dist/chunk-CMBFNA7L.js +27 -0
- package/dist/chunk-DAHRH4ON.js +331 -0
- package/dist/chunk-EBGIRKQY.js +616 -0
- package/dist/chunk-EUZND3CB.js +27 -0
- package/dist/chunk-F3FA4F32.js +292 -0
- package/dist/chunk-GCOK2LC3.js +282 -0
- package/dist/chunk-JAKHTMQU.js +1000 -0
- package/dist/chunk-JCI5M6U6.js +956 -0
- package/dist/chunk-KQPDEVVS.js +398 -0
- package/dist/chunk-NEKUBFPT.js +60 -0
- package/dist/chunk-NYVAC6P5.js +37 -0
- package/dist/chunk-OUZZEE4S.js +365 -0
- package/dist/chunk-P6W3STU4.js +2249 -0
- package/dist/chunk-PTQJDMRT.js +146 -0
- package/dist/chunk-QWZG56ET.js +2744 -0
- package/dist/chunk-TSOKIX5Z.js +654 -0
- package/dist/chunk-VMVDTCXB.js +712 -0
- package/dist/chunk-VRW3FULF.js +725 -0
- package/dist/chunk-WZSPOOER.js +84 -0
- package/dist/chunk-YT6HQ6AM.js +14 -0
- package/dist/chunk-ZD6OAMTH.js +277 -0
- package/dist/contracts-DDrwxvJ-.d.cts +245 -0
- package/dist/contracts-DDrwxvJ-.d.ts +245 -0
- package/dist/data.cjs +35 -2
- package/dist/data.d.cts +7 -0
- package/dist/data.d.ts +7 -0
- package/dist/data.js +9 -8
- package/dist/devtools.cjs +122 -0
- package/dist/devtools.d.cts +69 -461
- package/dist/devtools.d.ts +69 -461
- package/dist/devtools.js +127 -6
- package/dist/ecosystem.cjs +68 -23
- package/dist/ecosystem.d.cts +1 -1
- package/dist/ecosystem.d.ts +1 -1
- package/dist/ecosystem.js +10 -9
- package/dist/extras.cjs +1252 -82
- package/dist/extras.d.cts +5 -5
- package/dist/extras.d.ts +5 -5
- package/dist/extras.js +69 -24
- package/dist/index.cjs +708 -161
- package/dist/index.d.cts +397 -17
- package/dist/index.d.ts +397 -17
- package/dist/index.js +39 -17
- package/dist/introspect-BumjnBKr.d.cts +477 -0
- package/dist/introspect-CZrlcaYy.d.ts +477 -0
- package/dist/introspect-Cb0zgpi2.d.cts +477 -0
- package/dist/introspect-Y2xNXGSf.d.ts +477 -0
- package/dist/motion.js +4 -4
- package/dist/patterns.cjs +51 -2
- package/dist/patterns.d.cts +18 -8
- package/dist/patterns.d.ts +18 -8
- package/dist/patterns.js +7 -7
- package/dist/performance.js +4 -4
- package/dist/plugins.cjs +473 -98
- package/dist/plugins.d.cts +27 -4
- package/dist/plugins.d.ts +27 -4
- package/dist/plugins.js +156 -37
- package/dist/ssr-4PBXAOO3.js +40 -0
- package/dist/ssr-Do_SiVoL.d.cts +201 -0
- package/dist/ssr-Do_SiVoL.d.ts +201 -0
- package/dist/ssr.cjs +357 -77
- package/dist/ssr.d.cts +10 -1
- package/dist/ssr.d.ts +10 -1
- package/dist/ssr.js +13 -10
- package/dist/tagFactory-DaJ0YWX6.d.cts +47 -0
- package/dist/tagFactory-DaJ0YWX6.d.ts +47 -0
- package/dist/testing.cjs +233 -2
- package/dist/testing.d.cts +42 -1
- package/dist/testing.d.ts +42 -1
- package/dist/testing.js +129 -2
- package/dist/ui.cjs +374 -3
- package/dist/ui.d.cts +252 -2
- package/dist/ui.d.ts +252 -2
- package/dist/ui.js +328 -8
- package/dist/widgets.js +7 -7
- package/package.json +1 -1
package/dist/browser.cjs
CHANGED
|
@@ -20,21 +20,42 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
|
|
|
20
20
|
// browser.ts
|
|
21
21
|
var browser_exports = {};
|
|
22
22
|
__export(browser_exports, {
|
|
23
|
+
animationFrame: () => animationFrame,
|
|
23
24
|
battery: () => battery,
|
|
25
|
+
bounds: () => bounds,
|
|
26
|
+
broadcast: () => broadcast,
|
|
24
27
|
clipboard: () => clipboard,
|
|
25
28
|
colorScheme: () => colorScheme,
|
|
26
29
|
draggable: () => draggable,
|
|
27
30
|
dropZone: () => dropZone,
|
|
31
|
+
favicon: () => favicon,
|
|
28
32
|
formatCurrency: () => formatCurrency,
|
|
29
33
|
formatNumber: () => formatNumber,
|
|
34
|
+
fullscreen: () => fullscreen,
|
|
35
|
+
gamepad: () => gamepad,
|
|
30
36
|
geo: () => geo,
|
|
31
37
|
idle: () => idle,
|
|
38
|
+
imageLoader: () => imageLoader,
|
|
39
|
+
keyboard: () => keyboard,
|
|
32
40
|
media: () => media,
|
|
41
|
+
mouse: () => mouse,
|
|
42
|
+
mutationObserver: () => mutationObserver,
|
|
43
|
+
network: () => network,
|
|
33
44
|
online: () => online,
|
|
34
45
|
permissions: () => permissions,
|
|
46
|
+
pointerLock: () => pointerLock,
|
|
35
47
|
resize: () => resize,
|
|
36
48
|
scroll: () => scroll,
|
|
37
|
-
|
|
49
|
+
speech: () => speech,
|
|
50
|
+
svgFavicon: () => svgFavicon,
|
|
51
|
+
swipe: () => swipe,
|
|
52
|
+
textSelection: () => textSelection,
|
|
53
|
+
title: () => title,
|
|
54
|
+
urlState: () => urlState,
|
|
55
|
+
vibrate: () => vibrate,
|
|
56
|
+
visibility: () => visibility,
|
|
57
|
+
wakeLock: () => wakeLock,
|
|
58
|
+
windowSize: () => windowSize
|
|
38
59
|
});
|
|
39
60
|
module.exports = __toCommonJS(browser_exports);
|
|
40
61
|
|
|
@@ -843,21 +864,802 @@ function formatCurrency(value, currency, options) {
|
|
|
843
864
|
...formatOptions
|
|
844
865
|
}).format(value);
|
|
845
866
|
}
|
|
867
|
+
|
|
868
|
+
// src/browser/visibility.ts
|
|
869
|
+
function visibility() {
|
|
870
|
+
if (typeof document === "undefined") {
|
|
871
|
+
const [visible2] = signal(true);
|
|
872
|
+
return { visible: visible2, dispose: () => {
|
|
873
|
+
} };
|
|
874
|
+
}
|
|
875
|
+
const [visible, setVisible] = signal(!document.hidden);
|
|
876
|
+
const handler = () => setVisible(!document.hidden);
|
|
877
|
+
document.addEventListener("visibilitychange", handler);
|
|
878
|
+
function dispose() {
|
|
879
|
+
document.removeEventListener("visibilitychange", handler);
|
|
880
|
+
}
|
|
881
|
+
return { visible, dispose };
|
|
882
|
+
}
|
|
883
|
+
|
|
884
|
+
// src/browser/network.ts
|
|
885
|
+
function network() {
|
|
886
|
+
const connection = typeof navigator !== "undefined" ? navigator.connection ?? navigator.mozConnection ?? navigator.webkitConnection : void 0;
|
|
887
|
+
const [effectiveType, setEffectiveType] = signal(connection?.effectiveType ?? "unknown");
|
|
888
|
+
const [downlink, setDownlink] = signal(connection?.downlink ?? 0);
|
|
889
|
+
const [rtt, setRtt] = signal(connection?.rtt ?? 0);
|
|
890
|
+
const [saveData, setSaveData] = signal(connection?.saveData ?? false);
|
|
891
|
+
if (!connection) {
|
|
892
|
+
return { effectiveType, downlink, rtt, saveData, dispose: () => {
|
|
893
|
+
} };
|
|
894
|
+
}
|
|
895
|
+
const update = () => {
|
|
896
|
+
setEffectiveType(connection.effectiveType ?? "unknown");
|
|
897
|
+
setDownlink(connection.downlink ?? 0);
|
|
898
|
+
setRtt(connection.rtt ?? 0);
|
|
899
|
+
setSaveData(connection.saveData ?? false);
|
|
900
|
+
};
|
|
901
|
+
connection.addEventListener("change", update);
|
|
902
|
+
function dispose() {
|
|
903
|
+
connection?.removeEventListener("change", update);
|
|
904
|
+
}
|
|
905
|
+
return { effectiveType, downlink, rtt, saveData, dispose };
|
|
906
|
+
}
|
|
907
|
+
|
|
908
|
+
// src/browser/mouse.ts
|
|
909
|
+
function mouse(options = {}) {
|
|
910
|
+
const [x, setX] = signal(0);
|
|
911
|
+
const [y, setY] = signal(0);
|
|
912
|
+
if (typeof window === "undefined") {
|
|
913
|
+
return { x, y, dispose: () => {
|
|
914
|
+
} };
|
|
915
|
+
}
|
|
916
|
+
const target = options.target ?? window;
|
|
917
|
+
const trackTouch = options.touch ?? true;
|
|
918
|
+
const onMove = (e) => {
|
|
919
|
+
setX(e.clientX);
|
|
920
|
+
setY(e.clientY);
|
|
921
|
+
};
|
|
922
|
+
const onTouchMove = (e) => {
|
|
923
|
+
if (e.touches.length === 0) return;
|
|
924
|
+
setX(e.touches[0].clientX);
|
|
925
|
+
setY(e.touches[0].clientY);
|
|
926
|
+
};
|
|
927
|
+
target.addEventListener("mousemove", onMove, { passive: true });
|
|
928
|
+
if (trackTouch) {
|
|
929
|
+
target.addEventListener("touchmove", onTouchMove, { passive: true });
|
|
930
|
+
}
|
|
931
|
+
function dispose() {
|
|
932
|
+
target.removeEventListener("mousemove", onMove);
|
|
933
|
+
if (trackTouch) {
|
|
934
|
+
target.removeEventListener("touchmove", onTouchMove);
|
|
935
|
+
}
|
|
936
|
+
}
|
|
937
|
+
return { x, y, dispose };
|
|
938
|
+
}
|
|
939
|
+
|
|
940
|
+
// src/browser/swipe.ts
|
|
941
|
+
function swipe(target, options = {}) {
|
|
942
|
+
const threshold = options.threshold ?? 50;
|
|
943
|
+
const [direction, setDirection] = signal(null);
|
|
944
|
+
if (typeof window === "undefined") {
|
|
945
|
+
return { direction, dispose: () => {
|
|
946
|
+
} };
|
|
947
|
+
}
|
|
948
|
+
let startX = 0;
|
|
949
|
+
let startY = 0;
|
|
950
|
+
let tracking = false;
|
|
951
|
+
const onStart = (e) => {
|
|
952
|
+
if (e.touches.length === 0) return;
|
|
953
|
+
startX = e.touches[0].clientX;
|
|
954
|
+
startY = e.touches[0].clientY;
|
|
955
|
+
tracking = true;
|
|
956
|
+
};
|
|
957
|
+
const onEnd = (e) => {
|
|
958
|
+
if (!tracking) return;
|
|
959
|
+
tracking = false;
|
|
960
|
+
const touch = e.changedTouches[0];
|
|
961
|
+
if (!touch) return;
|
|
962
|
+
const dx = touch.clientX - startX;
|
|
963
|
+
const dy = touch.clientY - startY;
|
|
964
|
+
const absX = Math.abs(dx);
|
|
965
|
+
const absY = Math.abs(dy);
|
|
966
|
+
if (Math.max(absX, absY) < threshold) return;
|
|
967
|
+
let dir;
|
|
968
|
+
if (absX > absY) {
|
|
969
|
+
dir = dx > 0 ? "right" : "left";
|
|
970
|
+
} else {
|
|
971
|
+
dir = dy > 0 ? "down" : "up";
|
|
972
|
+
}
|
|
973
|
+
setDirection(dir);
|
|
974
|
+
options.onSwipe?.(dir, Math.max(absX, absY));
|
|
975
|
+
};
|
|
976
|
+
target.addEventListener("touchstart", onStart, { passive: true });
|
|
977
|
+
target.addEventListener("touchend", onEnd, { passive: true });
|
|
978
|
+
function dispose() {
|
|
979
|
+
target.removeEventListener("touchstart", onStart);
|
|
980
|
+
target.removeEventListener("touchend", onEnd);
|
|
981
|
+
}
|
|
982
|
+
return { direction, dispose };
|
|
983
|
+
}
|
|
984
|
+
|
|
985
|
+
// src/browser/windowSize.ts
|
|
986
|
+
function windowSize() {
|
|
987
|
+
if (typeof window === "undefined") {
|
|
988
|
+
const [width2] = signal(0);
|
|
989
|
+
const [height2] = signal(0);
|
|
990
|
+
return { width: width2, height: height2, dispose: () => {
|
|
991
|
+
} };
|
|
992
|
+
}
|
|
993
|
+
const [width, setWidth] = signal(window.innerWidth);
|
|
994
|
+
const [height, setHeight] = signal(window.innerHeight);
|
|
995
|
+
const handler = () => {
|
|
996
|
+
setWidth(window.innerWidth);
|
|
997
|
+
setHeight(window.innerHeight);
|
|
998
|
+
};
|
|
999
|
+
window.addEventListener("resize", handler, { passive: true });
|
|
1000
|
+
function dispose() {
|
|
1001
|
+
window.removeEventListener("resize", handler);
|
|
1002
|
+
}
|
|
1003
|
+
return { width, height, dispose };
|
|
1004
|
+
}
|
|
1005
|
+
|
|
1006
|
+
// src/browser/urlState.ts
|
|
1007
|
+
function urlState() {
|
|
1008
|
+
if (typeof window === "undefined") {
|
|
1009
|
+
const [params2] = signal(new URLSearchParams());
|
|
1010
|
+
const [hash2] = signal("");
|
|
1011
|
+
return {
|
|
1012
|
+
params: params2,
|
|
1013
|
+
hash: hash2,
|
|
1014
|
+
setParams: () => {
|
|
1015
|
+
},
|
|
1016
|
+
setHash: () => {
|
|
1017
|
+
},
|
|
1018
|
+
dispose: () => {
|
|
1019
|
+
}
|
|
1020
|
+
};
|
|
1021
|
+
}
|
|
1022
|
+
const [params, setParamsSignal] = signal(new URLSearchParams(window.location.search));
|
|
1023
|
+
const [hash, setHashSignal] = signal(window.location.hash);
|
|
1024
|
+
const syncFromLocation = () => {
|
|
1025
|
+
setParamsSignal(new URLSearchParams(window.location.search));
|
|
1026
|
+
setHashSignal(window.location.hash);
|
|
1027
|
+
};
|
|
1028
|
+
const onPopState = () => syncFromLocation();
|
|
1029
|
+
window.addEventListener("popstate", onPopState);
|
|
1030
|
+
function setParams(next, opts = {}) {
|
|
1031
|
+
const p = next instanceof URLSearchParams ? next : new URLSearchParams(next);
|
|
1032
|
+
const query = p.toString();
|
|
1033
|
+
const newUrl = `${window.location.pathname}${query ? `?${query}` : ""}${window.location.hash}`;
|
|
1034
|
+
if (opts.replace) window.history.replaceState(null, "", newUrl);
|
|
1035
|
+
else window.history.pushState(null, "", newUrl);
|
|
1036
|
+
setParamsSignal(new URLSearchParams(p));
|
|
1037
|
+
}
|
|
1038
|
+
function setHash(next, opts = {}) {
|
|
1039
|
+
const normalized = next.startsWith("#") ? next : next ? `#${next}` : "";
|
|
1040
|
+
const newUrl = `${window.location.pathname}${window.location.search}${normalized}`;
|
|
1041
|
+
if (opts.replace) window.history.replaceState(null, "", newUrl);
|
|
1042
|
+
else window.history.pushState(null, "", newUrl);
|
|
1043
|
+
setHashSignal(normalized);
|
|
1044
|
+
}
|
|
1045
|
+
function dispose() {
|
|
1046
|
+
window.removeEventListener("popstate", onPopState);
|
|
1047
|
+
}
|
|
1048
|
+
return { params, hash, setParams, setHash, dispose };
|
|
1049
|
+
}
|
|
1050
|
+
|
|
1051
|
+
// src/browser/broadcast.ts
|
|
1052
|
+
function broadcast(channelName) {
|
|
1053
|
+
if (typeof BroadcastChannel === "undefined") {
|
|
1054
|
+
const [last2] = signal(null);
|
|
1055
|
+
return { last: last2, post: () => {
|
|
1056
|
+
}, dispose: () => {
|
|
1057
|
+
} };
|
|
1058
|
+
}
|
|
1059
|
+
const [last, setLast] = signal(null);
|
|
1060
|
+
const channel = new BroadcastChannel(channelName);
|
|
1061
|
+
const handler = (ev) => setLast(ev.data);
|
|
1062
|
+
channel.addEventListener("message", handler);
|
|
1063
|
+
function post(message) {
|
|
1064
|
+
channel.postMessage(message);
|
|
1065
|
+
}
|
|
1066
|
+
function dispose() {
|
|
1067
|
+
channel.removeEventListener("message", handler);
|
|
1068
|
+
channel.close();
|
|
1069
|
+
}
|
|
1070
|
+
return { last, post, dispose };
|
|
1071
|
+
}
|
|
1072
|
+
|
|
1073
|
+
// src/browser/fullscreen.ts
|
|
1074
|
+
function fullscreen() {
|
|
1075
|
+
if (typeof document === "undefined") {
|
|
1076
|
+
const [isFullscreen2] = signal(false);
|
|
1077
|
+
const [element2] = signal(null);
|
|
1078
|
+
return {
|
|
1079
|
+
isFullscreen: isFullscreen2,
|
|
1080
|
+
element: element2,
|
|
1081
|
+
enter: async () => {
|
|
1082
|
+
},
|
|
1083
|
+
exit: async () => {
|
|
1084
|
+
},
|
|
1085
|
+
toggle: async () => {
|
|
1086
|
+
},
|
|
1087
|
+
dispose: () => {
|
|
1088
|
+
}
|
|
1089
|
+
};
|
|
1090
|
+
}
|
|
1091
|
+
const [isFullscreen, setIsFullscreen] = signal(!!document.fullscreenElement);
|
|
1092
|
+
const [element, setElement] = signal(document.fullscreenElement);
|
|
1093
|
+
const handler = () => {
|
|
1094
|
+
setIsFullscreen(!!document.fullscreenElement);
|
|
1095
|
+
setElement(document.fullscreenElement);
|
|
1096
|
+
};
|
|
1097
|
+
document.addEventListener("fullscreenchange", handler);
|
|
1098
|
+
async function enter(el) {
|
|
1099
|
+
if (!document.fullscreenElement && el.requestFullscreen) {
|
|
1100
|
+
await el.requestFullscreen();
|
|
1101
|
+
}
|
|
1102
|
+
}
|
|
1103
|
+
async function exit() {
|
|
1104
|
+
if (document.fullscreenElement && document.exitFullscreen) {
|
|
1105
|
+
await document.exitFullscreen();
|
|
1106
|
+
}
|
|
1107
|
+
}
|
|
1108
|
+
async function toggle(el) {
|
|
1109
|
+
if (document.fullscreenElement) await exit();
|
|
1110
|
+
else await enter(el);
|
|
1111
|
+
}
|
|
1112
|
+
function dispose() {
|
|
1113
|
+
document.removeEventListener("fullscreenchange", handler);
|
|
1114
|
+
}
|
|
1115
|
+
return { isFullscreen, element, enter, exit, toggle, dispose };
|
|
1116
|
+
}
|
|
1117
|
+
|
|
1118
|
+
// src/browser/wakeLock.ts
|
|
1119
|
+
function wakeLock() {
|
|
1120
|
+
const [active, setActive] = signal(false);
|
|
1121
|
+
if (typeof navigator === "undefined" || !("wakeLock" in navigator) || typeof document === "undefined") {
|
|
1122
|
+
return {
|
|
1123
|
+
active,
|
|
1124
|
+
request: async () => {
|
|
1125
|
+
},
|
|
1126
|
+
release: async () => {
|
|
1127
|
+
},
|
|
1128
|
+
dispose: () => {
|
|
1129
|
+
}
|
|
1130
|
+
};
|
|
1131
|
+
}
|
|
1132
|
+
const api = navigator.wakeLock;
|
|
1133
|
+
let sentinel = null;
|
|
1134
|
+
async function request() {
|
|
1135
|
+
try {
|
|
1136
|
+
sentinel = await api.request("screen");
|
|
1137
|
+
setActive(true);
|
|
1138
|
+
sentinel.addEventListener("release", () => {
|
|
1139
|
+
setActive(false);
|
|
1140
|
+
});
|
|
1141
|
+
} catch {
|
|
1142
|
+
setActive(false);
|
|
1143
|
+
}
|
|
1144
|
+
}
|
|
1145
|
+
async function release() {
|
|
1146
|
+
if (sentinel && !sentinel.released) {
|
|
1147
|
+
await sentinel.release();
|
|
1148
|
+
}
|
|
1149
|
+
sentinel = null;
|
|
1150
|
+
setActive(false);
|
|
1151
|
+
}
|
|
1152
|
+
const onVisibility = () => {
|
|
1153
|
+
if (sentinel?.released && !document.hidden) {
|
|
1154
|
+
void request();
|
|
1155
|
+
}
|
|
1156
|
+
};
|
|
1157
|
+
document.addEventListener("visibilitychange", onVisibility);
|
|
1158
|
+
function dispose() {
|
|
1159
|
+
document.removeEventListener("visibilitychange", onVisibility);
|
|
1160
|
+
void release();
|
|
1161
|
+
}
|
|
1162
|
+
return { active, request, release, dispose };
|
|
1163
|
+
}
|
|
1164
|
+
|
|
1165
|
+
// src/browser/animationFrame.ts
|
|
1166
|
+
function animationFrame(options = {}) {
|
|
1167
|
+
const [delta, setDelta] = signal(0);
|
|
1168
|
+
const [elapsed, setElapsed] = signal(0);
|
|
1169
|
+
const [running, setRunning] = signal(false);
|
|
1170
|
+
if (typeof requestAnimationFrame === "undefined") {
|
|
1171
|
+
return {
|
|
1172
|
+
delta,
|
|
1173
|
+
elapsed,
|
|
1174
|
+
running,
|
|
1175
|
+
pause: () => {
|
|
1176
|
+
},
|
|
1177
|
+
resume: () => {
|
|
1178
|
+
},
|
|
1179
|
+
dispose: () => {
|
|
1180
|
+
}
|
|
1181
|
+
};
|
|
1182
|
+
}
|
|
1183
|
+
let id = null;
|
|
1184
|
+
let prev = -1;
|
|
1185
|
+
let start = -1;
|
|
1186
|
+
const minFrameMs = options.fpsLimit ? 1e3 / options.fpsLimit : 0;
|
|
1187
|
+
const step = (now) => {
|
|
1188
|
+
if (start < 0) start = now;
|
|
1189
|
+
const firstTick = prev < 0;
|
|
1190
|
+
const dt = firstTick ? 0 : now - prev;
|
|
1191
|
+
if (firstTick || dt >= minFrameMs) {
|
|
1192
|
+
setDelta(dt);
|
|
1193
|
+
setElapsed(now - start);
|
|
1194
|
+
prev = now;
|
|
1195
|
+
}
|
|
1196
|
+
id = requestAnimationFrame(step);
|
|
1197
|
+
};
|
|
1198
|
+
function resume() {
|
|
1199
|
+
if (id !== null) return;
|
|
1200
|
+
setRunning(true);
|
|
1201
|
+
id = requestAnimationFrame(step);
|
|
1202
|
+
}
|
|
1203
|
+
function pause() {
|
|
1204
|
+
if (id !== null) {
|
|
1205
|
+
cancelAnimationFrame(id);
|
|
1206
|
+
id = null;
|
|
1207
|
+
}
|
|
1208
|
+
setRunning(false);
|
|
1209
|
+
prev = -1;
|
|
1210
|
+
start = -1;
|
|
1211
|
+
}
|
|
1212
|
+
function dispose() {
|
|
1213
|
+
pause();
|
|
1214
|
+
}
|
|
1215
|
+
if (options.immediate !== false) resume();
|
|
1216
|
+
return { delta, elapsed, running, pause, resume, dispose };
|
|
1217
|
+
}
|
|
1218
|
+
|
|
1219
|
+
// src/browser/mutationObserver.ts
|
|
1220
|
+
function mutationObserver(target, options = { childList: true, subtree: true }) {
|
|
1221
|
+
const [records, setRecords] = signal([]);
|
|
1222
|
+
if (typeof MutationObserver === "undefined") {
|
|
1223
|
+
return { records, dispose: () => {
|
|
1224
|
+
} };
|
|
1225
|
+
}
|
|
1226
|
+
const observer = new MutationObserver((batch2) => {
|
|
1227
|
+
setRecords(batch2);
|
|
1228
|
+
});
|
|
1229
|
+
observer.observe(target, options);
|
|
1230
|
+
function dispose() {
|
|
1231
|
+
observer.disconnect();
|
|
1232
|
+
}
|
|
1233
|
+
return { records, dispose };
|
|
1234
|
+
}
|
|
1235
|
+
|
|
1236
|
+
// src/browser/bounds.ts
|
|
1237
|
+
var ZERO = {
|
|
1238
|
+
x: 0,
|
|
1239
|
+
y: 0,
|
|
1240
|
+
width: 0,
|
|
1241
|
+
height: 0,
|
|
1242
|
+
top: 0,
|
|
1243
|
+
left: 0,
|
|
1244
|
+
right: 0,
|
|
1245
|
+
bottom: 0
|
|
1246
|
+
};
|
|
1247
|
+
function readRect(el) {
|
|
1248
|
+
const r = el.getBoundingClientRect();
|
|
1249
|
+
return {
|
|
1250
|
+
x: r.x,
|
|
1251
|
+
y: r.y,
|
|
1252
|
+
width: r.width,
|
|
1253
|
+
height: r.height,
|
|
1254
|
+
top: r.top,
|
|
1255
|
+
left: r.left,
|
|
1256
|
+
right: r.right,
|
|
1257
|
+
bottom: r.bottom
|
|
1258
|
+
};
|
|
1259
|
+
}
|
|
1260
|
+
function bounds(target) {
|
|
1261
|
+
const [rect, setRect] = signal(ZERO);
|
|
1262
|
+
if (typeof window === "undefined" || !target) {
|
|
1263
|
+
return {
|
|
1264
|
+
rect,
|
|
1265
|
+
refresh: () => {
|
|
1266
|
+
},
|
|
1267
|
+
dispose: () => {
|
|
1268
|
+
}
|
|
1269
|
+
};
|
|
1270
|
+
}
|
|
1271
|
+
function refresh() {
|
|
1272
|
+
setRect(readRect(target));
|
|
1273
|
+
}
|
|
1274
|
+
refresh();
|
|
1275
|
+
let resizeObserver = null;
|
|
1276
|
+
if (typeof ResizeObserver !== "undefined") {
|
|
1277
|
+
resizeObserver = new ResizeObserver(refresh);
|
|
1278
|
+
resizeObserver.observe(target);
|
|
1279
|
+
}
|
|
1280
|
+
const onScroll = () => refresh();
|
|
1281
|
+
window.addEventListener("scroll", onScroll, { passive: true, capture: true });
|
|
1282
|
+
function dispose() {
|
|
1283
|
+
resizeObserver?.disconnect();
|
|
1284
|
+
window.removeEventListener("scroll", onScroll, { capture: true });
|
|
1285
|
+
}
|
|
1286
|
+
return { rect, refresh, dispose };
|
|
1287
|
+
}
|
|
1288
|
+
|
|
1289
|
+
// src/browser/keyboard.ts
|
|
1290
|
+
function keyboard(options = {}) {
|
|
1291
|
+
const [pressed, setPressed] = signal(/* @__PURE__ */ new Set());
|
|
1292
|
+
if (typeof window === "undefined") {
|
|
1293
|
+
return {
|
|
1294
|
+
pressed,
|
|
1295
|
+
isPressed: () => false,
|
|
1296
|
+
dispose: () => {
|
|
1297
|
+
}
|
|
1298
|
+
};
|
|
1299
|
+
}
|
|
1300
|
+
const target = options.target ?? window;
|
|
1301
|
+
const filter = options.keys ? new Set(options.keys) : null;
|
|
1302
|
+
const onDown = (e) => {
|
|
1303
|
+
if (filter && !filter.has(e.key)) return;
|
|
1304
|
+
setPressed((prev) => {
|
|
1305
|
+
if (prev.has(e.key)) return prev;
|
|
1306
|
+
const next = new Set(prev);
|
|
1307
|
+
next.add(e.key);
|
|
1308
|
+
return next;
|
|
1309
|
+
});
|
|
1310
|
+
};
|
|
1311
|
+
const onUp = (e) => {
|
|
1312
|
+
if (filter && !filter.has(e.key)) return;
|
|
1313
|
+
setPressed((prev) => {
|
|
1314
|
+
if (!prev.has(e.key)) return prev;
|
|
1315
|
+
const next = new Set(prev);
|
|
1316
|
+
next.delete(e.key);
|
|
1317
|
+
return next;
|
|
1318
|
+
});
|
|
1319
|
+
};
|
|
1320
|
+
const onBlur = () => setPressed(/* @__PURE__ */ new Set());
|
|
1321
|
+
target.addEventListener("keydown", onDown);
|
|
1322
|
+
target.addEventListener("keyup", onUp);
|
|
1323
|
+
window.addEventListener("blur", onBlur);
|
|
1324
|
+
function isPressed(key) {
|
|
1325
|
+
return pressed().has(key);
|
|
1326
|
+
}
|
|
1327
|
+
function dispose() {
|
|
1328
|
+
target.removeEventListener("keydown", onDown);
|
|
1329
|
+
target.removeEventListener("keyup", onUp);
|
|
1330
|
+
window.removeEventListener("blur", onBlur);
|
|
1331
|
+
}
|
|
1332
|
+
return { pressed, isPressed, dispose };
|
|
1333
|
+
}
|
|
1334
|
+
|
|
1335
|
+
// src/browser/speech.ts
|
|
1336
|
+
function speech() {
|
|
1337
|
+
const [speaking, setSpeaking] = signal(false);
|
|
1338
|
+
const [paused, setPaused] = signal(false);
|
|
1339
|
+
if (typeof window === "undefined" || typeof window.speechSynthesis === "undefined") {
|
|
1340
|
+
return {
|
|
1341
|
+
speaking,
|
|
1342
|
+
paused,
|
|
1343
|
+
speak: () => {
|
|
1344
|
+
},
|
|
1345
|
+
pause: () => {
|
|
1346
|
+
},
|
|
1347
|
+
resume: () => {
|
|
1348
|
+
},
|
|
1349
|
+
cancel: () => {
|
|
1350
|
+
},
|
|
1351
|
+
dispose: () => {
|
|
1352
|
+
}
|
|
1353
|
+
};
|
|
1354
|
+
}
|
|
1355
|
+
const synth = window.speechSynthesis;
|
|
1356
|
+
const interval = setInterval(() => {
|
|
1357
|
+
setSpeaking(synth.speaking);
|
|
1358
|
+
setPaused(synth.paused);
|
|
1359
|
+
}, 200);
|
|
1360
|
+
function speak(text, options = {}) {
|
|
1361
|
+
const u = new SpeechSynthesisUtterance(text);
|
|
1362
|
+
if (options.lang) u.lang = options.lang;
|
|
1363
|
+
if (options.rate != null) u.rate = options.rate;
|
|
1364
|
+
if (options.pitch != null) u.pitch = options.pitch;
|
|
1365
|
+
if (options.volume != null) u.volume = options.volume;
|
|
1366
|
+
if (options.voice) {
|
|
1367
|
+
const voices = synth.getVoices();
|
|
1368
|
+
const match = voices.find((v) => v.name === options.voice);
|
|
1369
|
+
if (match) u.voice = match;
|
|
1370
|
+
}
|
|
1371
|
+
u.addEventListener("start", () => setSpeaking(true));
|
|
1372
|
+
u.addEventListener("end", () => {
|
|
1373
|
+
setSpeaking(false);
|
|
1374
|
+
setPaused(false);
|
|
1375
|
+
});
|
|
1376
|
+
u.addEventListener("error", () => {
|
|
1377
|
+
setSpeaking(false);
|
|
1378
|
+
setPaused(false);
|
|
1379
|
+
});
|
|
1380
|
+
synth.speak(u);
|
|
1381
|
+
}
|
|
1382
|
+
function dispose() {
|
|
1383
|
+
clearInterval(interval);
|
|
1384
|
+
synth.cancel();
|
|
1385
|
+
}
|
|
1386
|
+
return {
|
|
1387
|
+
speaking,
|
|
1388
|
+
paused,
|
|
1389
|
+
speak,
|
|
1390
|
+
pause: () => synth.pause(),
|
|
1391
|
+
resume: () => synth.resume(),
|
|
1392
|
+
cancel: () => synth.cancel(),
|
|
1393
|
+
dispose
|
|
1394
|
+
};
|
|
1395
|
+
}
|
|
1396
|
+
|
|
1397
|
+
// src/browser/gamepad.ts
|
|
1398
|
+
function gamepad() {
|
|
1399
|
+
const [pads, setPads] = signal([]);
|
|
1400
|
+
if (typeof window === "undefined" || typeof navigator === "undefined" || typeof navigator.getGamepads !== "function") {
|
|
1401
|
+
return { pads, dispose: () => {
|
|
1402
|
+
} };
|
|
1403
|
+
}
|
|
1404
|
+
let rafId = null;
|
|
1405
|
+
function snapshot(pad) {
|
|
1406
|
+
return {
|
|
1407
|
+
index: pad.index,
|
|
1408
|
+
id: pad.id,
|
|
1409
|
+
connected: pad.connected,
|
|
1410
|
+
buttons: pad.buttons.map((b) => ({ pressed: b.pressed, value: b.value })),
|
|
1411
|
+
axes: [...pad.axes]
|
|
1412
|
+
};
|
|
1413
|
+
}
|
|
1414
|
+
function equal(a, b) {
|
|
1415
|
+
if (a.length !== b.length) return false;
|
|
1416
|
+
for (let i = 0; i < a.length; i++) {
|
|
1417
|
+
const pa = a[i];
|
|
1418
|
+
const pb = b[i];
|
|
1419
|
+
if (pa.index !== pb.index || pa.connected !== pb.connected) return false;
|
|
1420
|
+
if (pa.buttons.length !== pb.buttons.length) return false;
|
|
1421
|
+
for (let j = 0; j < pa.buttons.length; j++) {
|
|
1422
|
+
if (pa.buttons[j].pressed !== pb.buttons[j].pressed) return false;
|
|
1423
|
+
if (pa.buttons[j].value !== pb.buttons[j].value) return false;
|
|
1424
|
+
}
|
|
1425
|
+
if (pa.axes.length !== pb.axes.length) return false;
|
|
1426
|
+
for (let j = 0; j < pa.axes.length; j++) {
|
|
1427
|
+
if (pa.axes[j] !== pb.axes[j]) return false;
|
|
1428
|
+
}
|
|
1429
|
+
}
|
|
1430
|
+
return true;
|
|
1431
|
+
}
|
|
1432
|
+
function poll() {
|
|
1433
|
+
const raw = navigator.getGamepads();
|
|
1434
|
+
const snap = Array.from(raw).filter((g) => g !== null).map(snapshot);
|
|
1435
|
+
const current = pads();
|
|
1436
|
+
if (!equal(current, snap)) setPads(snap);
|
|
1437
|
+
rafId = requestAnimationFrame(poll);
|
|
1438
|
+
}
|
|
1439
|
+
function startPolling() {
|
|
1440
|
+
if (rafId === null) poll();
|
|
1441
|
+
}
|
|
1442
|
+
function stopPolling() {
|
|
1443
|
+
if (rafId !== null) {
|
|
1444
|
+
cancelAnimationFrame(rafId);
|
|
1445
|
+
rafId = null;
|
|
1446
|
+
}
|
|
1447
|
+
}
|
|
1448
|
+
const onConnect = () => startPolling();
|
|
1449
|
+
const onDisconnect = () => {
|
|
1450
|
+
const raw = navigator.getGamepads();
|
|
1451
|
+
const hasAny = Array.from(raw).some((g) => g !== null);
|
|
1452
|
+
if (!hasAny) stopPolling();
|
|
1453
|
+
};
|
|
1454
|
+
window.addEventListener("gamepadconnected", onConnect);
|
|
1455
|
+
window.addEventListener("gamepaddisconnected", onDisconnect);
|
|
1456
|
+
const initial = Array.from(navigator.getGamepads()).some((g) => g !== null);
|
|
1457
|
+
if (initial) startPolling();
|
|
1458
|
+
function dispose() {
|
|
1459
|
+
stopPolling();
|
|
1460
|
+
window.removeEventListener("gamepadconnected", onConnect);
|
|
1461
|
+
window.removeEventListener("gamepaddisconnected", onDisconnect);
|
|
1462
|
+
}
|
|
1463
|
+
return { pads, dispose };
|
|
1464
|
+
}
|
|
1465
|
+
|
|
1466
|
+
// src/browser/pointerLock.ts
|
|
1467
|
+
function pointerLock() {
|
|
1468
|
+
const [locked, setLocked] = signal(false);
|
|
1469
|
+
if (typeof document === "undefined") {
|
|
1470
|
+
return {
|
|
1471
|
+
locked,
|
|
1472
|
+
request: () => {
|
|
1473
|
+
},
|
|
1474
|
+
exit: () => {
|
|
1475
|
+
},
|
|
1476
|
+
dispose: () => {
|
|
1477
|
+
}
|
|
1478
|
+
};
|
|
1479
|
+
}
|
|
1480
|
+
const handler = () => {
|
|
1481
|
+
setLocked(!!document.pointerLockElement);
|
|
1482
|
+
};
|
|
1483
|
+
document.addEventListener("pointerlockchange", handler);
|
|
1484
|
+
function request(element) {
|
|
1485
|
+
if (typeof element.requestPointerLock === "function") {
|
|
1486
|
+
element.requestPointerLock();
|
|
1487
|
+
}
|
|
1488
|
+
}
|
|
1489
|
+
function exit() {
|
|
1490
|
+
if (typeof document.exitPointerLock === "function") {
|
|
1491
|
+
document.exitPointerLock();
|
|
1492
|
+
}
|
|
1493
|
+
}
|
|
1494
|
+
function dispose() {
|
|
1495
|
+
document.removeEventListener("pointerlockchange", handler);
|
|
1496
|
+
}
|
|
1497
|
+
return { locked, request, exit, dispose };
|
|
1498
|
+
}
|
|
1499
|
+
|
|
1500
|
+
// src/browser/vibrate.ts
|
|
1501
|
+
function vibrate(pattern) {
|
|
1502
|
+
if (typeof navigator === "undefined" || typeof navigator.vibrate !== "function") {
|
|
1503
|
+
return false;
|
|
1504
|
+
}
|
|
1505
|
+
return navigator.vibrate(pattern);
|
|
1506
|
+
}
|
|
1507
|
+
|
|
1508
|
+
// src/browser/favicon.ts
|
|
1509
|
+
function favicon(url) {
|
|
1510
|
+
if (typeof document === "undefined") return;
|
|
1511
|
+
let link = document.querySelector("link[rel='icon']");
|
|
1512
|
+
if (!link) {
|
|
1513
|
+
link = document.createElement("link");
|
|
1514
|
+
link.rel = "icon";
|
|
1515
|
+
document.head.appendChild(link);
|
|
1516
|
+
}
|
|
1517
|
+
link.href = url;
|
|
1518
|
+
}
|
|
1519
|
+
function svgFavicon(svg) {
|
|
1520
|
+
return `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`;
|
|
1521
|
+
}
|
|
1522
|
+
|
|
1523
|
+
// src/browser/textSelection.ts
|
|
1524
|
+
function textSelection() {
|
|
1525
|
+
const [text, setText] = signal("");
|
|
1526
|
+
const [rect, setRect] = signal(null);
|
|
1527
|
+
if (typeof document === "undefined") {
|
|
1528
|
+
return {
|
|
1529
|
+
text,
|
|
1530
|
+
rect,
|
|
1531
|
+
hasSelection: () => false,
|
|
1532
|
+
clear: () => {
|
|
1533
|
+
},
|
|
1534
|
+
dispose: () => {
|
|
1535
|
+
}
|
|
1536
|
+
};
|
|
1537
|
+
}
|
|
1538
|
+
const handler = () => {
|
|
1539
|
+
const sel = document.getSelection();
|
|
1540
|
+
if (!sel || sel.rangeCount === 0 || sel.isCollapsed) {
|
|
1541
|
+
setText("");
|
|
1542
|
+
setRect(null);
|
|
1543
|
+
return;
|
|
1544
|
+
}
|
|
1545
|
+
setText(sel.toString());
|
|
1546
|
+
try {
|
|
1547
|
+
const r = sel.getRangeAt(0).getBoundingClientRect();
|
|
1548
|
+
setRect(r.width > 0 || r.height > 0 ? r : null);
|
|
1549
|
+
} catch {
|
|
1550
|
+
setRect(null);
|
|
1551
|
+
}
|
|
1552
|
+
};
|
|
1553
|
+
document.addEventListener("selectionchange", handler);
|
|
1554
|
+
function clear() {
|
|
1555
|
+
const sel = document.getSelection();
|
|
1556
|
+
sel?.removeAllRanges();
|
|
1557
|
+
setText("");
|
|
1558
|
+
setRect(null);
|
|
1559
|
+
}
|
|
1560
|
+
function dispose() {
|
|
1561
|
+
document.removeEventListener("selectionchange", handler);
|
|
1562
|
+
}
|
|
1563
|
+
return {
|
|
1564
|
+
text,
|
|
1565
|
+
rect,
|
|
1566
|
+
hasSelection: () => text().length > 0,
|
|
1567
|
+
clear,
|
|
1568
|
+
dispose
|
|
1569
|
+
};
|
|
1570
|
+
}
|
|
1571
|
+
|
|
1572
|
+
// src/browser/imageLoader.ts
|
|
1573
|
+
function imageLoader(src) {
|
|
1574
|
+
const [status, setStatus] = signal("pending");
|
|
1575
|
+
const [image, setImage] = signal(null);
|
|
1576
|
+
const [width, setWidth] = signal(0);
|
|
1577
|
+
const [height, setHeight] = signal(0);
|
|
1578
|
+
if (typeof Image === "undefined") {
|
|
1579
|
+
return {
|
|
1580
|
+
status,
|
|
1581
|
+
image,
|
|
1582
|
+
width,
|
|
1583
|
+
height,
|
|
1584
|
+
dispose: () => {
|
|
1585
|
+
}
|
|
1586
|
+
};
|
|
1587
|
+
}
|
|
1588
|
+
let current = null;
|
|
1589
|
+
let disposed = false;
|
|
1590
|
+
function start(url) {
|
|
1591
|
+
if (current) {
|
|
1592
|
+
current.onload = null;
|
|
1593
|
+
current.onerror = null;
|
|
1594
|
+
}
|
|
1595
|
+
setStatus("pending");
|
|
1596
|
+
setImage(null);
|
|
1597
|
+
const img = new Image();
|
|
1598
|
+
current = img;
|
|
1599
|
+
img.onload = () => {
|
|
1600
|
+
if (disposed || current !== img) return;
|
|
1601
|
+
setImage(img);
|
|
1602
|
+
setWidth(img.naturalWidth);
|
|
1603
|
+
setHeight(img.naturalHeight);
|
|
1604
|
+
setStatus("loaded");
|
|
1605
|
+
};
|
|
1606
|
+
img.onerror = () => {
|
|
1607
|
+
if (disposed || current !== img) return;
|
|
1608
|
+
setStatus("error");
|
|
1609
|
+
};
|
|
1610
|
+
img.src = url;
|
|
1611
|
+
}
|
|
1612
|
+
if (typeof src === "function") {
|
|
1613
|
+
start(src());
|
|
1614
|
+
} else {
|
|
1615
|
+
start(src);
|
|
1616
|
+
}
|
|
1617
|
+
function dispose() {
|
|
1618
|
+
disposed = true;
|
|
1619
|
+
if (current) {
|
|
1620
|
+
current.onload = null;
|
|
1621
|
+
current.onerror = null;
|
|
1622
|
+
current = null;
|
|
1623
|
+
}
|
|
1624
|
+
}
|
|
1625
|
+
return { status, image, width, height, dispose };
|
|
1626
|
+
}
|
|
846
1627
|
// Annotate the CommonJS export names for ESM import in node:
|
|
847
1628
|
0 && (module.exports = {
|
|
1629
|
+
animationFrame,
|
|
848
1630
|
battery,
|
|
1631
|
+
bounds,
|
|
1632
|
+
broadcast,
|
|
849
1633
|
clipboard,
|
|
850
1634
|
colorScheme,
|
|
851
1635
|
draggable,
|
|
852
1636
|
dropZone,
|
|
1637
|
+
favicon,
|
|
853
1638
|
formatCurrency,
|
|
854
1639
|
formatNumber,
|
|
1640
|
+
fullscreen,
|
|
1641
|
+
gamepad,
|
|
855
1642
|
geo,
|
|
856
1643
|
idle,
|
|
1644
|
+
imageLoader,
|
|
1645
|
+
keyboard,
|
|
857
1646
|
media,
|
|
1647
|
+
mouse,
|
|
1648
|
+
mutationObserver,
|
|
1649
|
+
network,
|
|
858
1650
|
online,
|
|
859
1651
|
permissions,
|
|
1652
|
+
pointerLock,
|
|
860
1653
|
resize,
|
|
861
1654
|
scroll,
|
|
862
|
-
|
|
1655
|
+
speech,
|
|
1656
|
+
svgFavicon,
|
|
1657
|
+
swipe,
|
|
1658
|
+
textSelection,
|
|
1659
|
+
title,
|
|
1660
|
+
urlState,
|
|
1661
|
+
vibrate,
|
|
1662
|
+
visibility,
|
|
1663
|
+
wakeLock,
|
|
1664
|
+
windowSize
|
|
863
1665
|
});
|