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