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,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