joopjs 2.0.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/CHANGELOG.md +678 -0
- package/README.md +583 -0
- package/dist/a11y.service-C-DQQfgO.d.mts +143 -0
- package/dist/a11y.service-CauEJrJe.d.ts +143 -0
- package/dist/adapters-B6slG6hQ.d.mts +84 -0
- package/dist/adapters-B6slG6hQ.d.ts +84 -0
- package/dist/aes.service-CkoupAww.d.mts +95 -0
- package/dist/aes.service-CkoupAww.d.ts +95 -0
- package/dist/ai/index.d.mts +99 -0
- package/dist/ai/index.d.ts +99 -0
- package/dist/ai/index.js +307 -0
- package/dist/ai/index.js.map +1 -0
- package/dist/ai/index.mjs +304 -0
- package/dist/ai/index.mjs.map +1 -0
- package/dist/analytics/index.d.mts +42 -0
- package/dist/analytics/index.d.ts +42 -0
- package/dist/analytics/index.js +139 -0
- package/dist/analytics/index.js.map +1 -0
- package/dist/analytics/index.mjs +136 -0
- package/dist/analytics/index.mjs.map +1 -0
- package/dist/angular/index.d.mts +148 -0
- package/dist/angular/index.d.ts +148 -0
- package/dist/angular/index.js +122 -0
- package/dist/angular/index.js.map +1 -0
- package/dist/angular/index.mjs +101 -0
- package/dist/angular/index.mjs.map +1 -0
- package/dist/api/index.d.mts +128 -0
- package/dist/api/index.d.ts +128 -0
- package/dist/api/index.js +1358 -0
- package/dist/api/index.js.map +1 -0
- package/dist/api/index.mjs +1332 -0
- package/dist/api/index.mjs.map +1 -0
- package/dist/auth/index.d.mts +105 -0
- package/dist/auth/index.d.ts +105 -0
- package/dist/auth/index.js +989 -0
- package/dist/auth/index.js.map +1 -0
- package/dist/auth/index.mjs +979 -0
- package/dist/auth/index.mjs.map +1 -0
- package/dist/auth.service-DNVB-L4U.d.mts +16 -0
- package/dist/auth.service-PjUUSUIt.d.ts +16 -0
- package/dist/banking/index.d.mts +1530 -0
- package/dist/banking/index.d.ts +1530 -0
- package/dist/banking/index.js +4739 -0
- package/dist/banking/index.js.map +1 -0
- package/dist/banking/index.mjs +4661 -0
- package/dist/banking/index.mjs.map +1 -0
- package/dist/cache/index.d.mts +40 -0
- package/dist/cache/index.d.ts +40 -0
- package/dist/cache/index.js +174 -0
- package/dist/cache/index.js.map +1 -0
- package/dist/cache/index.mjs +172 -0
- package/dist/cache/index.mjs.map +1 -0
- package/dist/client-profile.service-BuPeXVp5.d.mts +28 -0
- package/dist/client-profile.service-D5bRRYQp.d.ts +28 -0
- package/dist/config.models-Cqg04fAQ.d.mts +84 -0
- package/dist/config.models-Cqg04fAQ.d.ts +84 -0
- package/dist/config.service-CrCvI-JS.d.ts +31 -0
- package/dist/config.service-Cz4QQLlf.d.mts +31 -0
- package/dist/core/index.d.mts +4 -0
- package/dist/core/index.d.ts +4 -0
- package/dist/core/index.js +631 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/index.mjs +619 -0
- package/dist/core/index.mjs.map +1 -0
- package/dist/crypto-utils-DriNhLdx.d.mts +30 -0
- package/dist/crypto-utils-DriNhLdx.d.ts +30 -0
- package/dist/data-storage.service-DT6xaTxE.d.ts +51 -0
- package/dist/data-storage.service-LvhGRCmw.d.mts +51 -0
- package/dist/deeplink/index.d.mts +39 -0
- package/dist/deeplink/index.d.ts +39 -0
- package/dist/deeplink/index.js +268 -0
- package/dist/deeplink/index.js.map +1 -0
- package/dist/deeplink/index.mjs +265 -0
- package/dist/deeplink/index.mjs.map +1 -0
- package/dist/deeplink.service-Ctd5u243.d.mts +35 -0
- package/dist/deeplink.service-uUuTnY9_.d.ts +35 -0
- package/dist/dev/index.d.mts +20 -0
- package/dist/dev/index.d.ts +20 -0
- package/dist/dev/index.js +51 -0
- package/dist/dev/index.js.map +1 -0
- package/dist/dev/index.mjs +49 -0
- package/dist/dev/index.mjs.map +1 -0
- package/dist/device/index.d.mts +108 -0
- package/dist/device/index.d.ts +108 -0
- package/dist/device/index.js +960 -0
- package/dist/device/index.js.map +1 -0
- package/dist/device/index.mjs +951 -0
- package/dist/device/index.mjs.map +1 -0
- package/dist/differential-privacy-BcAv1G80.d.mts +210 -0
- package/dist/differential-privacy-C8mAUjZr.d.ts +210 -0
- package/dist/encryption/index.d.mts +75 -0
- package/dist/encryption/index.d.ts +75 -0
- package/dist/encryption/index.js +605 -0
- package/dist/encryption/index.js.map +1 -0
- package/dist/encryption/index.mjs +598 -0
- package/dist/encryption/index.mjs.map +1 -0
- package/dist/form-validator-3tkmzr_o.d.mts +72 -0
- package/dist/form-validator-3tkmzr_o.d.ts +72 -0
- package/dist/forms/index.d.mts +59 -0
- package/dist/forms/index.d.ts +59 -0
- package/dist/forms/index.js +446 -0
- package/dist/forms/index.js.map +1 -0
- package/dist/forms/index.mjs +442 -0
- package/dist/forms/index.mjs.map +1 -0
- package/dist/i18n/index.d.mts +37 -0
- package/dist/i18n/index.d.ts +37 -0
- package/dist/i18n/index.js +147 -0
- package/dist/i18n/index.js.map +1 -0
- package/dist/i18n/index.mjs +145 -0
- package/dist/i18n/index.mjs.map +1 -0
- package/dist/idempotency.service-_6LqhivP.d.mts +372 -0
- package/dist/idempotency.service-eOKoISRD.d.ts +372 -0
- package/dist/index-B_ksKpS1.d.mts +202 -0
- package/dist/index-CqDKWTUP.d.mts +28 -0
- package/dist/index-CqDKWTUP.d.ts +28 -0
- package/dist/index-DFqEoX_l.d.ts +202 -0
- package/dist/index-Dz0gOur2.d.mts +36 -0
- package/dist/index-Dz0gOur2.d.ts +36 -0
- package/dist/index.d.mts +1336 -0
- package/dist/index.d.ts +1336 -0
- package/dist/index.js +19464 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +19155 -0
- package/dist/index.mjs.map +1 -0
- package/dist/india/index.d.mts +75 -0
- package/dist/india/index.d.ts +75 -0
- package/dist/india/index.js +325 -0
- package/dist/india/index.js.map +1 -0
- package/dist/india/index.mjs +303 -0
- package/dist/india/index.mjs.map +1 -0
- package/dist/joop-Bx7Iwj5p.d.mts +155 -0
- package/dist/joop-CA3DMeOO.d.ts +155 -0
- package/dist/native-bridge/index.d.mts +27 -0
- package/dist/native-bridge/index.d.ts +27 -0
- package/dist/native-bridge/index.js +98 -0
- package/dist/native-bridge/index.js.map +1 -0
- package/dist/native-bridge/index.mjs +96 -0
- package/dist/native-bridge/index.mjs.map +1 -0
- package/dist/network/index.d.mts +85 -0
- package/dist/network/index.d.ts +85 -0
- package/dist/network/index.js +454 -0
- package/dist/network/index.js.map +1 -0
- package/dist/network/index.mjs +451 -0
- package/dist/network/index.mjs.map +1 -0
- package/dist/network-monitor-BIwPSXme.d.mts +179 -0
- package/dist/network-monitor-Bqp2hvZr.d.ts +179 -0
- package/dist/notification.service-Dm4fvfZf.d.mts +25 -0
- package/dist/notification.service-tEMKatWJ.d.ts +25 -0
- package/dist/observability/index.d.mts +179 -0
- package/dist/observability/index.d.ts +179 -0
- package/dist/observability/index.js +559 -0
- package/dist/observability/index.js.map +1 -0
- package/dist/observability/index.mjs +552 -0
- package/dist/observability/index.mjs.map +1 -0
- package/dist/oidc-client-DIJcClmB.d.mts +190 -0
- package/dist/oidc-client-DxhyE59t.d.ts +190 -0
- package/dist/platform/index.d.mts +73 -0
- package/dist/platform/index.d.ts +73 -0
- package/dist/platform/index.js +127 -0
- package/dist/platform/index.js.map +1 -0
- package/dist/platform/index.mjs +125 -0
- package/dist/platform/index.mjs.map +1 -0
- package/dist/pwa/index.d.mts +31 -0
- package/dist/pwa/index.d.ts +31 -0
- package/dist/pwa/index.js +247 -0
- package/dist/pwa/index.js.map +1 -0
- package/dist/pwa/index.mjs +244 -0
- package/dist/pwa/index.mjs.map +1 -0
- package/dist/react/index.d.mts +133 -0
- package/dist/react/index.d.ts +133 -0
- package/dist/react/index.js +632 -0
- package/dist/react/index.js.map +1 -0
- package/dist/react/index.mjs +630 -0
- package/dist/react/index.mjs.map +1 -0
- package/dist/router/index.d.mts +39 -0
- package/dist/router/index.d.ts +39 -0
- package/dist/router/index.js +168 -0
- package/dist/router/index.js.map +1 -0
- package/dist/router/index.mjs +166 -0
- package/dist/router/index.mjs.map +1 -0
- package/dist/security/index.d.mts +206 -0
- package/dist/security/index.d.ts +206 -0
- package/dist/security/index.js +1297 -0
- package/dist/security/index.js.map +1 -0
- package/dist/security/index.mjs +1285 -0
- package/dist/security/index.mjs.map +1 -0
- package/dist/session/index.d.mts +115 -0
- package/dist/session/index.d.ts +115 -0
- package/dist/session/index.js +297 -0
- package/dist/session/index.js.map +1 -0
- package/dist/session/index.mjs +292 -0
- package/dist/session/index.mjs.map +1 -0
- package/dist/state/index.d.mts +43 -0
- package/dist/state/index.d.ts +43 -0
- package/dist/state/index.js +156 -0
- package/dist/state/index.js.map +1 -0
- package/dist/state/index.mjs +152 -0
- package/dist/state/index.mjs.map +1 -0
- package/dist/statement-parser-BHQtXwCM.d.ts +260 -0
- package/dist/statement-parser-C2qNmb49.d.mts +260 -0
- package/dist/storage/index.d.mts +40 -0
- package/dist/storage/index.d.ts +40 -0
- package/dist/storage/index.js +256 -0
- package/dist/storage/index.js.map +1 -0
- package/dist/storage/index.mjs +252 -0
- package/dist/storage/index.mjs.map +1 -0
- package/dist/sync/index.d.mts +69 -0
- package/dist/sync/index.d.ts +69 -0
- package/dist/sync/index.js +330 -0
- package/dist/sync/index.js.map +1 -0
- package/dist/sync/index.mjs +323 -0
- package/dist/sync/index.mjs.map +1 -0
- package/dist/sync-engine-DCIMRG5s.d.ts +61 -0
- package/dist/sync-engine-DZqyKHkK.d.mts +61 -0
- package/dist/theme/index.d.mts +53 -0
- package/dist/theme/index.d.ts +53 -0
- package/dist/theme/index.js +169 -0
- package/dist/theme/index.js.map +1 -0
- package/dist/theme/index.mjs +167 -0
- package/dist/theme/index.mjs.map +1 -0
- package/dist/ui/index.d.mts +66 -0
- package/dist/ui/index.d.ts +66 -0
- package/dist/ui/index.js +811 -0
- package/dist/ui/index.js.map +1 -0
- package/dist/ui/index.mjs +803 -0
- package/dist/ui/index.mjs.map +1 -0
- package/dist/utilities/index.d.mts +199 -0
- package/dist/utilities/index.d.ts +199 -0
- package/dist/utilities/index.js +1991 -0
- package/dist/utilities/index.js.map +1 -0
- package/dist/utilities/index.mjs +1923 -0
- package/dist/utilities/index.mjs.map +1 -0
- package/dist/validation/index.d.mts +60 -0
- package/dist/validation/index.d.ts +60 -0
- package/dist/validation/index.js +460 -0
- package/dist/validation/index.js.map +1 -0
- package/dist/validation/index.mjs +455 -0
- package/dist/validation/index.mjs.map +1 -0
- package/dist/vue/index.d.mts +135 -0
- package/dist/vue/index.d.ts +135 -0
- package/dist/vue/index.js +621 -0
- package/dist/vue/index.js.map +1 -0
- package/dist/vue/index.mjs +619 -0
- package/dist/vue/index.mjs.map +1 -0
- package/dist/watermark.service-Detur5tq.d.ts +235 -0
- package/dist/watermark.service-QNegMeQZ.d.mts +235 -0
- package/dist/workers/index.d.mts +42 -0
- package/dist/workers/index.d.ts +42 -0
- package/dist/workers/index.js +359 -0
- package/dist/workers/index.js.map +1 -0
- package/dist/workers/index.mjs +356 -0
- package/dist/workers/index.mjs.map +1 -0
- package/dist/workflow/index.d.mts +99 -0
- package/dist/workflow/index.d.ts +99 -0
- package/dist/workflow/index.js +282 -0
- package/dist/workflow/index.js.map +1 -0
- package/dist/workflow/index.mjs +279 -0
- package/dist/workflow/index.mjs.map +1 -0
- package/package.json +226 -0
|
@@ -0,0 +1,803 @@
|
|
|
1
|
+
// src/events/index.ts
|
|
2
|
+
var JoopSubject = class {
|
|
3
|
+
_listeners = [];
|
|
4
|
+
subscribe(listener) {
|
|
5
|
+
this._listeners.push(listener);
|
|
6
|
+
return () => {
|
|
7
|
+
this._listeners = this._listeners.filter((l) => l !== listener);
|
|
8
|
+
};
|
|
9
|
+
}
|
|
10
|
+
next(value) {
|
|
11
|
+
for (const listener of this._listeners) {
|
|
12
|
+
listener(value);
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
asObservable() {
|
|
16
|
+
return new JoopObservable((listener) => this.subscribe(listener));
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
var JoopBehaviorSubject = class extends JoopSubject {
|
|
20
|
+
_value;
|
|
21
|
+
constructor(initialValue) {
|
|
22
|
+
super();
|
|
23
|
+
this._value = initialValue;
|
|
24
|
+
}
|
|
25
|
+
getValue() {
|
|
26
|
+
return this._value;
|
|
27
|
+
}
|
|
28
|
+
next(value) {
|
|
29
|
+
this._value = value;
|
|
30
|
+
super.next(value);
|
|
31
|
+
}
|
|
32
|
+
subscribe(listener) {
|
|
33
|
+
listener(this._value);
|
|
34
|
+
return super.subscribe(listener);
|
|
35
|
+
}
|
|
36
|
+
asObservable() {
|
|
37
|
+
return new JoopObservable((listener) => this.subscribe(listener));
|
|
38
|
+
}
|
|
39
|
+
};
|
|
40
|
+
var JoopObservable = class {
|
|
41
|
+
constructor(_subscribeFn) {
|
|
42
|
+
this._subscribeFn = _subscribeFn;
|
|
43
|
+
}
|
|
44
|
+
_subscribeFn;
|
|
45
|
+
subscribe(listener) {
|
|
46
|
+
return this._subscribeFn(listener);
|
|
47
|
+
}
|
|
48
|
+
/** Returns the current value without subscribing (only meaningful for BehaviorSubject-backed observables). */
|
|
49
|
+
getOnce() {
|
|
50
|
+
let result;
|
|
51
|
+
const unsub = this.subscribe((v) => {
|
|
52
|
+
result = v;
|
|
53
|
+
});
|
|
54
|
+
unsub();
|
|
55
|
+
return result;
|
|
56
|
+
}
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
// src/ui/loader.service.ts
|
|
60
|
+
var JoopLoaderService = class {
|
|
61
|
+
_count = 0;
|
|
62
|
+
_loading$ = new JoopBehaviorSubject(false);
|
|
63
|
+
_delayTimer = null;
|
|
64
|
+
_cfg;
|
|
65
|
+
constructor(config = {}) {
|
|
66
|
+
this._cfg = config;
|
|
67
|
+
}
|
|
68
|
+
/** Increment loader counter — shows loader when count goes 0 → 1 */
|
|
69
|
+
show() {
|
|
70
|
+
this._count++;
|
|
71
|
+
if (this._count === 1) {
|
|
72
|
+
if (this._cfg.delayMs && this._cfg.delayMs > 0) {
|
|
73
|
+
this._delayTimer = setTimeout(() => this._loading$.next(true), this._cfg.delayMs);
|
|
74
|
+
} else {
|
|
75
|
+
this._loading$.next(true);
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
/** Decrement loader counter — hides loader when count reaches 0 */
|
|
80
|
+
hide() {
|
|
81
|
+
if (this._count > 0) this._count--;
|
|
82
|
+
if (this._count === 0) {
|
|
83
|
+
if (this._delayTimer) {
|
|
84
|
+
clearTimeout(this._delayTimer);
|
|
85
|
+
this._delayTimer = null;
|
|
86
|
+
}
|
|
87
|
+
this._loading$.next(false);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
/** Force-show regardless of count */
|
|
91
|
+
forceShow() {
|
|
92
|
+
this._count = Math.max(this._count, 1);
|
|
93
|
+
this._loading$.next(true);
|
|
94
|
+
}
|
|
95
|
+
/** Force-hide and reset counter */
|
|
96
|
+
reset() {
|
|
97
|
+
this._count = 0;
|
|
98
|
+
if (this._delayTimer) {
|
|
99
|
+
clearTimeout(this._delayTimer);
|
|
100
|
+
this._delayTimer = null;
|
|
101
|
+
}
|
|
102
|
+
this._loading$.next(false);
|
|
103
|
+
}
|
|
104
|
+
isLoading() {
|
|
105
|
+
return this._loading$.getValue();
|
|
106
|
+
}
|
|
107
|
+
isLoading$() {
|
|
108
|
+
return this._loading$.asObservable();
|
|
109
|
+
}
|
|
110
|
+
get pendingCount() {
|
|
111
|
+
return this._count;
|
|
112
|
+
}
|
|
113
|
+
};
|
|
114
|
+
|
|
115
|
+
// src/ui/alert.service.ts
|
|
116
|
+
var COLORS = {
|
|
117
|
+
success: { bg: "#d4edda", border: "#28a745", icon: "\u2713" },
|
|
118
|
+
error: { bg: "#f8d7da", border: "#dc3545", icon: "\u2715" },
|
|
119
|
+
warning: { bg: "#fff3cd", border: "#ffc107", icon: "\u26A0" },
|
|
120
|
+
info: { bg: "#d1ecf1", border: "#17a2b8", icon: "\u2139" }
|
|
121
|
+
};
|
|
122
|
+
var TOAST_CONTAINER_ID = "__joop_toast_container__";
|
|
123
|
+
var OVERLAY_ID = "__joop_dialog_overlay__";
|
|
124
|
+
var JoopAlertService = class {
|
|
125
|
+
_inline$ = new JoopBehaviorSubject(null);
|
|
126
|
+
_open$ = new JoopSubject();
|
|
127
|
+
inline$() {
|
|
128
|
+
return this._inline$.asObservable();
|
|
129
|
+
}
|
|
130
|
+
open$() {
|
|
131
|
+
return this._open$.asObservable();
|
|
132
|
+
}
|
|
133
|
+
showInline(data) {
|
|
134
|
+
this._inline$.next(data);
|
|
135
|
+
}
|
|
136
|
+
hideInline() {
|
|
137
|
+
this._inline$.next(null);
|
|
138
|
+
}
|
|
139
|
+
toast(message, options = {}) {
|
|
140
|
+
if (typeof document === "undefined") return;
|
|
141
|
+
const { type = "info", duration = 3e3, position = "bottom", closeable = false } = options;
|
|
142
|
+
const c = COLORS[type];
|
|
143
|
+
const container = this._getToastContainer(position);
|
|
144
|
+
const toast = document.createElement("div");
|
|
145
|
+
toast.setAttribute("role", "alert");
|
|
146
|
+
Object.assign(toast.style, {
|
|
147
|
+
background: c.bg,
|
|
148
|
+
borderLeft: `4px solid ${c.border}`,
|
|
149
|
+
padding: "12px 16px",
|
|
150
|
+
marginBottom: "8px",
|
|
151
|
+
borderRadius: "4px",
|
|
152
|
+
display: "flex",
|
|
153
|
+
alignItems: "center",
|
|
154
|
+
gap: "8px",
|
|
155
|
+
boxShadow: "0 2px 8px rgba(0,0,0,0.15)",
|
|
156
|
+
fontSize: "14px",
|
|
157
|
+
maxWidth: "320px",
|
|
158
|
+
opacity: "0",
|
|
159
|
+
transition: "opacity 0.2s ease",
|
|
160
|
+
cursor: closeable ? "pointer" : "default"
|
|
161
|
+
});
|
|
162
|
+
const icon = document.createElement("span");
|
|
163
|
+
icon.textContent = c.icon;
|
|
164
|
+
icon.style.fontWeight = "bold";
|
|
165
|
+
const text = document.createElement("span");
|
|
166
|
+
text.textContent = message;
|
|
167
|
+
text.style.flex = "1";
|
|
168
|
+
toast.appendChild(icon);
|
|
169
|
+
toast.appendChild(text);
|
|
170
|
+
if (closeable) {
|
|
171
|
+
const btn = document.createElement("button");
|
|
172
|
+
btn.textContent = "\xD7";
|
|
173
|
+
Object.assign(btn.style, { background: "none", border: "none", cursor: "pointer", fontSize: "16px", padding: "0 4px" });
|
|
174
|
+
btn.onclick = () => this._removeToast(toast);
|
|
175
|
+
toast.appendChild(btn);
|
|
176
|
+
}
|
|
177
|
+
container.appendChild(toast);
|
|
178
|
+
requestAnimationFrame(() => {
|
|
179
|
+
toast.style.opacity = "1";
|
|
180
|
+
});
|
|
181
|
+
if (duration > 0) {
|
|
182
|
+
setTimeout(() => this._removeToast(toast), duration);
|
|
183
|
+
}
|
|
184
|
+
this._open$.next("toast");
|
|
185
|
+
}
|
|
186
|
+
dialog(options) {
|
|
187
|
+
return new Promise((resolve) => {
|
|
188
|
+
if (typeof document === "undefined") {
|
|
189
|
+
resolve("close");
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
const { title, message, type = "info", closeable = true, htmlContent = false } = options;
|
|
193
|
+
const buttons = options.buttons ?? [{ label: "OK", action: "ok", style: "primary" }];
|
|
194
|
+
const c = COLORS[type];
|
|
195
|
+
const overlay = document.createElement("div");
|
|
196
|
+
overlay.id = OVERLAY_ID;
|
|
197
|
+
Object.assign(overlay.style, {
|
|
198
|
+
position: "fixed",
|
|
199
|
+
inset: "0",
|
|
200
|
+
background: "rgba(0,0,0,0.5)",
|
|
201
|
+
display: "flex",
|
|
202
|
+
alignItems: "center",
|
|
203
|
+
justifyContent: "center",
|
|
204
|
+
zIndex: "99999"
|
|
205
|
+
});
|
|
206
|
+
const dialog = document.createElement("div");
|
|
207
|
+
Object.assign(dialog.style, {
|
|
208
|
+
background: "#fff",
|
|
209
|
+
borderRadius: "8px",
|
|
210
|
+
padding: "24px",
|
|
211
|
+
minWidth: "280px",
|
|
212
|
+
maxWidth: "480px",
|
|
213
|
+
boxShadow: "0 8px 24px rgba(0,0,0,0.2)",
|
|
214
|
+
position: "relative"
|
|
215
|
+
});
|
|
216
|
+
if (title) {
|
|
217
|
+
const h = document.createElement("h3");
|
|
218
|
+
h.textContent = title;
|
|
219
|
+
Object.assign(h.style, { margin: "0 0 12px", fontSize: "18px", color: c.border });
|
|
220
|
+
dialog.appendChild(h);
|
|
221
|
+
}
|
|
222
|
+
const msg = document.createElement("div");
|
|
223
|
+
if (htmlContent) {
|
|
224
|
+
msg.innerHTML = message;
|
|
225
|
+
} else {
|
|
226
|
+
msg.textContent = message;
|
|
227
|
+
}
|
|
228
|
+
Object.assign(msg.style, { marginBottom: "20px", fontSize: "14px", lineHeight: "1.5" });
|
|
229
|
+
dialog.appendChild(msg);
|
|
230
|
+
const btnRow = document.createElement("div");
|
|
231
|
+
Object.assign(btnRow.style, { display: "flex", gap: "8px", justifyContent: "flex-end" });
|
|
232
|
+
const finish = (action) => {
|
|
233
|
+
overlay.remove();
|
|
234
|
+
resolve(action);
|
|
235
|
+
};
|
|
236
|
+
for (const btn of buttons) {
|
|
237
|
+
const el = document.createElement("button");
|
|
238
|
+
el.textContent = btn.label;
|
|
239
|
+
const isPrimary = btn.style === "primary";
|
|
240
|
+
const isDanger = btn.style === "danger";
|
|
241
|
+
Object.assign(el.style, {
|
|
242
|
+
padding: "8px 16px",
|
|
243
|
+
borderRadius: "4px",
|
|
244
|
+
cursor: "pointer",
|
|
245
|
+
fontSize: "14px",
|
|
246
|
+
border: "none",
|
|
247
|
+
background: isDanger ? "#dc3545" : isPrimary ? c.border : "#e9ecef",
|
|
248
|
+
color: isPrimary || isDanger ? "#fff" : "#333"
|
|
249
|
+
});
|
|
250
|
+
el.onclick = () => finish(btn.action);
|
|
251
|
+
btnRow.appendChild(el);
|
|
252
|
+
}
|
|
253
|
+
dialog.appendChild(btnRow);
|
|
254
|
+
if (closeable) {
|
|
255
|
+
const closeBtn = document.createElement("button");
|
|
256
|
+
closeBtn.textContent = "\xD7";
|
|
257
|
+
Object.assign(closeBtn.style, {
|
|
258
|
+
position: "absolute",
|
|
259
|
+
top: "12px",
|
|
260
|
+
right: "16px",
|
|
261
|
+
background: "none",
|
|
262
|
+
border: "none",
|
|
263
|
+
cursor: "pointer",
|
|
264
|
+
fontSize: "20px",
|
|
265
|
+
color: "#666"
|
|
266
|
+
});
|
|
267
|
+
closeBtn.onclick = () => finish("close");
|
|
268
|
+
overlay.addEventListener("click", (e) => {
|
|
269
|
+
if (e.target === overlay) finish("close");
|
|
270
|
+
});
|
|
271
|
+
dialog.appendChild(closeBtn);
|
|
272
|
+
}
|
|
273
|
+
overlay.appendChild(dialog);
|
|
274
|
+
document.body.appendChild(overlay);
|
|
275
|
+
this._open$.next("dialog");
|
|
276
|
+
});
|
|
277
|
+
}
|
|
278
|
+
dismiss() {
|
|
279
|
+
document.getElementById(OVERLAY_ID)?.remove();
|
|
280
|
+
document.getElementById(TOAST_CONTAINER_ID)?.remove();
|
|
281
|
+
this._inline$.next(null);
|
|
282
|
+
}
|
|
283
|
+
_getToastContainer(position) {
|
|
284
|
+
const existing = document.getElementById(TOAST_CONTAINER_ID);
|
|
285
|
+
if (existing) return existing;
|
|
286
|
+
const container = document.createElement("div");
|
|
287
|
+
container.id = TOAST_CONTAINER_ID;
|
|
288
|
+
const top = position.startsWith("top");
|
|
289
|
+
const right = position.endsWith("right");
|
|
290
|
+
Object.assign(container.style, {
|
|
291
|
+
position: "fixed",
|
|
292
|
+
[top ? "top" : "bottom"]: "16px",
|
|
293
|
+
[right ? "right" : "left"]: right ? "16px" : "50%",
|
|
294
|
+
transform: right ? "none" : "translateX(-50%)",
|
|
295
|
+
zIndex: "99998",
|
|
296
|
+
display: "flex",
|
|
297
|
+
flexDirection: top ? "column" : "column-reverse",
|
|
298
|
+
alignItems: right ? "flex-end" : "center"
|
|
299
|
+
});
|
|
300
|
+
document.body.appendChild(container);
|
|
301
|
+
return container;
|
|
302
|
+
}
|
|
303
|
+
_removeToast(el) {
|
|
304
|
+
el.style.opacity = "0";
|
|
305
|
+
setTimeout(() => el.remove(), 200);
|
|
306
|
+
}
|
|
307
|
+
};
|
|
308
|
+
|
|
309
|
+
// src/ui/pagination.service.ts
|
|
310
|
+
var JoopPaginationService = class {
|
|
311
|
+
_state$ = new JoopBehaviorSubject(this._compute(1, 10, 0));
|
|
312
|
+
_compute(page, pageSize, total) {
|
|
313
|
+
const totalPages = Math.max(1, Math.ceil(total / pageSize));
|
|
314
|
+
const clamped = Math.min(Math.max(1, page), totalPages);
|
|
315
|
+
const from = total === 0 ? 0 : (clamped - 1) * pageSize + 1;
|
|
316
|
+
const to = Math.min(clamped * pageSize, total);
|
|
317
|
+
return { page: clamped, pageSize, total, totalPages, hasPrev: clamped > 1, hasNext: clamped < totalPages, from, to };
|
|
318
|
+
}
|
|
319
|
+
_update(patch) {
|
|
320
|
+
const { page, pageSize, total } = this._state$.getValue();
|
|
321
|
+
const next = { page, pageSize, total, ...patch };
|
|
322
|
+
this._state$.next(this._compute(next.page, next.pageSize, next.total));
|
|
323
|
+
}
|
|
324
|
+
setTotal(total) {
|
|
325
|
+
this._update({ total, page: 1 });
|
|
326
|
+
}
|
|
327
|
+
setPage(page) {
|
|
328
|
+
this._update({ page });
|
|
329
|
+
}
|
|
330
|
+
setPageSize(size) {
|
|
331
|
+
this._update({ pageSize: size, page: 1 });
|
|
332
|
+
}
|
|
333
|
+
nextPage() {
|
|
334
|
+
const s = this._state$.getValue();
|
|
335
|
+
if (s.hasNext) this._update({ page: s.page + 1 });
|
|
336
|
+
}
|
|
337
|
+
prevPage() {
|
|
338
|
+
const s = this._state$.getValue();
|
|
339
|
+
if (s.hasPrev) this._update({ page: s.page - 1 });
|
|
340
|
+
}
|
|
341
|
+
firstPage() {
|
|
342
|
+
this._update({ page: 1 });
|
|
343
|
+
}
|
|
344
|
+
lastPage() {
|
|
345
|
+
this._update({ page: this._state$.getValue().totalPages });
|
|
346
|
+
}
|
|
347
|
+
getState() {
|
|
348
|
+
return this._state$.getValue();
|
|
349
|
+
}
|
|
350
|
+
state$() {
|
|
351
|
+
return this._state$.asObservable();
|
|
352
|
+
}
|
|
353
|
+
/** Returns page numbers for pagination UI — with ellipsis as -1 */
|
|
354
|
+
getPages(maxVisible = 7) {
|
|
355
|
+
const { page, totalPages } = this._state$.getValue();
|
|
356
|
+
if (totalPages <= maxVisible) return Array.from({ length: totalPages }, (_, i) => i + 1);
|
|
357
|
+
const pages = [];
|
|
358
|
+
const half = Math.floor(maxVisible / 2);
|
|
359
|
+
let start = Math.max(2, page - half);
|
|
360
|
+
let end = Math.min(totalPages - 1, page + half);
|
|
361
|
+
if (page - half < 2) end = Math.min(totalPages - 1, maxVisible - 1);
|
|
362
|
+
if (page + half > totalPages - 1) start = Math.max(2, totalPages - maxVisible + 2);
|
|
363
|
+
pages.push(1);
|
|
364
|
+
if (start > 2) pages.push(-1);
|
|
365
|
+
for (let i = start; i <= end; i++) pages.push(i);
|
|
366
|
+
if (end < totalPages - 1) pages.push(-1);
|
|
367
|
+
pages.push(totalPages);
|
|
368
|
+
return pages;
|
|
369
|
+
}
|
|
370
|
+
reset() {
|
|
371
|
+
this._state$.next(this._compute(1, this._state$.getValue().pageSize, 0));
|
|
372
|
+
}
|
|
373
|
+
};
|
|
374
|
+
|
|
375
|
+
// src/ui/print.service.ts
|
|
376
|
+
var JoopPrintService = class {
|
|
377
|
+
/** Print the current page */
|
|
378
|
+
print() {
|
|
379
|
+
window.print();
|
|
380
|
+
}
|
|
381
|
+
/** Print a specific element by ID or reference */
|
|
382
|
+
printElement(elementOrId, options = {}) {
|
|
383
|
+
const el = typeof elementOrId === "string" ? document.getElementById(elementOrId) : elementOrId;
|
|
384
|
+
if (!el) return;
|
|
385
|
+
this.printHtml(el.outerHTML, options);
|
|
386
|
+
}
|
|
387
|
+
/** Open a new window with custom HTML and print it */
|
|
388
|
+
printHtml(html, options = {}) {
|
|
389
|
+
const win = window.open("", "_blank", "width=800,height=600");
|
|
390
|
+
if (!win) return;
|
|
391
|
+
const { styles = "", title = document.title, bodyClass = "" } = options;
|
|
392
|
+
const defaultStyles = `
|
|
393
|
+
body { font-family: Arial, sans-serif; font-size: 12pt; margin: 20mm; }
|
|
394
|
+
@media print { body { margin: 0; } }
|
|
395
|
+
table { border-collapse: collapse; width: 100%; }
|
|
396
|
+
th, td { border: 1px solid #ccc; padding: 6px 10px; text-align: left; }
|
|
397
|
+
th { background: #f5f5f5; font-weight: bold; }
|
|
398
|
+
`;
|
|
399
|
+
win.document.write(`<!DOCTYPE html>
|
|
400
|
+
<html>
|
|
401
|
+
<head>
|
|
402
|
+
<meta charset="utf-8">
|
|
403
|
+
<title>${title}</title>
|
|
404
|
+
<style>${defaultStyles}${styles}</style>
|
|
405
|
+
</head>
|
|
406
|
+
<body class="${bodyClass}">${html}</body>
|
|
407
|
+
</html>`);
|
|
408
|
+
win.document.close();
|
|
409
|
+
win.focus();
|
|
410
|
+
setTimeout(() => {
|
|
411
|
+
win.print();
|
|
412
|
+
win.close();
|
|
413
|
+
}, 300);
|
|
414
|
+
}
|
|
415
|
+
/** Print a plain-text receipt in a monospace layout */
|
|
416
|
+
printReceipt(lines, options = {}) {
|
|
417
|
+
const html = `<pre style="font-family:monospace;font-size:11pt;white-space:pre-wrap">${lines.join("\n")}</pre>`;
|
|
418
|
+
this.printHtml(html, { ...options, styles: `${options.styles ?? ""}body{margin:10mm;}` });
|
|
419
|
+
}
|
|
420
|
+
/** Build a receipt-style text block for printing */
|
|
421
|
+
buildReceiptText(title, rows, width = 42) {
|
|
422
|
+
const line = "\u2500".repeat(width);
|
|
423
|
+
const center = (s) => s.padStart(Math.floor((width + s.length) / 2)).padEnd(width);
|
|
424
|
+
const row = ([l, v]) => `${l.padEnd(width - v.length - 2)} ${v}`;
|
|
425
|
+
return [center(title), line, ...rows.map(row), line];
|
|
426
|
+
}
|
|
427
|
+
};
|
|
428
|
+
|
|
429
|
+
// src/ui/a11y.service.ts
|
|
430
|
+
var JoopA11yService = class {
|
|
431
|
+
_liveRegions = /* @__PURE__ */ new Map();
|
|
432
|
+
_trapStack = [];
|
|
433
|
+
_skipLink = null;
|
|
434
|
+
_focusHandlers = [];
|
|
435
|
+
_focusListener = null;
|
|
436
|
+
// ── ARIA Live Regions ────────────────────────────────────────────────────
|
|
437
|
+
announce(message, politeness = "polite") {
|
|
438
|
+
const region = this._getOrCreateLiveRegion(politeness);
|
|
439
|
+
region.textContent = "";
|
|
440
|
+
setTimeout(() => {
|
|
441
|
+
region.textContent = message;
|
|
442
|
+
}, 50);
|
|
443
|
+
}
|
|
444
|
+
_getOrCreateLiveRegion(politeness) {
|
|
445
|
+
if (!this._liveRegions.has(politeness)) {
|
|
446
|
+
const el = document.createElement("div");
|
|
447
|
+
el.setAttribute("aria-live", politeness);
|
|
448
|
+
el.setAttribute("aria-atomic", "true");
|
|
449
|
+
el.setAttribute("aria-relevant", "additions text");
|
|
450
|
+
Object.assign(el.style, { position: "absolute", width: "1px", height: "1px", padding: "0", overflow: "hidden", clip: "rect(0,0,0,0)", whiteSpace: "nowrap", border: "0" });
|
|
451
|
+
document.body.appendChild(el);
|
|
452
|
+
this._liveRegions.set(politeness, el);
|
|
453
|
+
}
|
|
454
|
+
return this._liveRegions.get(politeness);
|
|
455
|
+
}
|
|
456
|
+
clearAnnouncements() {
|
|
457
|
+
for (const el of this._liveRegions.values()) el.textContent = "";
|
|
458
|
+
}
|
|
459
|
+
// ── Focus Trap ───────────────────────────────────────────────────────────
|
|
460
|
+
trapFocus(container, opts = {}) {
|
|
461
|
+
const prev = document.activeElement;
|
|
462
|
+
const focusable = () => container.querySelectorAll('a[href],button:not([disabled]),input:not([disabled]),select:not([disabled]),textarea:not([disabled]),[tabindex]:not([tabindex="-1"])');
|
|
463
|
+
const handleKeydown = (e) => {
|
|
464
|
+
if (e.key !== "Tab") return;
|
|
465
|
+
const elements = Array.from(focusable());
|
|
466
|
+
if (!elements.length) {
|
|
467
|
+
e.preventDefault();
|
|
468
|
+
return;
|
|
469
|
+
}
|
|
470
|
+
const first = elements[0], last = elements[elements.length - 1];
|
|
471
|
+
if (e.shiftKey && document.activeElement === first) {
|
|
472
|
+
e.preventDefault();
|
|
473
|
+
last.focus();
|
|
474
|
+
} else if (!e.shiftKey && document.activeElement === last) {
|
|
475
|
+
e.preventDefault();
|
|
476
|
+
first.focus();
|
|
477
|
+
}
|
|
478
|
+
};
|
|
479
|
+
container.addEventListener("keydown", handleKeydown);
|
|
480
|
+
const initialEl = opts.initialFocus ? container.querySelector(opts.initialFocus) : focusable()[0];
|
|
481
|
+
initialEl?.focus();
|
|
482
|
+
const release = () => {
|
|
483
|
+
container.removeEventListener("keydown", handleKeydown);
|
|
484
|
+
if (opts.returnFocusOnDeactivate !== false && prev instanceof HTMLElement) prev.focus();
|
|
485
|
+
};
|
|
486
|
+
this._trapStack.push({ el: container, prev, release });
|
|
487
|
+
return release;
|
|
488
|
+
}
|
|
489
|
+
releaseFocus() {
|
|
490
|
+
const entry = this._trapStack.pop();
|
|
491
|
+
entry?.release();
|
|
492
|
+
}
|
|
493
|
+
releaseAllTraps() {
|
|
494
|
+
while (this._trapStack.length) this.releaseFocus();
|
|
495
|
+
}
|
|
496
|
+
// ── Skip Link ────────────────────────────────────────────────────────────
|
|
497
|
+
createSkipLink(targetId, label = "Skip to main content") {
|
|
498
|
+
const link = document.createElement("a");
|
|
499
|
+
link.href = `#${targetId}`;
|
|
500
|
+
link.textContent = label;
|
|
501
|
+
link.className = "joop-skip-link";
|
|
502
|
+
Object.assign(link.style, {
|
|
503
|
+
position: "absolute",
|
|
504
|
+
top: "-9999px",
|
|
505
|
+
left: "0",
|
|
506
|
+
zIndex: "9999",
|
|
507
|
+
padding: "8px 16px",
|
|
508
|
+
background: "#fff",
|
|
509
|
+
color: "#000",
|
|
510
|
+
textDecoration: "underline"
|
|
511
|
+
});
|
|
512
|
+
link.addEventListener("focus", () => {
|
|
513
|
+
link.style.top = "0";
|
|
514
|
+
});
|
|
515
|
+
link.addEventListener("blur", () => {
|
|
516
|
+
link.style.top = "-9999px";
|
|
517
|
+
});
|
|
518
|
+
document.body.insertBefore(link, document.body.firstChild);
|
|
519
|
+
this._skipLink = link;
|
|
520
|
+
return link;
|
|
521
|
+
}
|
|
522
|
+
removeSkipLink() {
|
|
523
|
+
this._skipLink?.remove();
|
|
524
|
+
this._skipLink = null;
|
|
525
|
+
}
|
|
526
|
+
// ── Keyboard helpers ─────────────────────────────────────────────────────
|
|
527
|
+
/** Start tracking focus changes */
|
|
528
|
+
trackFocus() {
|
|
529
|
+
const handler = (e) => {
|
|
530
|
+
if (e.target instanceof Element) this._focusHandlers.forEach((h) => h(e.target));
|
|
531
|
+
};
|
|
532
|
+
document.addEventListener("focusin", handler);
|
|
533
|
+
this._focusListener = handler;
|
|
534
|
+
return () => {
|
|
535
|
+
document.removeEventListener("focusin", handler);
|
|
536
|
+
this._focusListener = null;
|
|
537
|
+
};
|
|
538
|
+
}
|
|
539
|
+
onFocusChange(handler) {
|
|
540
|
+
this._focusHandlers.push(handler);
|
|
541
|
+
return () => {
|
|
542
|
+
this._focusHandlers = this._focusHandlers.filter((h) => h !== handler);
|
|
543
|
+
};
|
|
544
|
+
}
|
|
545
|
+
/** Set visible page title + meta description */
|
|
546
|
+
setPageTitle(title, suffix) {
|
|
547
|
+
document.title = suffix ? `${title} \u2014 ${suffix}` : title;
|
|
548
|
+
}
|
|
549
|
+
/** Make an element visually hidden but accessible */
|
|
550
|
+
visuallyHide(el) {
|
|
551
|
+
Object.assign(el.style, { position: "absolute", width: "1px", height: "1px", padding: "0", overflow: "hidden", clip: "rect(0,0,0,0)", whiteSpace: "nowrap", border: "0" });
|
|
552
|
+
}
|
|
553
|
+
/** Get all focusable elements within a container */
|
|
554
|
+
getFocusable(container = document.body) {
|
|
555
|
+
return Array.from(container.querySelectorAll(
|
|
556
|
+
'a[href],button:not([disabled]),input:not([disabled]),select:not([disabled]),textarea:not([disabled]),[tabindex]:not([tabindex="-1"])'
|
|
557
|
+
));
|
|
558
|
+
}
|
|
559
|
+
/** Check whether an element is visible and not hidden from accessibility */
|
|
560
|
+
isAccessible(el) {
|
|
561
|
+
const style = window.getComputedStyle(el);
|
|
562
|
+
return style.display !== "none" && style.visibility !== "hidden" && el.getAttribute("aria-hidden") !== "true";
|
|
563
|
+
}
|
|
564
|
+
destroy() {
|
|
565
|
+
for (const el of this._liveRegions.values()) el.remove();
|
|
566
|
+
this._liveRegions.clear();
|
|
567
|
+
this.releaseAllTraps();
|
|
568
|
+
this.removeSkipLink();
|
|
569
|
+
if (this._focusListener) document.removeEventListener("focusin", this._focusListener);
|
|
570
|
+
}
|
|
571
|
+
};
|
|
572
|
+
|
|
573
|
+
// src/ui/virtual-scroll.ts
|
|
574
|
+
var JoopVirtualScroll = class {
|
|
575
|
+
_container = null;
|
|
576
|
+
_items = [];
|
|
577
|
+
_renderFn = null;
|
|
578
|
+
_opts = { itemHeight: 40, overscan: 3 };
|
|
579
|
+
_scrollListener = null;
|
|
580
|
+
_rangeHandlers = [];
|
|
581
|
+
_innerEl = null;
|
|
582
|
+
_spacerTop = null;
|
|
583
|
+
_spacerBottom = null;
|
|
584
|
+
_rendered = [];
|
|
585
|
+
_lastRange = { start: 0, end: 0, offsetTop: 0 };
|
|
586
|
+
attach(container, items, renderFn, opts) {
|
|
587
|
+
this.detach();
|
|
588
|
+
this._container = container;
|
|
589
|
+
this._items = items;
|
|
590
|
+
this._renderFn = renderFn;
|
|
591
|
+
if (opts) Object.assign(this._opts, opts);
|
|
592
|
+
container.style.overflow = "auto";
|
|
593
|
+
container.style.position = "relative";
|
|
594
|
+
this._spacerTop = _spacer();
|
|
595
|
+
this._innerEl = document.createElement("div");
|
|
596
|
+
this._spacerBottom = _spacer();
|
|
597
|
+
container.appendChild(this._spacerTop);
|
|
598
|
+
container.appendChild(this._innerEl);
|
|
599
|
+
container.appendChild(this._spacerBottom);
|
|
600
|
+
const render = () => this._render();
|
|
601
|
+
container.addEventListener("scroll", render);
|
|
602
|
+
this._scrollListener = () => container.removeEventListener("scroll", render);
|
|
603
|
+
this._render();
|
|
604
|
+
}
|
|
605
|
+
detach() {
|
|
606
|
+
this._scrollListener?.();
|
|
607
|
+
this._spacerTop?.remove();
|
|
608
|
+
this._innerEl?.remove();
|
|
609
|
+
this._spacerBottom?.remove();
|
|
610
|
+
this._scrollListener = null;
|
|
611
|
+
this._container = null;
|
|
612
|
+
this._rendered = [];
|
|
613
|
+
}
|
|
614
|
+
updateItems(items) {
|
|
615
|
+
this._items = items;
|
|
616
|
+
this._render();
|
|
617
|
+
}
|
|
618
|
+
scrollTo(index) {
|
|
619
|
+
if (!this._container) return;
|
|
620
|
+
this._container.scrollTop = index * this._opts.itemHeight;
|
|
621
|
+
}
|
|
622
|
+
getVisibleRange() {
|
|
623
|
+
return { ...this._lastRange };
|
|
624
|
+
}
|
|
625
|
+
onRangeChange(handler) {
|
|
626
|
+
this._rangeHandlers.push(handler);
|
|
627
|
+
return () => {
|
|
628
|
+
this._rangeHandlers = this._rangeHandlers.filter((h) => h !== handler);
|
|
629
|
+
};
|
|
630
|
+
}
|
|
631
|
+
_render() {
|
|
632
|
+
if (!this._container || !this._innerEl || !this._spacerTop || !this._spacerBottom || !this._renderFn) return;
|
|
633
|
+
const scrollTop = this._container.scrollTop;
|
|
634
|
+
const viewHeight = this._container.clientHeight;
|
|
635
|
+
const { itemHeight, overscan } = this._opts;
|
|
636
|
+
const total = this._items.length;
|
|
637
|
+
const start = Math.max(0, Math.floor(scrollTop / itemHeight) - overscan);
|
|
638
|
+
const end = Math.min(total, Math.ceil((scrollTop + viewHeight) / itemHeight) + overscan);
|
|
639
|
+
this._lastRange = { start, end, offsetTop: start * itemHeight };
|
|
640
|
+
this._rangeHandlers.forEach((h) => h(this._lastRange));
|
|
641
|
+
this._spacerTop.style.height = `${start * itemHeight}px`;
|
|
642
|
+
this._spacerBottom.style.height = `${(total - end) * itemHeight}px`;
|
|
643
|
+
this._innerEl.innerHTML = "";
|
|
644
|
+
this._rendered = [];
|
|
645
|
+
for (let i = start; i < end; i++) {
|
|
646
|
+
const el = this._renderFn(this._items[i], i);
|
|
647
|
+
el.style.height = `${itemHeight}px`;
|
|
648
|
+
this._innerEl.appendChild(el);
|
|
649
|
+
this._rendered.push(el);
|
|
650
|
+
}
|
|
651
|
+
}
|
|
652
|
+
};
|
|
653
|
+
function _spacer() {
|
|
654
|
+
const el = document.createElement("div");
|
|
655
|
+
el.style.height = "0";
|
|
656
|
+
el.style.pointerEvents = "none";
|
|
657
|
+
return el;
|
|
658
|
+
}
|
|
659
|
+
|
|
660
|
+
// src/ui/banking-keyboard.ts
|
|
661
|
+
function shuffle(arr) {
|
|
662
|
+
const a = [...arr];
|
|
663
|
+
for (let i = a.length - 1; i > 0; i--) {
|
|
664
|
+
const j = Math.floor(Math.random() * (i + 1));
|
|
665
|
+
[a[i], a[j]] = [a[j], a[i]];
|
|
666
|
+
}
|
|
667
|
+
return a;
|
|
668
|
+
}
|
|
669
|
+
var JoopBankingKeyboard = class {
|
|
670
|
+
_container = null;
|
|
671
|
+
_value = "";
|
|
672
|
+
_config;
|
|
673
|
+
_opts;
|
|
674
|
+
constructor(opts = {}) {
|
|
675
|
+
this._opts = opts;
|
|
676
|
+
this._config = {
|
|
677
|
+
mode: opts.mode ?? "pin",
|
|
678
|
+
scramble: opts.scramble ?? true,
|
|
679
|
+
haptic: opts.haptic ?? true,
|
|
680
|
+
maskInput: opts.maskInput ?? (opts.mode === "amount" ? false : true),
|
|
681
|
+
maxLength: opts.maxLength,
|
|
682
|
+
theme: opts.theme ?? "light"
|
|
683
|
+
};
|
|
684
|
+
}
|
|
685
|
+
render(container) {
|
|
686
|
+
this._container = container;
|
|
687
|
+
this._container.innerHTML = "";
|
|
688
|
+
this._container.setAttribute("role", "application");
|
|
689
|
+
this._container.setAttribute("aria-label", "Secure keyboard");
|
|
690
|
+
this._container.style.cssText = `
|
|
691
|
+
display: grid; gap: 8px; padding: 12px;
|
|
692
|
+
background: ${this._config.theme === "dark" ? "#1a1a2e" : "#f8f9fa"};
|
|
693
|
+
border-radius: 12px; user-select: none; touch-action: manipulation;
|
|
694
|
+
`;
|
|
695
|
+
this._renderDisplay();
|
|
696
|
+
this._renderKeys();
|
|
697
|
+
}
|
|
698
|
+
getValue() {
|
|
699
|
+
return this._value;
|
|
700
|
+
}
|
|
701
|
+
clear() {
|
|
702
|
+
this._value = "";
|
|
703
|
+
this._opts.onClear?.();
|
|
704
|
+
this._opts.onInput?.("");
|
|
705
|
+
this._refresh();
|
|
706
|
+
}
|
|
707
|
+
destroy() {
|
|
708
|
+
if (this._container) {
|
|
709
|
+
this._container.innerHTML = "";
|
|
710
|
+
this._container = null;
|
|
711
|
+
}
|
|
712
|
+
}
|
|
713
|
+
_renderDisplay() {
|
|
714
|
+
if (!this._container) return;
|
|
715
|
+
const display = document.createElement("div");
|
|
716
|
+
const maxLen = this._config.maxLength ?? (this._config.mode === "pin" ? 6 : 16);
|
|
717
|
+
const dots = this._config.maskInput ? Array.from({ length: maxLen }, (_, i) => `<span style="
|
|
718
|
+
display:inline-block; width:12px; height:12px; border-radius:50%; margin:0 4px;
|
|
719
|
+
background:${i < this._value.length ? "#2563eb" : "#d1d5db"};
|
|
720
|
+
transition:background .15s;
|
|
721
|
+
"></span>`).join("") : `<span style="font-size:24px;font-weight:700;letter-spacing:4px;color:${this._config.theme === "dark" ? "#fff" : "#111"}">${this._value || "\xA0"}</span>`;
|
|
722
|
+
display.style.cssText = "text-align:center; padding:12px 0; min-height:40px;";
|
|
723
|
+
display.innerHTML = dots;
|
|
724
|
+
this._container.appendChild(display);
|
|
725
|
+
}
|
|
726
|
+
_renderKeys() {
|
|
727
|
+
if (!this._container) return;
|
|
728
|
+
let digits = ["1", "2", "3", "4", "5", "6", "7", "8", "9", "0"];
|
|
729
|
+
if (this._config.scramble) {
|
|
730
|
+
const numericPart = shuffle(["1", "2", "3", "4", "5", "6", "7", "8", "9"]);
|
|
731
|
+
digits = [...numericPart, "0"];
|
|
732
|
+
}
|
|
733
|
+
const grid = document.createElement("div");
|
|
734
|
+
grid.style.cssText = "display:grid; grid-template-columns: repeat(3,1fr); gap:8px;";
|
|
735
|
+
const btnStyle = (bg, color) => `background:${bg}; color:${color}; border:none; border-radius:10px; font-size:22px;
|
|
736
|
+
font-weight:600; padding:16px; cursor:pointer; transition:opacity .1s;
|
|
737
|
+
-webkit-tap-highlight-color:transparent;`;
|
|
738
|
+
const keyColor = this._config.theme === "dark" ? "#2a2a3e" : "#ffffff";
|
|
739
|
+
const textColor = this._config.theme === "dark" ? "#e2e8f0" : "#1a202c";
|
|
740
|
+
digits.slice(0, 9).forEach((d) => {
|
|
741
|
+
const btn = this._makeKey(d, btnStyle(keyColor, textColor));
|
|
742
|
+
grid.appendChild(btn);
|
|
743
|
+
});
|
|
744
|
+
if (this._config.mode === "amount" || this._config.mode === "numeric") {
|
|
745
|
+
const decBtn = this._makeKey(".", btnStyle(keyColor, textColor));
|
|
746
|
+
grid.appendChild(decBtn);
|
|
747
|
+
} else {
|
|
748
|
+
grid.appendChild(this._makeSpacer());
|
|
749
|
+
}
|
|
750
|
+
const zeroBtn = this._makeKey("0", btnStyle(keyColor, textColor));
|
|
751
|
+
grid.appendChild(zeroBtn);
|
|
752
|
+
const backBtn = this._makeKey("\u232B", btnStyle("#ef4444", "#fff"));
|
|
753
|
+
backBtn.setAttribute("aria-label", "Backspace");
|
|
754
|
+
backBtn.addEventListener("click", () => {
|
|
755
|
+
this._value = this._value.slice(0, -1);
|
|
756
|
+
this._opts.onInput?.(this._value);
|
|
757
|
+
this._refresh();
|
|
758
|
+
});
|
|
759
|
+
grid.appendChild(backBtn);
|
|
760
|
+
this._container.appendChild(grid);
|
|
761
|
+
const actionRow = document.createElement("div");
|
|
762
|
+
actionRow.style.cssText = "display:flex; gap:8px; margin-top:4px;";
|
|
763
|
+
const clearBtn = document.createElement("button");
|
|
764
|
+
clearBtn.textContent = "Clear";
|
|
765
|
+
clearBtn.style.cssText = btnStyle("#6b7280", "#fff") + "flex:1; font-size:14px; padding:12px;";
|
|
766
|
+
clearBtn.addEventListener("click", () => this.clear());
|
|
767
|
+
const submitBtn = document.createElement("button");
|
|
768
|
+
submitBtn.textContent = "Submit";
|
|
769
|
+
submitBtn.style.cssText = btnStyle("#2563eb", "#fff") + "flex:2; font-size:14px; padding:12px;";
|
|
770
|
+
submitBtn.addEventListener("click", () => {
|
|
771
|
+
this._opts.onSubmit?.(this._value);
|
|
772
|
+
});
|
|
773
|
+
actionRow.appendChild(clearBtn);
|
|
774
|
+
actionRow.appendChild(submitBtn);
|
|
775
|
+
this._container.appendChild(actionRow);
|
|
776
|
+
}
|
|
777
|
+
_makeKey(label, style) {
|
|
778
|
+
const btn = document.createElement("button");
|
|
779
|
+
btn.textContent = label;
|
|
780
|
+
btn.style.cssText = style;
|
|
781
|
+
btn.addEventListener("click", () => {
|
|
782
|
+
const maxLen = this._config.maxLength ?? (this._config.mode === "pin" ? 6 : this._config.mode === "otp" ? 6 : 16);
|
|
783
|
+
if (this._value.length >= maxLen) return;
|
|
784
|
+
if (this._config.haptic && navigator.vibrate) navigator.vibrate(10);
|
|
785
|
+
this._value += label;
|
|
786
|
+
this._opts.onInput?.(this._value);
|
|
787
|
+
this._refresh();
|
|
788
|
+
});
|
|
789
|
+
return btn;
|
|
790
|
+
}
|
|
791
|
+
_makeSpacer() {
|
|
792
|
+
const el = document.createElement("div");
|
|
793
|
+
return el;
|
|
794
|
+
}
|
|
795
|
+
_refresh() {
|
|
796
|
+
if (!this._container) return;
|
|
797
|
+
this.render(this._container);
|
|
798
|
+
}
|
|
799
|
+
};
|
|
800
|
+
|
|
801
|
+
export { JoopA11yService, JoopAlertService, JoopBankingKeyboard, JoopLoaderService, JoopPaginationService, JoopPrintService, JoopVirtualScroll };
|
|
802
|
+
//# sourceMappingURL=index.mjs.map
|
|
803
|
+
//# sourceMappingURL=index.mjs.map
|