@qbs-origin/origin-form 0.5.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 (182) hide show
  1. package/README.md +24 -0
  2. package/esm2022/lib/action-step-handler.mjs +163 -0
  3. package/esm2022/lib/auth-client.service.mjs +69 -0
  4. package/esm2022/lib/enums/label.keys.mjs +721 -0
  5. package/esm2022/lib/form-css.helper.mjs +367 -0
  6. package/esm2022/lib/formly/baseFormlyControlComponent.mjs +52 -0
  7. package/esm2022/lib/formly/baseFormlyStepComponent.mjs +59 -0
  8. package/esm2022/lib/formly/custom-section-separator.component.mjs +32 -0
  9. package/esm2022/lib/formly/form-section-separator.component.mjs +36 -0
  10. package/esm2022/lib/formly/formly-action.mjs +56 -0
  11. package/esm2022/lib/formly/formly-checkbox/formly-checkbox.component.mjs +52 -0
  12. package/esm2022/lib/formly/formly-dictionary-dropdown-tree/formly-dictionary-dropdown-tree.component.mjs +261 -0
  13. package/esm2022/lib/formly/formly-download-documents/formly-download-documents.component.mjs +126 -0
  14. package/esm2022/lib/formly/formly-enrol-card/formly-enrol-card.component.mjs +120 -0
  15. package/esm2022/lib/formly/formly-field-stepper/formly-field-stepper.component.mjs +762 -0
  16. package/esm2022/lib/formly/formly-generate-documents/formly-generate-documents.component.mjs +57 -0
  17. package/esm2022/lib/formly/formly-identification.component.mjs +84 -0
  18. package/esm2022/lib/formly/formly-open-banking/formly-open-banking.component.mjs +590 -0
  19. package/esm2022/lib/formly/formly-paragraph/formly-paragraph.component.mjs +35 -0
  20. package/esm2022/lib/formly/formly-radio/formly-radio-component.mjs +49 -0
  21. package/esm2022/lib/formly/formly-row-fille.mjs +12 -0
  22. package/esm2022/lib/formly/formly-scan-id/formly-scan-id.component.mjs +284 -0
  23. package/esm2022/lib/formly/formly-sign/formly-sign.component.mjs +173 -0
  24. package/esm2022/lib/formly/formly-upload-documents/formly-upload-documents.component.mjs +198 -0
  25. package/esm2022/lib/formly/formly-validate-contact-info/formly-validate-contact-info.component.mjs +124 -0
  26. package/esm2022/lib/formly/formly-view-documents/formly-view-documents.component.mjs +245 -0
  27. package/esm2022/lib/formly/formly-view-offers/formly-view-offers.component.mjs +160 -0
  28. package/esm2022/lib/model-population.helper.mjs +265 -0
  29. package/esm2022/lib/models/application-type.model.mjs +12 -0
  30. package/esm2022/lib/models/application.model.mjs +30 -0
  31. package/esm2022/lib/models/auth/users.model.mjs +2 -0
  32. package/esm2022/lib/models/dictionary.model.mjs +20 -0
  33. package/esm2022/lib/models/flux.model.mjs +105 -0
  34. package/esm2022/lib/models/forms.model.mjs +572 -0
  35. package/esm2022/lib/models/label-info.model.mjs +2 -0
  36. package/esm2022/lib/models/label.model.mjs +2 -0
  37. package/esm2022/lib/models/language.model.mjs +3 -0
  38. package/esm2022/lib/models/list.model.mjs +2 -0
  39. package/esm2022/lib/models/partner.model.mjs +3 -0
  40. package/esm2022/lib/models/treeview.model.mjs +15 -0
  41. package/esm2022/lib/origin-form-auth.service.mjs +40 -0
  42. package/esm2022/lib/origin-form-config.model.mjs +2 -0
  43. package/esm2022/lib/origin-form-token.interceptor.mjs +35 -0
  44. package/esm2022/lib/origin-form.component.mjs +2391 -0
  45. package/esm2022/lib/origin-form.module.mjs +479 -0
  46. package/esm2022/lib/origin-form.service.mjs +14 -0
  47. package/esm2022/lib/others/check-list.database.mjs +55 -0
  48. package/esm2022/lib/others/config-service.mjs +42 -0
  49. package/esm2022/lib/others/dictionary-label-info.mjs +3 -0
  50. package/esm2022/lib/others/environment-type.mjs +21 -0
  51. package/esm2022/lib/others/external-link.directive.mjs +49 -0
  52. package/esm2022/lib/others/flux-helper.mjs +1397 -0
  53. package/esm2022/lib/others/picker.component.mjs +119 -0
  54. package/esm2022/lib/others/translation.pipe.mjs +21 -0
  55. package/esm2022/lib/others/translations-helper.mjs +258 -0
  56. package/esm2022/lib/others/utils.mjs +272 -0
  57. package/esm2022/lib/services/applicationData.service.mjs +145 -0
  58. package/esm2022/lib/services/auth-http.service.mjs +80 -0
  59. package/esm2022/lib/services/dialog.service.mjs +56 -0
  60. package/esm2022/lib/services/dictionary.service.mjs +198 -0
  61. package/esm2022/lib/services/forms.service.mjs +47 -0
  62. package/esm2022/lib/services/labels.service.mjs +29 -0
  63. package/esm2022/lib/services/language.service.mjs +24 -0
  64. package/esm2022/lib/services/open-banking.service.mjs +194 -0
  65. package/esm2022/lib/services/origin-form-signalr-handler.service.mjs +107 -0
  66. package/esm2022/lib/services/origin-form-signalr.service.mjs +105 -0
  67. package/esm2022/lib/services/otp.service.mjs +28 -0
  68. package/esm2022/lib/services/proxy.service.mjs +79 -0
  69. package/esm2022/lib/services/scroll-to-error.service.mjs +369 -0
  70. package/esm2022/lib/services/translation.service.mjs +27 -0
  71. package/esm2022/lib/shared-components/confirmation.component.mjs +34 -0
  72. package/esm2022/lib/shared-components/dictionaries-tree.component.mjs +301 -0
  73. package/esm2022/lib/shared-components/grid.component.mjs +241 -0
  74. package/esm2022/lib/shared-components/treeview/treeview.component.mjs +224 -0
  75. package/esm2022/lib/theme-css.mjs +2254 -0
  76. package/esm2022/lib/theme-injector.service.mjs +26 -0
  77. package/esm2022/public-api.mjs +4 -0
  78. package/esm2022/qbs-origin-origin-form.mjs +5 -0
  79. package/fesm2022/qbs-origin-origin-form.mjs +15215 -0
  80. package/fesm2022/qbs-origin-origin-form.mjs.map +1 -0
  81. package/index.d.ts +5 -0
  82. package/lib/action-step-handler.d.ts +49 -0
  83. package/lib/auth-client.service.d.ts +17 -0
  84. package/lib/enums/label.keys.d.ts +720 -0
  85. package/lib/form-css.helper.d.ts +28 -0
  86. package/lib/formly/baseFormlyControlComponent.d.ts +25 -0
  87. package/lib/formly/baseFormlyStepComponent.d.ts +29 -0
  88. package/lib/formly/custom-section-separator.component.d.ts +6 -0
  89. package/lib/formly/form-section-separator.component.d.ts +10 -0
  90. package/lib/formly/formly-action.d.ts +13 -0
  91. package/lib/formly/formly-checkbox/formly-checkbox.component.d.ts +15 -0
  92. package/lib/formly/formly-dictionary-dropdown-tree/formly-dictionary-dropdown-tree.component.d.ts +45 -0
  93. package/lib/formly/formly-download-documents/formly-download-documents.component.d.ts +22 -0
  94. package/lib/formly/formly-enrol-card/formly-enrol-card.component.d.ts +114 -0
  95. package/lib/formly/formly-field-stepper/formly-field-stepper.component.d.ts +79 -0
  96. package/lib/formly/formly-generate-documents/formly-generate-documents.component.d.ts +17 -0
  97. package/lib/formly/formly-identification.component.d.ts +19 -0
  98. package/lib/formly/formly-open-banking/formly-open-banking.component.d.ts +119 -0
  99. package/lib/formly/formly-paragraph/formly-paragraph.component.d.ts +10 -0
  100. package/lib/formly/formly-radio/formly-radio-component.d.ts +15 -0
  101. package/lib/formly/formly-row-fille.d.ts +6 -0
  102. package/lib/formly/formly-scan-id/formly-scan-id.component.d.ts +41 -0
  103. package/lib/formly/formly-sign/formly-sign.component.d.ts +36 -0
  104. package/lib/formly/formly-upload-documents/formly-upload-documents.component.d.ts +25 -0
  105. package/lib/formly/formly-validate-contact-info/formly-validate-contact-info.component.d.ts +79 -0
  106. package/lib/formly/formly-view-documents/formly-view-documents.component.d.ts +33 -0
  107. package/lib/formly/formly-view-offers/formly-view-offers.component.d.ts +23 -0
  108. package/lib/model-population.helper.d.ts +8 -0
  109. package/lib/models/application-type.model.d.ts +27 -0
  110. package/lib/models/application.model.d.ts +107 -0
  111. package/lib/models/auth/users.model.d.ts +20 -0
  112. package/lib/models/dictionary.model.d.ts +77 -0
  113. package/lib/models/flux.model.d.ts +101 -0
  114. package/lib/models/forms.model.d.ts +504 -0
  115. package/lib/models/label-info.model.d.ts +10 -0
  116. package/lib/models/label.model.d.ts +4 -0
  117. package/lib/models/language.model.d.ts +5 -0
  118. package/lib/models/list.model.d.ts +8 -0
  119. package/lib/models/partner.model.d.ts +12 -0
  120. package/lib/models/treeview.model.d.ts +17 -0
  121. package/lib/origin-form-auth.service.d.ts +15 -0
  122. package/lib/origin-form-config.model.d.ts +12 -0
  123. package/lib/origin-form-token.interceptor.d.ts +12 -0
  124. package/lib/origin-form.component.d.ts +231 -0
  125. package/lib/origin-form.module.d.ts +84 -0
  126. package/lib/origin-form.service.d.ts +6 -0
  127. package/lib/others/check-list.database.d.ts +16 -0
  128. package/lib/others/config-service.d.ts +22 -0
  129. package/lib/others/dictionary-label-info.d.ts +6 -0
  130. package/lib/others/environment-type.d.ts +8 -0
  131. package/lib/others/external-link.directive.d.ts +12 -0
  132. package/lib/others/flux-helper.d.ts +115 -0
  133. package/lib/others/picker.component.d.ts +36 -0
  134. package/lib/others/translation.pipe.d.ts +10 -0
  135. package/lib/others/translations-helper.d.ts +31 -0
  136. package/lib/others/utils.d.ts +37 -0
  137. package/lib/services/applicationData.service.d.ts +35 -0
  138. package/lib/services/auth-http.service.d.ts +21 -0
  139. package/lib/services/dialog.service.d.ts +20 -0
  140. package/lib/services/dictionary.service.d.ts +89 -0
  141. package/lib/services/forms.service.d.ts +17 -0
  142. package/lib/services/labels.service.d.ts +13 -0
  143. package/lib/services/language.service.d.ts +14 -0
  144. package/lib/services/open-banking.service.d.ts +137 -0
  145. package/lib/services/origin-form-signalr-handler.service.d.ts +29 -0
  146. package/lib/services/origin-form-signalr.service.d.ts +24 -0
  147. package/lib/services/otp.service.d.ts +22 -0
  148. package/lib/services/proxy.service.d.ts +29 -0
  149. package/lib/services/scroll-to-error.service.d.ts +54 -0
  150. package/lib/services/translation.service.d.ts +10 -0
  151. package/lib/shared-components/confirmation.component.d.ts +77 -0
  152. package/lib/shared-components/dictionaries-tree.component.d.ts +51 -0
  153. package/lib/shared-components/grid.component.d.ts +138 -0
  154. package/lib/shared-components/treeview/treeview.component.d.ts +121 -0
  155. package/lib/theme-css.d.ts +2 -0
  156. package/lib/theme-injector.service.d.ts +8 -0
  157. package/package.json +42 -0
  158. package/public-api.d.ts +3 -0
  159. package/schematics-compiled/collection.json +10 -0
  160. package/schematics-compiled/ng-add/index.d.ts +2 -0
  161. package/schematics-compiled/ng-add/index.js +67 -0
  162. package/schematics-compiled/ng-add/index.js.map +1 -0
  163. package/schematics-compiled/ng-add/schema.json +8 -0
  164. package/src/lib/assets/fonts/Figtree-Bold.ttf +0 -0
  165. package/src/lib/assets/fonts/Figtree-Light.ttf +0 -0
  166. package/src/lib/assets/fonts/Figtree-Regular.ttf +0 -0
  167. package/src/lib/assets/fonts/Sora-ExtraBold.ttf +0 -0
  168. package/src/lib/assets/fonts/Sora-Light.ttf +0 -0
  169. package/src/lib/assets/fonts/Sora-Regular.ttf +0 -0
  170. package/src/lib/assets/fonts/ttrounds-bold-webfont.woff +0 -0
  171. package/src/lib/assets/fonts/ttrounds-bold-webfont.woff2 +0 -0
  172. package/src/lib/assets/fonts/ttrounds-regular-webfont.woff +0 -0
  173. package/src/lib/assets/fonts/ttrounds-regular-webfont.woff2 +0 -0
  174. package/src/lib/assets/fonts/ttrounds-thin-webfont.woff +0 -0
  175. package/src/lib/assets/fonts/ttrounds-thin-webfont.woff2 +0 -0
  176. package/src/lib/assets/images/flag/icon-flag-de.svg +10 -0
  177. package/src/lib/assets/images/flag/icon-flag-en.svg +1 -0
  178. package/src/lib/assets/images/flag/icon-flag-es.svg +11 -0
  179. package/src/lib/assets/images/flag/icon-flag-fr.svg +1 -0
  180. package/src/lib/assets/images/flag/icon-flag-ro.svg +11 -0
  181. package/src/lib/assets/images/flag/origin-form/new-id-card.png +0 -0
  182. package/src/lib/assets/images/flag/origin-form/old-id-card.png +0 -0
@@ -0,0 +1,590 @@
1
+ import { Component, } from '@angular/core';
2
+ import { FormControl } from '@angular/forms';
3
+ import { BehaviorSubject, interval } from 'rxjs';
4
+ import { switchMap, takeWhile, finalize } from 'rxjs/operators';
5
+ import { BaseFormlyStepComponent } from '../baseFormlyStepComponent';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "../../services/open-banking.service";
8
+ import * as i2 from "../../services/dialog.service";
9
+ import * as i3 from "@angular/platform-browser";
10
+ import * as i4 from "@angular/common";
11
+ import * as i5 from "@angular/material/button";
12
+ import * as i6 from "@angular/material/icon";
13
+ import * as i7 from "@angular/material/progress-spinner";
14
+ var OpenBankingState;
15
+ (function (OpenBankingState) {
16
+ OpenBankingState["Initial"] = "initial";
17
+ OpenBankingState["BankSelection"] = "bankSelection";
18
+ OpenBankingState["Consent"] = "consent";
19
+ OpenBankingState["Processing"] = "processing";
20
+ OpenBankingState["Error"] = "error";
21
+ OpenBankingState["Complete"] = "complete";
22
+ })(OpenBankingState || (OpenBankingState = {}));
23
+ export class FormlyOpenBankingComponent extends BaseFormlyStepComponent {
24
+ constructor(openBankingService, dialog, sanitizer, cdr) {
25
+ super(cdr);
26
+ this.openBankingService = openBankingService;
27
+ this.dialog = dialog;
28
+ this.sanitizer = sanitizer;
29
+ this.config = {};
30
+ this.currentState = new BehaviorSubject(OpenBankingState.Initial);
31
+ this.banks = [];
32
+ this.selectedBank = '';
33
+ this.authorizationUrl = null;
34
+ this.consentId = '';
35
+ this._errorMessage = '';
36
+ this._isConsentError = false;
37
+ this._consentErrorStatus = '';
38
+ this.isLoading = false;
39
+ this.isLoadingSubject = new BehaviorSubject(false);
40
+ this.accounts = [];
41
+ this.transactions = [];
42
+ this.selectedAccounts = [];
43
+ this.transactionPages = new Map();
44
+ }
45
+ async ngOnInit() {
46
+ await super.ngOnInit();
47
+ if (this.stepData?.config) {
48
+ this.config = this.stepData.config;
49
+ }
50
+ // If translations are in stepData.translations, merge them into config
51
+ if (this.stepData?.translations) {
52
+ this.mergeTranslationsIntoConfig();
53
+ }
54
+ }
55
+ mergeTranslationsIntoConfig() {
56
+ // Map translation names to config properties
57
+ const translationMap = {
58
+ 'initialMessageTranslations': 'initialMessageTranslations',
59
+ 'waitingMessageTranslations': 'waitingMessageTranslations',
60
+ 'consentErrorMessageTranslations': 'consentErrorMessageTranslations',
61
+ 'completionMessageTranslations': 'completionMessageTranslations',
62
+ 'retryButtonLabelTranslations': 'retryButtonLabelTranslations',
63
+ 'nextButtonTranslations': 'nextButtonTranslations',
64
+ 'continueButtonTranslations': 'continueButtonTranslations',
65
+ 'selectBankTranslations': 'selectBankTranslations',
66
+ 'consentTitleTranslations': 'consentTitleTranslations',
67
+ 'consentDescriptionTranslations': 'consentDescriptionTranslations',
68
+ 'waitingForAuthorizationTranslations': 'waitingForAuthorizationTranslations',
69
+ 'popupWindowMessageTranslations': 'popupWindowMessageTranslations',
70
+ 'popupBlockerMessageTranslations': 'popupBlockerMessageTranslations',
71
+ 'accountsRetrievedTranslations': 'accountsRetrievedTranslations',
72
+ 'transactionsRetrievedTranslations': 'transactionsRetrievedTranslations',
73
+ 'accountsFoundTranslations': 'accountsFoundTranslations',
74
+ 'transactionsSummaryTranslations': 'transactionsSummaryTranslations'
75
+ };
76
+ this.stepData.translations?.forEach((translation) => {
77
+ const configKey = translationMap[translation.name];
78
+ if (configKey && translation.values && translation.values.length > 0) {
79
+ // Use the translation values from stepData if config doesn't have all languages
80
+ this.config[configKey] = translation.values;
81
+ }
82
+ });
83
+ }
84
+ ngAfterViewInit() {
85
+ // Only set initial state if we haven't already set a state
86
+ if (!this.currentState.value) {
87
+ this.currentState.next(OpenBankingState.Initial);
88
+ }
89
+ }
90
+ onPageSelected() {
91
+ // Only reset to initial if we haven't started the flow yet
92
+ // This prevents reset when language changes
93
+ if (!this.currentState.value) {
94
+ this.currentState.next(OpenBankingState.Initial);
95
+ }
96
+ // Update language if it changed
97
+ this.cdr.detectChanges();
98
+ }
99
+ determineNextStep() {
100
+ // Get account information from form if configured
101
+ const accountNumber = this.getFieldValue(this.config.accountNumberCollected);
102
+ // If we should extract bank from IBAN and have an IBAN
103
+ if (this.config.extractBankFromIBAN && accountNumber && this.isIBAN(accountNumber)) {
104
+ // Load banks first, then match IBAN to bank
105
+ this.loadBanksAndMatchIBAN(accountNumber);
106
+ return;
107
+ }
108
+ // If we should show all banks
109
+ if (this.config.showAllBanks) {
110
+ this.loadBanks();
111
+ }
112
+ else {
113
+ // Go directly to consent without bank selection
114
+ this.initiateConsent();
115
+ }
116
+ }
117
+ getFieldValue(fieldId) {
118
+ if (!fieldId)
119
+ return '';
120
+ return this.form.get(fieldId)?.value || '';
121
+ }
122
+ isIBAN(value) {
123
+ // Basic IBAN validation for Romanian IBANs
124
+ return /^RO\d{2}[A-Z]{4}\d{16}$/.test(value.replace(/\s/g, ''));
125
+ }
126
+ extractBankFromIBAN(iban) {
127
+ // Extract bank code from Romanian IBAN (positions 5-8)
128
+ const cleanIban = iban.replace(/\s/g, '');
129
+ if (cleanIban.startsWith('RO') && cleanIban.length >= 8) {
130
+ return cleanIban.substring(4, 8);
131
+ }
132
+ return '';
133
+ }
134
+ loadBanksAndMatchIBAN(iban) {
135
+ this.setLoadingState(true);
136
+ const countryCode = 'RO'; // Default to Romania
137
+ this.openBankingService.getBanks(countryCode).subscribe({
138
+ next: (banks) => {
139
+ this.banks = banks.filter(b => b.isActive);
140
+ // Extract bank code from IBAN
141
+ const ibanBankCode = this.extractBankFromIBAN(iban);
142
+ if (ibanBankCode) {
143
+ // Try to match against bank codes
144
+ const matchedBank = this.banks.find(bank => bank.bankCode.toUpperCase() === ibanBankCode.toUpperCase());
145
+ if (matchedBank) {
146
+ // Bank found - select it and go directly to consent
147
+ this.selectedBank = matchedBank.bankCode;
148
+ this.selectedBankObject = matchedBank;
149
+ this.setLoadingState(false);
150
+ this.initiateConsent();
151
+ return;
152
+ }
153
+ }
154
+ // If no match found, show bank selection
155
+ this.currentState.next(OpenBankingState.BankSelection);
156
+ this.setLoadingState(false);
157
+ },
158
+ error: (error) => {
159
+ this.handleError(error);
160
+ this.setLoadingState(false);
161
+ },
162
+ });
163
+ }
164
+ loadBanks() {
165
+ this.setLoadingState(true);
166
+ const countryCode = 'RO'; // Default to Romania
167
+ this.openBankingService.getBanks(countryCode).subscribe({
168
+ next: (banks) => {
169
+ this.banks = banks.filter(b => b.isActive);
170
+ this.currentState.next(OpenBankingState.BankSelection);
171
+ this.setLoadingState(false);
172
+ },
173
+ error: (error) => {
174
+ this.handleError(error);
175
+ this.setLoadingState(false);
176
+ },
177
+ });
178
+ }
179
+ selectBank(bank) {
180
+ this.selectedBank = bank.bankCode;
181
+ this.selectedBankObject = bank;
182
+ }
183
+ proceedToNextStep() {
184
+ if (this.currentState.value === OpenBankingState.Initial) {
185
+ // User clicked Next on initial screen - determine what to show next
186
+ this.determineNextStep();
187
+ }
188
+ else if (this.currentState.value === OpenBankingState.BankSelection) {
189
+ if (this.selectedBank) {
190
+ this.initiateConsent();
191
+ }
192
+ }
193
+ }
194
+ initiateConsent() {
195
+ this.setLoadingState(true);
196
+ // Get user information
197
+ const email = this.getFieldValue(this.config.emailCollected) || 'user@example.com';
198
+ // Generate a unique PSU ID for this session
199
+ const psuId = this.appDataId || 'user-' + Date.now() + '-' + Math.random().toString(36).substr(2, 9);
200
+ const consentRequest = {
201
+ bankCode: this.selectedBank || 'DEFAULT',
202
+ psuEmail: email,
203
+ psuIntermediaryId: psuId,
204
+ periodOfValidity: 90,
205
+ tcAccepted: this.config.tcAcceptedByDefault !== false
206
+ };
207
+ this.openBankingService.initializeConsent(consentRequest).subscribe({
208
+ next: (response) => {
209
+ this.consentId = response.consentId;
210
+ // Check if bank requires redirect
211
+ if (this.selectedBankObject?.requiresRedirect !== false) {
212
+ // Try iframe first, but most banks will block it
213
+ if (this.shouldUseIframe()) {
214
+ this.authorizationUrl = this.sanitizer.bypassSecurityTrustResourceUrl(response.authorizationUrl);
215
+ this.currentState.next(OpenBankingState.Consent);
216
+ // Set a flag to detect if iframe fails
217
+ this.checkIframeLoad(response.authorizationUrl);
218
+ }
219
+ else {
220
+ // Open in new window/tab
221
+ this.openConsentInNewWindow(response.authorizationUrl);
222
+ }
223
+ }
224
+ this.setLoadingState(false);
225
+ // Start polling for consent status
226
+ this.startConsentPolling();
227
+ },
228
+ error: (error) => {
229
+ this.handleError(error);
230
+ this.setLoadingState(false);
231
+ },
232
+ });
233
+ }
234
+ shouldUseIframe() {
235
+ // Most banks don't allow iframes due to X-Frame-Options
236
+ // So default to false (use popup window instead)
237
+ return false;
238
+ }
239
+ openConsentInNewWindow(authorizationUrl) {
240
+ const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
241
+ let consentWindow;
242
+ if (isMobile) {
243
+ // On mobile, open in same tab (full redirect) or new tab
244
+ // Using _blank will open a new tab and allow user to return
245
+ consentWindow = window.open(authorizationUrl, '_blank');
246
+ }
247
+ else {
248
+ // On desktop, open in a popup window
249
+ const width = 800;
250
+ const height = 600;
251
+ const left = (window.screen.width - width) / 2;
252
+ const top = (window.screen.height - height) / 2;
253
+ consentWindow = window.open(authorizationUrl, 'bankConsent', `width=${width},height=${height},left=${left},top=${top},toolbar=no,menubar=no,scrollbars=yes,resizable=yes`);
254
+ }
255
+ if (consentWindow) {
256
+ // Show a message that consent is in progress
257
+ this.currentState.next(OpenBankingState.Consent);
258
+ // Check if window is closed periodically (works better on desktop)
259
+ if (!isMobile) {
260
+ const checkWindowClosed = setInterval(() => {
261
+ if (consentWindow && consentWindow.closed) {
262
+ clearInterval(checkWindowClosed);
263
+ // Window was closed, the polling will detect the consent status
264
+ }
265
+ }, 1000);
266
+ }
267
+ // On mobile, we rely entirely on polling since window.closed doesn't work reliably
268
+ }
269
+ else {
270
+ // Popup was blocked
271
+ this._isConsentError = false;
272
+ this._errorMessage = 'Please allow popups for this site to complete bank authorization. Then click Retry to try again.';
273
+ this.currentState.next(OpenBankingState.Error);
274
+ }
275
+ }
276
+ checkIframeLoad(authorizationUrl) {
277
+ // If iframe fails to load (X-Frame-Options), fall back to popup
278
+ setTimeout(() => {
279
+ // Check if we're still in consent state (iframe might have failed)
280
+ if (this.currentState.value === OpenBankingState.Consent && this.authorizationUrl) {
281
+ // Clear iframe and open in new window
282
+ console.warn('Iframe might be blocked, opening in new window');
283
+ this.authorizationUrl = null;
284
+ this.openConsentInNewWindow(authorizationUrl);
285
+ }
286
+ }, 2000);
287
+ }
288
+ startConsentPolling() {
289
+ // Poll every 5 seconds for consent status (user needs time to complete bank authorization)
290
+ this.pollSubscription = interval(10000)
291
+ .pipe(switchMap(() => this.openBankingService.getConsentStatus(this.selectedBank || 'DEFAULT', this.consentId)), takeWhile((status) => status.status === 'received', true), finalize(() => this.pollSubscription?.unsubscribe()))
292
+ .subscribe({
293
+ next: (status) => {
294
+ if (status.status === 'valid') {
295
+ this.onConsentApproved();
296
+ }
297
+ else if (status.status === 'rejected' ||
298
+ status.status === 'expired' ||
299
+ status.status === 'terminated' ||
300
+ status.status === 'revokedByPsu') {
301
+ this.onConsentRejected(status);
302
+ }
303
+ },
304
+ error: (error) => {
305
+ this.handleError(error);
306
+ },
307
+ });
308
+ }
309
+ onConsentApproved() {
310
+ this.currentState.next(OpenBankingState.Processing);
311
+ this.fetchAccountData();
312
+ }
313
+ onConsentRejected(status) {
314
+ this._isConsentError = true;
315
+ this._consentErrorStatus = status.status;
316
+ this._errorMessage = ''; // Clear to use translated message
317
+ this.currentState.next(OpenBankingState.Error);
318
+ }
319
+ fetchAccountData() {
320
+ this.setLoadingState(true);
321
+ // First, fetch accounts
322
+ this.openBankingService.getAccounts(this.consentId, this.selectedBank || 'DEFAULT').subscribe({
323
+ next: (accounts) => {
324
+ this.accounts = accounts;
325
+ // Select all accounts for transaction fetching
326
+ this.selectedAccounts = accounts.map(a => a.accountId);
327
+ // Fetch transactions for each account
328
+ this.fetchTransactionsForAccounts();
329
+ },
330
+ error: (error) => {
331
+ this.handleError(error);
332
+ this.setLoadingState(false);
333
+ },
334
+ });
335
+ }
336
+ fetchTransactionsForAccounts() {
337
+ if (this.selectedAccounts.length === 0) {
338
+ this.completeProcess();
339
+ return;
340
+ }
341
+ const transactionRequests = this.selectedAccounts.map(accountId => {
342
+ // Calculate date range based on bank-specific limits
343
+ const bankCode = this.selectedBank || 'DEFAULT';
344
+ const maxDays = this.openBankingService.getMaxTransactionPeriod(bankCode);
345
+ const toDate = new Date();
346
+ const fromDate = new Date();
347
+ // If unlimited (-1) or very long period, use 365 days as practical limit
348
+ // Otherwise use the bank's specific limit
349
+ const daysToFetch = maxDays === -1 ? 365 : Math.min(maxDays, 365);
350
+ fromDate.setDate(fromDate.getDate() - daysToFetch);
351
+ return this.openBankingService.getTransactions({
352
+ consentId: this.consentId,
353
+ bankCode: bankCode,
354
+ accountId: accountId,
355
+ fromDate: fromDate.toISOString(),
356
+ toDate: toDate.toISOString(),
357
+ page: 1,
358
+ pageSize: 50
359
+ });
360
+ });
361
+ // Execute all transaction requests
362
+ Promise.all(transactionRequests.map(req => req.toPromise()))
363
+ .then(results => {
364
+ // Store transaction pages
365
+ results.forEach((page, index) => {
366
+ if (page) {
367
+ this.transactionPages.set(this.selectedAccounts[index], page);
368
+ // Flatten all transactions into single array
369
+ this.transactions.push(...page.items);
370
+ }
371
+ });
372
+ this.completeProcess();
373
+ })
374
+ .catch(error => {
375
+ console.error('Error fetching transactions:', error);
376
+ // Even if transactions fail, we can continue with accounts
377
+ this.completeProcess();
378
+ });
379
+ }
380
+ completeProcess() {
381
+ // Store data in form model
382
+ this.storeAccountData();
383
+ this.currentState.next(OpenBankingState.Complete);
384
+ this.setLoadingState(false);
385
+ // Auto-proceed to next step after a short delay
386
+ setTimeout(() => {
387
+ this.onEvent('success');
388
+ }, 2000);
389
+ }
390
+ storeAccountData() {
391
+ // Store the fetched data in the form model for use in later steps
392
+ const group = this.form;
393
+ // Add accounts and transactions to form
394
+ group.addControl('openBankingAccounts', new FormControl(this.accounts));
395
+ group.addControl('openBankingTransactions', new FormControl(this.transactions));
396
+ group.addControl('openBankingConsentId', new FormControl(this.consentId));
397
+ group.addControl('openBankingBankCode', new FormControl(this.selectedBank));
398
+ // Store account summary
399
+ const accountSummary = {
400
+ totalAccounts: this.accounts.length,
401
+ totalBalance: this.accounts.reduce((sum, acc) => sum + acc.balance, 0),
402
+ currencies: [...new Set(this.accounts.map(a => a.currency))],
403
+ transactionCount: this.transactions.length,
404
+ oldestTransaction: this.transactions.length > 0
405
+ ? this.transactions.reduce((oldest, t) => new Date(t.bookingDate || t.transactionDate || '') < new Date(oldest.bookingDate || oldest.transactionDate || '') ? t : oldest).bookingDate
406
+ : null,
407
+ newestTransaction: this.transactions.length > 0
408
+ ? this.transactions.reduce((newest, t) => new Date(t.bookingDate || t.transactionDate || '') > new Date(newest.bookingDate || newest.transactionDate || '') ? t : newest).bookingDate
409
+ : null
410
+ };
411
+ group.addControl('openBankingAccountSummary', new FormControl(accountSummary));
412
+ // Populate the step's controls for SaveAppData
413
+ this.populateStepControls();
414
+ }
415
+ populateStepControls() {
416
+ // Find and populate the controls in the step sections
417
+ if (!this.stepData?.sections)
418
+ return;
419
+ // Get email from config mapping or use empty string
420
+ const email = this.getFieldValue(this.config.emailCollected) || '';
421
+ // Get the first account as the primary account
422
+ const primaryAccount = this.accounts.length > 0 ? this.accounts[0] : null;
423
+ // Calculate total balance
424
+ const totalBalance = this.accounts.reduce((sum, acc) => sum + acc.balance, 0);
425
+ // Calculate transaction period in months
426
+ const months = this.getTransactionPeriodMonths();
427
+ // Map of control identifiers to values
428
+ const controlValues = {
429
+ 'OpenBankingConsentStatus': true, // Consent was successfully validated
430
+ 'OpenBankingEmail': email,
431
+ 'OpenBankingAccountNumber': primaryAccount?.iban || '',
432
+ 'OpenBankingAccountHolder': primaryAccount?.accountHolderName || '',
433
+ 'OpenBankingBankCode': this.selectedBank || '',
434
+ 'OpenBankingConsentId': this.consentId,
435
+ 'OpenBankingAccountsCount': this.accounts.length,
436
+ 'OpenBankingAccountsData': JSON.stringify(this.accounts),
437
+ 'OpenBankingTotalBalance': totalBalance,
438
+ 'OpenBankingTransactionsCount': this.transactions.length,
439
+ 'OpenBankingTransactionsData': JSON.stringify(this.transactions),
440
+ 'OpenBankingTransactionsPeriod': months
441
+ };
442
+ // Add all control values to the form
443
+ const group = this.form;
444
+ Object.keys(controlValues).forEach(controlId => {
445
+ if (!group.contains(controlId)) {
446
+ group.addControl(controlId, new FormControl(controlValues[controlId]));
447
+ }
448
+ else {
449
+ group.get(controlId)?.setValue(controlValues[controlId]);
450
+ }
451
+ });
452
+ // The values are now stored in the form model and will be saved with SaveAppData
453
+ // When the form is submitted, these values will be associated with the step's controls
454
+ }
455
+ retry() {
456
+ this._errorMessage = '';
457
+ this._isConsentError = false;
458
+ this._consentErrorStatus = '';
459
+ this.accounts = [];
460
+ this.transactions = [];
461
+ this.selectedAccounts = [];
462
+ this.transactionPages.clear();
463
+ this.consentId = '';
464
+ this.selectedBank = '';
465
+ this.selectedBankObject = undefined;
466
+ this.banks = [];
467
+ this.authorizationUrl = null;
468
+ // Reset to initial state - user needs to click Next again
469
+ this.currentState.next(OpenBankingState.Initial);
470
+ }
471
+ handleError(error) {
472
+ console.error('OpenBanking error:', error);
473
+ this._isConsentError = false;
474
+ this._consentErrorStatus = '';
475
+ this._errorMessage = error.error?.message || error.message || 'An error occurred. Please try again.';
476
+ this.currentState.next(OpenBankingState.Error);
477
+ }
478
+ getTranslatedMessage(translations, defaultMessage = '') {
479
+ if (!translations || translations.length === 0)
480
+ return defaultMessage;
481
+ const langIso = this.props?.['langIso'] || this.langIso || 'en';
482
+ const translation = translations.find(t => t.languageIso === langIso);
483
+ return translation?.value || translations[0]?.value || defaultMessage;
484
+ }
485
+ get initialMessage() {
486
+ return this.getTranslatedMessage(this.config.initialMessageTranslations, 'To continue, we need to verify your bank account information. Click Next to proceed.');
487
+ }
488
+ get waitingMessage() {
489
+ return this.getTranslatedMessage(this.config.waitingMessageTranslations, 'Please wait while we retrieve your account information...');
490
+ }
491
+ get completionMessage() {
492
+ return this.getTranslatedMessage(this.config.completionMessageTranslations, 'Account information successfully retrieved.');
493
+ }
494
+ get retryButtonLabel() {
495
+ return this.getTranslatedMessage(this.config.retryButtonLabelTranslations, 'Retry');
496
+ }
497
+ get errorMessage() {
498
+ // If it's a consent error, use the translated consent error message
499
+ if (this._isConsentError) {
500
+ const message = this.getTranslatedMessage(this.config.consentErrorMessageTranslations, 'You must provide consent to continue. Please try again.');
501
+ return this._consentErrorStatus ? `${message} (Status: ${this._consentErrorStatus})` : message;
502
+ }
503
+ // Otherwise return the stored error message (from API or default)
504
+ return this._errorMessage || 'An error occurred. Please try again.';
505
+ }
506
+ // Button translations
507
+ get nextButtonLabel() {
508
+ return this.getTranslatedMessage(this.config.nextButtonTranslations, 'Next');
509
+ }
510
+ get continueButtonLabel() {
511
+ return this.getTranslatedMessage(this.config.continueButtonTranslations, 'Continue');
512
+ }
513
+ // UI label translations
514
+ get selectBankTitle() {
515
+ return this.getTranslatedMessage(this.config.selectBankTranslations, 'Select your bank');
516
+ }
517
+ get consentTitle() {
518
+ return this.getTranslatedMessage(this.config.consentTitleTranslations, 'Bank Authorization');
519
+ }
520
+ get consentDescription() {
521
+ return this.getTranslatedMessage(this.config.consentDescriptionTranslations, 'Please complete the authorization process with your bank');
522
+ }
523
+ get waitingForAuthorizationText() {
524
+ return this.getTranslatedMessage(this.config.waitingForAuthorizationTranslations, 'Waiting for authorization...');
525
+ }
526
+ get popupWindowMessage() {
527
+ return this.getTranslatedMessage(this.config.popupWindowMessageTranslations, 'A new window has opened for bank authorization. Please complete the process there.');
528
+ }
529
+ get popupBlockerMessage() {
530
+ return this.getTranslatedMessage(this.config.popupBlockerMessageTranslations, 'If the window didn\'t open, please check your popup blocker settings.');
531
+ }
532
+ get accountsRetrievedLabel() {
533
+ return this.getTranslatedMessage(this.config.accountsRetrievedTranslations, 'Accounts retrieved');
534
+ }
535
+ get transactionsRetrievedLabel() {
536
+ return this.getTranslatedMessage(this.config.transactionsRetrievedTranslations, 'Transactions retrieved');
537
+ }
538
+ get accountsSummaryText() {
539
+ const count = this.accounts.length;
540
+ const template = this.getTranslatedMessage(this.config.accountsFoundTranslations, '{0} account(s) found');
541
+ return template.replace('{0}', count.toString());
542
+ }
543
+ get transactionsSummaryText() {
544
+ const count = this.transactions.length;
545
+ const months = this.getTransactionPeriodMonths();
546
+ const template = this.getTranslatedMessage(this.config.transactionsSummaryTranslations, '{0} transaction(s) retrieved for the last {1} month(s)');
547
+ return template.replace('{0}', count.toString()).replace('{1}', months.toString());
548
+ }
549
+ getTransactionPeriodMonths() {
550
+ if (!this.selectedBank)
551
+ return 3;
552
+ const maxDays = this.openBankingService.getMaxTransactionPeriod(this.selectedBank);
553
+ const daysToFetch = maxDays === -1 ? 365 : Math.min(maxDays, 365);
554
+ return Math.round(daysToFetch / 30);
555
+ }
556
+ get isInitialState() {
557
+ return this.currentState.value === OpenBankingState.Initial;
558
+ }
559
+ get isBankSelectionState() {
560
+ return this.currentState.value === OpenBankingState.BankSelection;
561
+ }
562
+ get isConsentState() {
563
+ return this.currentState.value === OpenBankingState.Consent;
564
+ }
565
+ get isProcessingState() {
566
+ return this.currentState.value === OpenBankingState.Processing;
567
+ }
568
+ get isErrorState() {
569
+ return this.currentState.value === OpenBankingState.Error;
570
+ }
571
+ get isCompleteState() {
572
+ return this.currentState.value === OpenBankingState.Complete;
573
+ }
574
+ setLoadingState(loading) {
575
+ this.isLoading = loading;
576
+ this.isLoadingSubject.next(loading);
577
+ this.cdr.detectChanges();
578
+ }
579
+ ngOnDestroy() {
580
+ this.pollSubscription?.unsubscribe();
581
+ this.isLoadingSubject.complete();
582
+ }
583
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyOpenBankingComponent, deps: [{ token: i1.OpenBankingService }, { token: i2.DialogService }, { token: i3.DomSanitizer }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
584
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FormlyOpenBankingComponent, selector: "app-formly-open-banking", usesInheritance: true, ngImport: i0, template: "<div class=\"open-banking-container\">\n <!-- Initial State -->\n <div *ngIf=\"isInitialState\" class=\"state-initial\">\n <div class=\"message-container\">\n <p class=\"message\">{{ initialMessage }}</p>\n <button mat-raised-button color=\"primary\" (click)=\"proceedToNextStep()\" [disabled]=\"isLoadingSubject | async\">\n {{ nextButtonLabel }}\n </button>\n </div>\n </div>\n\n <!-- Bank Selection State -->\n <div *ngIf=\"isBankSelectionState\" class=\"state-bank-selection\">\n <h3 class=\"section-title\">{{ selectBankTitle }}</h3>\n\n <!-- Bank Grid -->\n <div class=\"banks-grid\" *ngIf=\"banks.length > 0\">\n <div\n *ngFor=\"let bank of banks\"\n class=\"bank-card\"\n [class.selected]=\"selectedBank === bank.bankCode\"\n (click)=\"selectBank(bank)\">\n <div class=\"bank-logo\" *ngIf=\"bank.logoUrl\">\n <img [src]=\"bank.logoUrl\" [alt]=\"bank.displayName\">\n </div>\n <div class=\"bank-info\">\n <h4>{{ bank.displayName }}</h4>\n <span class=\"bank-code\">{{ bank.bankCode }}</span>\n </div>\n <mat-icon class=\"check-icon\" *ngIf=\"selectedBank === bank.bankCode\">check_circle</mat-icon>\n </div>\n </div>\n\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"proceedToNextStep()\"\n [disabled]=\"!selectedBank || (isLoadingSubject | async)\"\n class=\"m-t-16\">\n {{ continueButtonLabel }}\n </button>\n </div>\n\n <!-- Consent State -->\n <div *ngIf=\"isConsentState\" class=\"state-consent\">\n <div class=\"consent-header\">\n <h3>{{ consentTitle }}</h3>\n <p>{{ consentDescription }}</p>\n </div>\n\n <!-- Show iframe only if we have an authorization URL -->\n <div class=\"consent-container\" *ngIf=\"authorizationUrl\">\n <iframe\n [src]=\"authorizationUrl\"\n class=\"consent-iframe\"\n frameborder=\"0\"\n allowfullscreen\n sandbox=\"allow-forms allow-scripts allow-same-origin allow-popups allow-popups-to-escape-sandbox allow-top-navigation\">\n </iframe>\n </div>\n\n <!-- Show waiting message when consent is in popup or iframe failed -->\n <div class=\"consent-waiting\" *ngIf=\"!authorizationUrl\">\n <mat-spinner diameter=\"40\"></mat-spinner>\n <p>{{ waitingForAuthorizationText }}</p>\n <p class=\"consent-popup-hint\">\n <mat-icon>open_in_new</mat-icon>\n <span>{{ popupWindowMessage }}</span>\n </p>\n <p class=\"consent-popup-note\">\n {{ popupBlockerMessage }}\n </p>\n </div>\n </div>\n\n <!-- Processing State -->\n <div *ngIf=\"isProcessingState\" class=\"state-processing\">\n <div class=\"loading-container\">\n <mat-spinner diameter=\"50\"></mat-spinner>\n <p class=\"loading-message\">{{ waitingMessage }}</p>\n <div class=\"progress-details\">\n <p *ngIf=\"accounts.length > 0\">\n {{ accountsRetrievedLabel }}: {{ accounts.length }}\n </p>\n <p *ngIf=\"transactions.length > 0\">\n {{ transactionsRetrievedLabel }}: {{ transactions.length }}\n </p>\n </div>\n </div>\n </div>\n\n <!-- Error State -->\n <div *ngIf=\"isErrorState\" class=\"state-error\">\n <div class=\"error-container\">\n <mat-icon color=\"warn\" class=\"error-icon\">error_outline</mat-icon>\n <p class=\"error-message\">{{ errorMessage }}</p>\n <button mat-raised-button color=\"primary\" (click)=\"retry()\">\n {{ retryButtonLabel }}\n </button>\n </div>\n </div>\n\n <!-- Complete State -->\n <div *ngIf=\"isCompleteState\" class=\"state-complete\">\n <div class=\"success-container\">\n <mat-icon color=\"primary\" class=\"success-icon\">check_circle</mat-icon>\n <p class=\"success-message\">{{ completionMessage }}</p>\n\n <!-- Display summary of accounts found -->\n <div class=\"accounts-summary\" *ngIf=\"accounts.length > 0\">\n <h4>{{ accountsSummaryText }}</h4>\n <div class=\"accounts-list\">\n <div class=\"account-card\" *ngFor=\"let account of accounts\">\n <div class=\"account-header\">\n <mat-icon class=\"account-icon\">account_balance</mat-icon>\n <div class=\"account-info\">\n <div class=\"account-name\">{{ account.name || 'Account' }}</div>\n <div class=\"account-iban\">{{ account.iban }}</div>\n </div>\n </div>\n <div class=\"account-details\">\n <div class=\"account-balance\">\n <span class=\"balance-amount\">{{ account.balance | currency:account.currency:'symbol':'1.2-2' }}</span>\n </div>\n <div class=\"account-holder\">{{ account.accountHolderName }}</div>\n </div>\n </div>\n </div>\n\n <div class=\"transaction-summary\" *ngIf=\"transactions.length > 0\">\n <p>{{ transactionsSummaryText }}</p>\n </div>\n </div>\n </div>\n </div>\n</div>", styles: [".open-banking-container{padding:20px;min-height:400px}.open-banking-container .section-title{margin-bottom:20px;font-size:1.25rem;font-weight:500;color:#000000de}.open-banking-container .message-container,.open-banking-container .loading-container,.open-banking-container .error-container,.open-banking-container .success-container{display:flex;flex-direction:column;align-items:center;justify-content:center;text-align:center;min-height:200px}.open-banking-container .message,.open-banking-container .loading-message,.open-banking-container .error-message,.open-banking-container .success-message{font-size:1rem;margin:20px 0;max-width:600px;line-height:1.5}.open-banking-container .error-message{color:#f44336}.open-banking-container .success-message{color:#4caf50}.open-banking-container .error-icon,.open-banking-container .success-icon{font-size:48px;width:48px;height:48px}.open-banking-container .state-bank-selection .banks-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:16px;margin-bottom:24px}.open-banking-container .state-bank-selection .bank-card{border:2px solid #e0e0e0;border-radius:8px;padding:16px;cursor:pointer;transition:all .3s ease;position:relative;background:#fff;min-height:120px;display:flex;flex-direction:column;align-items:center;justify-content:center}.open-banking-container .state-bank-selection .bank-card:hover{border-color:#3f51b5;box-shadow:0 4px 8px #0000001a;transform:translateY(-2px)}.open-banking-container .state-bank-selection .bank-card.selected{border-color:#3f51b5;background:#f5f7ff}.open-banking-container .state-bank-selection .bank-card .bank-logo{width:80px;height:40px;margin-bottom:8px;display:flex;align-items:center;justify-content:center}.open-banking-container .state-bank-selection .bank-card .bank-logo img{max-width:100%;max-height:100%;object-fit:contain}.open-banking-container .state-bank-selection .bank-card .bank-info{text-align:center}.open-banking-container .state-bank-selection .bank-card .bank-info h4{margin:0;font-size:.9rem;font-weight:500;color:#333}.open-banking-container .state-bank-selection .bank-card .bank-info .bank-code{font-size:.75rem;color:#666;margin-top:4px;display:block}.open-banking-container .state-bank-selection .bank-card .check-icon{position:absolute;top:8px;right:8px;color:#3f51b5;font-size:20px}.open-banking-container .state-bank-selection .bank-dropdown{margin:20px 0}.open-banking-container .state-bank-selection .bank-selector{width:100%;max-width:400px}.open-banking-container .state-consent .consent-header{text-align:center;margin-bottom:20px}.open-banking-container .state-consent .consent-header h3{font-size:1.25rem;font-weight:500;margin-bottom:8px}.open-banking-container .state-consent .consent-header p{color:#666;font-size:.95rem}.open-banking-container .state-consent .consent-container{width:100%;height:600px;border:1px solid #e0e0e0;border-radius:4px;overflow:hidden;background:#fff}.open-banking-container .state-consent .consent-iframe{width:100%;height:100%}.open-banking-container .state-consent .consent-waiting{display:flex;flex-direction:column;align-items:center;justify-content:center;min-height:300px}.open-banking-container .state-consent .consent-waiting p{margin-top:16px;color:#666}.open-banking-container .state-consent .consent-waiting .consent-popup-hint{display:flex;align-items:center;gap:8px;margin-top:24px;font-size:1.1rem;color:#333}.open-banking-container .state-consent .consent-waiting .consent-popup-hint mat-icon{color:#3f51b5}.open-banking-container .state-consent .consent-waiting .consent-popup-note{margin-top:12px;font-size:.9rem;color:#999;font-style:italic}.open-banking-container .progress-details{margin-top:20px}.open-banking-container .progress-details p{font-size:.9rem;color:#666;margin:4px 0}.open-banking-container .accounts-summary{margin-top:30px;width:100%;max-width:800px}.open-banking-container .accounts-summary h4{font-size:1.15rem;font-weight:500;margin-bottom:20px;color:#000000de}.open-banking-container .accounts-summary .accounts-list{display:flex;flex-direction:column;gap:16px}.open-banking-container .accounts-summary .account-card{border:1px solid #e0e0e0;border-radius:8px;padding:16px;background:#fff;box-shadow:0 1px 3px #0000001a;transition:box-shadow .2s ease}.open-banking-container .accounts-summary .account-card:hover{box-shadow:0 2px 6px #00000026}.open-banking-container .accounts-summary .account-header{display:flex;align-items:flex-start;gap:12px;margin-bottom:12px}.open-banking-container .accounts-summary .account-icon{color:#3f51b5;margin-top:2px}.open-banking-container .accounts-summary .account-info{flex:1}.open-banking-container .accounts-summary .account-name{font-weight:500;font-size:1rem;color:#333;margin-bottom:4px}.open-banking-container .accounts-summary .account-iban{font-size:.85rem;color:#666;font-family:Courier New,monospace;letter-spacing:.5px}.open-banking-container .accounts-summary .account-details{display:flex;justify-content:space-between;align-items:center;padding-left:36px}.open-banking-container .accounts-summary .account-balance{display:flex;flex-direction:column}.open-banking-container .accounts-summary .balance-amount{font-size:1.25rem;font-weight:600;color:#4caf50}.open-banking-container .accounts-summary .account-holder{font-size:.9rem;color:#666;text-align:right}.open-banking-container .accounts-summary .transaction-summary{margin-top:16px;padding:12px;background:#e8f5e9;border-radius:4px}.open-banking-container .accounts-summary .transaction-summary p{margin:0;color:#2e7d32;font-size:.95rem}.open-banking-container ::ng-deep .mat-spinner{margin:0 auto}.open-banking-container ::ng-deep .mat-progress-spinner circle,.open-banking-container ::ng-deep .mat-spinner circle{stroke:#3f51b5}@media (max-width: 768px){.open-banking-container{padding:10px}.open-banking-container .state-bank-selection .banks-grid{grid-template-columns:repeat(auto-fill,minmax(150px,1fr));gap:12px}.open-banking-container .state-bank-selection .bank-card{min-height:100px;padding:12px}.open-banking-container .state-bank-selection .bank-card .bank-logo{width:60px;height:30px}.open-banking-container .consent-container{height:500px}.open-banking-container .accounts-summary{max-width:100%}}@media (max-width: 480px){.open-banking-container .state-bank-selection .banks-grid{grid-template-columns:1fr}}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i6.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i7.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i4.CurrencyPipe, name: "currency" }] }); }
585
+ }
586
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyOpenBankingComponent, decorators: [{
587
+ type: Component,
588
+ args: [{ selector: 'app-formly-open-banking', template: "<div class=\"open-banking-container\">\n <!-- Initial State -->\n <div *ngIf=\"isInitialState\" class=\"state-initial\">\n <div class=\"message-container\">\n <p class=\"message\">{{ initialMessage }}</p>\n <button mat-raised-button color=\"primary\" (click)=\"proceedToNextStep()\" [disabled]=\"isLoadingSubject | async\">\n {{ nextButtonLabel }}\n </button>\n </div>\n </div>\n\n <!-- Bank Selection State -->\n <div *ngIf=\"isBankSelectionState\" class=\"state-bank-selection\">\n <h3 class=\"section-title\">{{ selectBankTitle }}</h3>\n\n <!-- Bank Grid -->\n <div class=\"banks-grid\" *ngIf=\"banks.length > 0\">\n <div\n *ngFor=\"let bank of banks\"\n class=\"bank-card\"\n [class.selected]=\"selectedBank === bank.bankCode\"\n (click)=\"selectBank(bank)\">\n <div class=\"bank-logo\" *ngIf=\"bank.logoUrl\">\n <img [src]=\"bank.logoUrl\" [alt]=\"bank.displayName\">\n </div>\n <div class=\"bank-info\">\n <h4>{{ bank.displayName }}</h4>\n <span class=\"bank-code\">{{ bank.bankCode }}</span>\n </div>\n <mat-icon class=\"check-icon\" *ngIf=\"selectedBank === bank.bankCode\">check_circle</mat-icon>\n </div>\n </div>\n\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"proceedToNextStep()\"\n [disabled]=\"!selectedBank || (isLoadingSubject | async)\"\n class=\"m-t-16\">\n {{ continueButtonLabel }}\n </button>\n </div>\n\n <!-- Consent State -->\n <div *ngIf=\"isConsentState\" class=\"state-consent\">\n <div class=\"consent-header\">\n <h3>{{ consentTitle }}</h3>\n <p>{{ consentDescription }}</p>\n </div>\n\n <!-- Show iframe only if we have an authorization URL -->\n <div class=\"consent-container\" *ngIf=\"authorizationUrl\">\n <iframe\n [src]=\"authorizationUrl\"\n class=\"consent-iframe\"\n frameborder=\"0\"\n allowfullscreen\n sandbox=\"allow-forms allow-scripts allow-same-origin allow-popups allow-popups-to-escape-sandbox allow-top-navigation\">\n </iframe>\n </div>\n\n <!-- Show waiting message when consent is in popup or iframe failed -->\n <div class=\"consent-waiting\" *ngIf=\"!authorizationUrl\">\n <mat-spinner diameter=\"40\"></mat-spinner>\n <p>{{ waitingForAuthorizationText }}</p>\n <p class=\"consent-popup-hint\">\n <mat-icon>open_in_new</mat-icon>\n <span>{{ popupWindowMessage }}</span>\n </p>\n <p class=\"consent-popup-note\">\n {{ popupBlockerMessage }}\n </p>\n </div>\n </div>\n\n <!-- Processing State -->\n <div *ngIf=\"isProcessingState\" class=\"state-processing\">\n <div class=\"loading-container\">\n <mat-spinner diameter=\"50\"></mat-spinner>\n <p class=\"loading-message\">{{ waitingMessage }}</p>\n <div class=\"progress-details\">\n <p *ngIf=\"accounts.length > 0\">\n {{ accountsRetrievedLabel }}: {{ accounts.length }}\n </p>\n <p *ngIf=\"transactions.length > 0\">\n {{ transactionsRetrievedLabel }}: {{ transactions.length }}\n </p>\n </div>\n </div>\n </div>\n\n <!-- Error State -->\n <div *ngIf=\"isErrorState\" class=\"state-error\">\n <div class=\"error-container\">\n <mat-icon color=\"warn\" class=\"error-icon\">error_outline</mat-icon>\n <p class=\"error-message\">{{ errorMessage }}</p>\n <button mat-raised-button color=\"primary\" (click)=\"retry()\">\n {{ retryButtonLabel }}\n </button>\n </div>\n </div>\n\n <!-- Complete State -->\n <div *ngIf=\"isCompleteState\" class=\"state-complete\">\n <div class=\"success-container\">\n <mat-icon color=\"primary\" class=\"success-icon\">check_circle</mat-icon>\n <p class=\"success-message\">{{ completionMessage }}</p>\n\n <!-- Display summary of accounts found -->\n <div class=\"accounts-summary\" *ngIf=\"accounts.length > 0\">\n <h4>{{ accountsSummaryText }}</h4>\n <div class=\"accounts-list\">\n <div class=\"account-card\" *ngFor=\"let account of accounts\">\n <div class=\"account-header\">\n <mat-icon class=\"account-icon\">account_balance</mat-icon>\n <div class=\"account-info\">\n <div class=\"account-name\">{{ account.name || 'Account' }}</div>\n <div class=\"account-iban\">{{ account.iban }}</div>\n </div>\n </div>\n <div class=\"account-details\">\n <div class=\"account-balance\">\n <span class=\"balance-amount\">{{ account.balance | currency:account.currency:'symbol':'1.2-2' }}</span>\n </div>\n <div class=\"account-holder\">{{ account.accountHolderName }}</div>\n </div>\n </div>\n </div>\n\n <div class=\"transaction-summary\" *ngIf=\"transactions.length > 0\">\n <p>{{ transactionsSummaryText }}</p>\n </div>\n </div>\n </div>\n </div>\n</div>", styles: [".open-banking-container{padding:20px;min-height:400px}.open-banking-container .section-title{margin-bottom:20px;font-size:1.25rem;font-weight:500;color:#000000de}.open-banking-container .message-container,.open-banking-container .loading-container,.open-banking-container .error-container,.open-banking-container .success-container{display:flex;flex-direction:column;align-items:center;justify-content:center;text-align:center;min-height:200px}.open-banking-container .message,.open-banking-container .loading-message,.open-banking-container .error-message,.open-banking-container .success-message{font-size:1rem;margin:20px 0;max-width:600px;line-height:1.5}.open-banking-container .error-message{color:#f44336}.open-banking-container .success-message{color:#4caf50}.open-banking-container .error-icon,.open-banking-container .success-icon{font-size:48px;width:48px;height:48px}.open-banking-container .state-bank-selection .banks-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:16px;margin-bottom:24px}.open-banking-container .state-bank-selection .bank-card{border:2px solid #e0e0e0;border-radius:8px;padding:16px;cursor:pointer;transition:all .3s ease;position:relative;background:#fff;min-height:120px;display:flex;flex-direction:column;align-items:center;justify-content:center}.open-banking-container .state-bank-selection .bank-card:hover{border-color:#3f51b5;box-shadow:0 4px 8px #0000001a;transform:translateY(-2px)}.open-banking-container .state-bank-selection .bank-card.selected{border-color:#3f51b5;background:#f5f7ff}.open-banking-container .state-bank-selection .bank-card .bank-logo{width:80px;height:40px;margin-bottom:8px;display:flex;align-items:center;justify-content:center}.open-banking-container .state-bank-selection .bank-card .bank-logo img{max-width:100%;max-height:100%;object-fit:contain}.open-banking-container .state-bank-selection .bank-card .bank-info{text-align:center}.open-banking-container .state-bank-selection .bank-card .bank-info h4{margin:0;font-size:.9rem;font-weight:500;color:#333}.open-banking-container .state-bank-selection .bank-card .bank-info .bank-code{font-size:.75rem;color:#666;margin-top:4px;display:block}.open-banking-container .state-bank-selection .bank-card .check-icon{position:absolute;top:8px;right:8px;color:#3f51b5;font-size:20px}.open-banking-container .state-bank-selection .bank-dropdown{margin:20px 0}.open-banking-container .state-bank-selection .bank-selector{width:100%;max-width:400px}.open-banking-container .state-consent .consent-header{text-align:center;margin-bottom:20px}.open-banking-container .state-consent .consent-header h3{font-size:1.25rem;font-weight:500;margin-bottom:8px}.open-banking-container .state-consent .consent-header p{color:#666;font-size:.95rem}.open-banking-container .state-consent .consent-container{width:100%;height:600px;border:1px solid #e0e0e0;border-radius:4px;overflow:hidden;background:#fff}.open-banking-container .state-consent .consent-iframe{width:100%;height:100%}.open-banking-container .state-consent .consent-waiting{display:flex;flex-direction:column;align-items:center;justify-content:center;min-height:300px}.open-banking-container .state-consent .consent-waiting p{margin-top:16px;color:#666}.open-banking-container .state-consent .consent-waiting .consent-popup-hint{display:flex;align-items:center;gap:8px;margin-top:24px;font-size:1.1rem;color:#333}.open-banking-container .state-consent .consent-waiting .consent-popup-hint mat-icon{color:#3f51b5}.open-banking-container .state-consent .consent-waiting .consent-popup-note{margin-top:12px;font-size:.9rem;color:#999;font-style:italic}.open-banking-container .progress-details{margin-top:20px}.open-banking-container .progress-details p{font-size:.9rem;color:#666;margin:4px 0}.open-banking-container .accounts-summary{margin-top:30px;width:100%;max-width:800px}.open-banking-container .accounts-summary h4{font-size:1.15rem;font-weight:500;margin-bottom:20px;color:#000000de}.open-banking-container .accounts-summary .accounts-list{display:flex;flex-direction:column;gap:16px}.open-banking-container .accounts-summary .account-card{border:1px solid #e0e0e0;border-radius:8px;padding:16px;background:#fff;box-shadow:0 1px 3px #0000001a;transition:box-shadow .2s ease}.open-banking-container .accounts-summary .account-card:hover{box-shadow:0 2px 6px #00000026}.open-banking-container .accounts-summary .account-header{display:flex;align-items:flex-start;gap:12px;margin-bottom:12px}.open-banking-container .accounts-summary .account-icon{color:#3f51b5;margin-top:2px}.open-banking-container .accounts-summary .account-info{flex:1}.open-banking-container .accounts-summary .account-name{font-weight:500;font-size:1rem;color:#333;margin-bottom:4px}.open-banking-container .accounts-summary .account-iban{font-size:.85rem;color:#666;font-family:Courier New,monospace;letter-spacing:.5px}.open-banking-container .accounts-summary .account-details{display:flex;justify-content:space-between;align-items:center;padding-left:36px}.open-banking-container .accounts-summary .account-balance{display:flex;flex-direction:column}.open-banking-container .accounts-summary .balance-amount{font-size:1.25rem;font-weight:600;color:#4caf50}.open-banking-container .accounts-summary .account-holder{font-size:.9rem;color:#666;text-align:right}.open-banking-container .accounts-summary .transaction-summary{margin-top:16px;padding:12px;background:#e8f5e9;border-radius:4px}.open-banking-container .accounts-summary .transaction-summary p{margin:0;color:#2e7d32;font-size:.95rem}.open-banking-container ::ng-deep .mat-spinner{margin:0 auto}.open-banking-container ::ng-deep .mat-progress-spinner circle,.open-banking-container ::ng-deep .mat-spinner circle{stroke:#3f51b5}@media (max-width: 768px){.open-banking-container{padding:10px}.open-banking-container .state-bank-selection .banks-grid{grid-template-columns:repeat(auto-fill,minmax(150px,1fr));gap:12px}.open-banking-container .state-bank-selection .bank-card{min-height:100px;padding:12px}.open-banking-container .state-bank-selection .bank-card .bank-logo{width:60px;height:30px}.open-banking-container .consent-container{height:500px}.open-banking-container .accounts-summary{max-width:100%}}@media (max-width: 480px){.open-banking-container .state-bank-selection .banks-grid{grid-template-columns:1fr}}\n"] }]
589
+ }], ctorParameters: () => [{ type: i1.OpenBankingService }, { type: i2.DialogService }, { type: i3.DomSanitizer }, { type: i0.ChangeDetectorRef }] });
590
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formly-open-banking.component.js","sourceRoot":"","sources":["../../../../../../projects/origin-form/src/lib/formly/formly-open-banking/formly-open-banking.component.ts","../../../../../../projects/origin-form/src/lib/formly/formly-open-banking/formly-open-banking.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  AfterViewInit,\n  ChangeDetectorRef,\n  OnDestroy,\n} from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { BehaviorSubject, interval, Subscription } from 'rxjs';\nimport { switchMap, takeWhile, finalize } from 'rxjs/operators';\nimport { BaseFormlyStepComponent } from '../baseFormlyStepComponent';\n\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { DialogService } from '../../services/dialog.service';\nimport { Bank, Account, Transaction, TransactionPage, OpenBankingService, ConsentInitRequest, ConsentStatus } from '../../services/open-banking.service';\n\ninterface OpenBankingConfig {\n  accountNumberCollected?: string;\n  accountHolderCollected?: string;\n  emailCollected?: string;\n  showAllBanks?: boolean;\n  extractBankFromIBAN?: boolean;\n  tcAcceptedByDefault?: boolean;\n  // Message translations\n  initialMessageTranslations?: any[];\n  waitingMessageTranslations?: any[];\n  consentErrorMessageTranslations?: any[];\n  completionMessageTranslations?: any[];\n  retryButtonLabelTranslations?: any[];\n  // Button translations\n  nextButtonTranslations?: any[];\n  continueButtonTranslations?: any[];\n  // UI labels translations\n  selectBankTranslations?: any[];\n  consentTitleTranslations?: any[];\n  consentDescriptionTranslations?: any[];\n  waitingForAuthorizationTranslations?: any[];\n  popupWindowMessageTranslations?: any[];\n  popupBlockerMessageTranslations?: any[];\n  accountsRetrievedTranslations?: any[];\n  transactionsRetrievedTranslations?: any[];\n  accountsFoundTranslations?: any[];\n  transactionsSummaryTranslations?: any[];\n}\n\nenum OpenBankingState {\n  Initial = 'initial',\n  BankSelection = 'bankSelection',\n  Consent = 'consent',\n  Processing = 'processing',\n  Error = 'error',\n  Complete = 'complete',\n}\n\n@Component({\n  selector: 'app-formly-open-banking',\n  templateUrl: './formly-open-banking.component.html',\n  styleUrls: ['./formly-open-banking.component.scss'],\n})\nexport class FormlyOpenBankingComponent\n  extends BaseFormlyStepComponent\n  implements OnInit, AfterViewInit, OnDestroy\n{\n  config: OpenBankingConfig = {};\n  currentState = new BehaviorSubject<OpenBankingState>(OpenBankingState.Initial);\n  banks: Bank[] = [];\n  selectedBank: string = '';\n  selectedBankObject?: Bank;\n  authorizationUrl: SafeResourceUrl | null = null;\n  consentId: string = '';\n  private _errorMessage: string = '';\n  private _isConsentError: boolean = false;\n  private _consentErrorStatus: string = '';\n  override isLoading: boolean = false;\n  isLoadingSubject = new BehaviorSubject<boolean>(false);\n  accounts: Account[] = [];\n  transactions: Transaction[] = [];\n  pollSubscription?: Subscription;\n  selectedAccounts: string[] = [];\n  transactionPages: Map<string, TransactionPage> = new Map();\n\n  constructor(\n    private openBankingService: OpenBankingService,\n    private dialog: DialogService,\n    private sanitizer: DomSanitizer,\n    cdr: ChangeDetectorRef\n  ) {\n    super(cdr);\n  }\n\n  override async ngOnInit(): Promise<void> {\n    await super.ngOnInit();\n    if (this.stepData?.config) {\n      this.config = this.stepData.config as OpenBankingConfig;\n    }\n    // If translations are in stepData.translations, merge them into config\n    if (this.stepData?.translations) {\n      this.mergeTranslationsIntoConfig();\n    }\n  }\n\n  private mergeTranslationsIntoConfig(): void {\n    // Map translation names to config properties\n    const translationMap: { [key: string]: keyof OpenBankingConfig } = {\n      'initialMessageTranslations': 'initialMessageTranslations',\n      'waitingMessageTranslations': 'waitingMessageTranslations',\n      'consentErrorMessageTranslations': 'consentErrorMessageTranslations',\n      'completionMessageTranslations': 'completionMessageTranslations',\n      'retryButtonLabelTranslations': 'retryButtonLabelTranslations',\n      'nextButtonTranslations': 'nextButtonTranslations',\n      'continueButtonTranslations': 'continueButtonTranslations',\n      'selectBankTranslations': 'selectBankTranslations',\n      'consentTitleTranslations': 'consentTitleTranslations',\n      'consentDescriptionTranslations': 'consentDescriptionTranslations',\n      'waitingForAuthorizationTranslations': 'waitingForAuthorizationTranslations',\n      'popupWindowMessageTranslations': 'popupWindowMessageTranslations',\n      'popupBlockerMessageTranslations': 'popupBlockerMessageTranslations',\n      'accountsRetrievedTranslations': 'accountsRetrievedTranslations',\n      'transactionsRetrievedTranslations': 'transactionsRetrievedTranslations',\n      'accountsFoundTranslations': 'accountsFoundTranslations',\n      'transactionsSummaryTranslations': 'transactionsSummaryTranslations'\n    };\n\n    this.stepData.translations?.forEach((translation: { name: string | number; values: string | any[]; }) => {\n      const configKey = translationMap[translation.name];\n      if (configKey && translation.values && translation.values.length > 0) {\n        // Use the translation values from stepData if config doesn't have all languages\n        (this.config as any)[configKey] = translation.values;\n      }\n    });\n  }\n\n  ngAfterViewInit(): void {\n    // Only set initial state if we haven't already set a state\n    if (!this.currentState.value) {\n      this.currentState.next(OpenBankingState.Initial);\n    }\n  }\n\n  override onPageSelected() {\n    // Only reset to initial if we haven't started the flow yet\n    // This prevents reset when language changes\n    if (!this.currentState.value) {\n      this.currentState.next(OpenBankingState.Initial);\n    }\n    // Update language if it changed\n    this.cdr.detectChanges();\n  }\n\n  private determineNextStep(): void {\n    // Get account information from form if configured\n    const accountNumber = this.getFieldValue(this.config.accountNumberCollected);\n\n    // If we should extract bank from IBAN and have an IBAN\n    if (this.config.extractBankFromIBAN && accountNumber && this.isIBAN(accountNumber)) {\n      // Load banks first, then match IBAN to bank\n      this.loadBanksAndMatchIBAN(accountNumber);\n      return;\n    }\n\n    // If we should show all banks\n    if (this.config.showAllBanks) {\n      this.loadBanks();\n    } else {\n      // Go directly to consent without bank selection\n      this.initiateConsent();\n    }\n  }\n\n  private getFieldValue(fieldId: string | undefined): string {\n    if (!fieldId) return '';\n    return this.form.get(fieldId)?.value || '';\n  }\n\n  private isIBAN(value: string): boolean {\n    // Basic IBAN validation for Romanian IBANs\n    return /^RO\\d{2}[A-Z]{4}\\d{16}$/.test(value.replace(/\\s/g, ''));\n  }\n\n  private extractBankFromIBAN(iban: string): string {\n    // Extract bank code from Romanian IBAN (positions 5-8)\n    const cleanIban = iban.replace(/\\s/g, '');\n    if (cleanIban.startsWith('RO') && cleanIban.length >= 8) {\n      return cleanIban.substring(4, 8);\n    }\n    return '';\n  }\n\n  private loadBanksAndMatchIBAN(iban: string): void {\n    this.setLoadingState(true);\n    const countryCode = 'RO'; // Default to Romania\n\n    this.openBankingService.getBanks(countryCode).subscribe({\n      next: (banks) => {\n        this.banks = banks.filter(b => b.isActive);\n\n        // Extract bank code from IBAN\n        const ibanBankCode = this.extractBankFromIBAN(iban);\n\n        if (ibanBankCode) {\n          // Try to match against bank codes\n          const matchedBank = this.banks.find(bank =>\n            bank.bankCode.toUpperCase() === ibanBankCode.toUpperCase()\n          );\n\n          if (matchedBank) {\n            // Bank found - select it and go directly to consent\n            this.selectedBank = matchedBank.bankCode;\n            this.selectedBankObject = matchedBank;\n            this.setLoadingState(false);\n            this.initiateConsent();\n            return;\n          }\n        }\n\n        // If no match found, show bank selection\n        this.currentState.next(OpenBankingState.BankSelection);\n        this.setLoadingState(false);\n      },\n      error: (error) => {\n        this.handleError(error);\n        this.setLoadingState(false);\n      },\n    });\n  }\n\n  private loadBanks(): void {\n    this.setLoadingState(true);\n    const countryCode = 'RO'; // Default to Romania\n\n    this.openBankingService.getBanks(countryCode).subscribe({\n      next: (banks) => {\n        this.banks = banks.filter(b => b.isActive);\n        this.currentState.next(OpenBankingState.BankSelection);\n        this.setLoadingState(false);\n      },\n      error: (error) => {\n        this.handleError(error);\n        this.setLoadingState(false);\n      },\n    });\n  }\n\n  selectBank(bank: Bank): void {\n    this.selectedBank = bank.bankCode;\n    this.selectedBankObject = bank;\n  }\n\n  proceedToNextStep(): void {\n    if (this.currentState.value === OpenBankingState.Initial) {\n      // User clicked Next on initial screen - determine what to show next\n      this.determineNextStep();\n    } else if (this.currentState.value === OpenBankingState.BankSelection) {\n      if (this.selectedBank) {\n        this.initiateConsent();\n      }\n    }\n  }\n\n  private initiateConsent(): void {\n    this.setLoadingState(true);\n\n    // Get user information\n    const email = this.getFieldValue(this.config.emailCollected) || 'user@example.com';\n    // Generate a unique PSU ID for this session\n    const psuId = this.appDataId || 'user-' + Date.now() + '-' + Math.random().toString(36).substr(2, 9);\n\n    const consentRequest: ConsentInitRequest = {\n      bankCode: this.selectedBank || 'DEFAULT',\n      psuEmail: email,\n      psuIntermediaryId: psuId,\n      periodOfValidity: 90,\n      tcAccepted: this.config.tcAcceptedByDefault !== false\n    };\n\n    this.openBankingService.initializeConsent(consentRequest).subscribe({\n      next: (response) => {\n        this.consentId = response.consentId;\n\n        // Check if bank requires redirect\n        if (this.selectedBankObject?.requiresRedirect !== false) {\n          // Try iframe first, but most banks will block it\n          if (this.shouldUseIframe()) {\n            this.authorizationUrl = this.sanitizer.bypassSecurityTrustResourceUrl(response.authorizationUrl);\n            this.currentState.next(OpenBankingState.Consent);\n            // Set a flag to detect if iframe fails\n            this.checkIframeLoad(response.authorizationUrl);\n          } else {\n            // Open in new window/tab\n            this.openConsentInNewWindow(response.authorizationUrl);\n          }\n        }\n\n        this.setLoadingState(false);\n\n        // Start polling for consent status\n        this.startConsentPolling();\n      },\n      error: (error) => {\n        this.handleError(error);\n        this.setLoadingState(false);\n      },\n    });\n  }\n\n  private shouldUseIframe(): boolean {\n    // Most banks don't allow iframes due to X-Frame-Options\n    // So default to false (use popup window instead)\n    return false;\n  }\n\n  private openConsentInNewWindow(authorizationUrl: string): void {\n    const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);\n\n    let consentWindow: Window | null;\n\n    if (isMobile) {\n      // On mobile, open in same tab (full redirect) or new tab\n      // Using _blank will open a new tab and allow user to return\n      consentWindow = window.open(authorizationUrl, '_blank');\n    } else {\n      // On desktop, open in a popup window\n      const width = 800;\n      const height = 600;\n      const left = (window.screen.width - width) / 2;\n      const top = (window.screen.height - height) / 2;\n\n      consentWindow = window.open(\n        authorizationUrl,\n        'bankConsent',\n        `width=${width},height=${height},left=${left},top=${top},toolbar=no,menubar=no,scrollbars=yes,resizable=yes`\n      );\n    }\n\n    if (consentWindow) {\n      // Show a message that consent is in progress\n      this.currentState.next(OpenBankingState.Consent);\n\n      // Check if window is closed periodically (works better on desktop)\n      if (!isMobile) {\n        const checkWindowClosed = setInterval(() => {\n          if (consentWindow && consentWindow.closed) {\n            clearInterval(checkWindowClosed);\n            // Window was closed, the polling will detect the consent status\n          }\n        }, 1000);\n      }\n      // On mobile, we rely entirely on polling since window.closed doesn't work reliably\n    } else {\n      // Popup was blocked\n      this._isConsentError = false;\n      this._errorMessage = 'Please allow popups for this site to complete bank authorization. Then click Retry to try again.';\n      this.currentState.next(OpenBankingState.Error);\n    }\n  }\n\n  private checkIframeLoad(authorizationUrl: string): void {\n    // If iframe fails to load (X-Frame-Options), fall back to popup\n    setTimeout(() => {\n      // Check if we're still in consent state (iframe might have failed)\n      if (this.currentState.value === OpenBankingState.Consent && this.authorizationUrl) {\n        // Clear iframe and open in new window\n        console.warn('Iframe might be blocked, opening in new window');\n        this.authorizationUrl = null;\n        this.openConsentInNewWindow(authorizationUrl);\n      }\n    }, 2000);\n  }\n\n  private startConsentPolling(): void {\n    // Poll every 5 seconds for consent status (user needs time to complete bank authorization)\n    this.pollSubscription = interval(10000)\n      .pipe(\n        switchMap(() => this.openBankingService.getConsentStatus(this.selectedBank || 'DEFAULT', this.consentId)),\n        takeWhile((status) => status.status === 'received', true),\n        finalize(() => this.pollSubscription?.unsubscribe())\n      )\n      .subscribe({\n        next: (status: ConsentStatus) => {\n          if (status.status === 'valid') {\n            this.onConsentApproved();\n          } else if (status.status === 'rejected' ||\n                     status.status === 'expired' ||\n                     status.status === 'terminated' ||\n                     status.status === 'revokedByPsu') {\n            this.onConsentRejected(status);\n          }\n        },\n        error: (error) => {\n          this.handleError(error);\n        },\n      });\n  }\n\n  private onConsentApproved(): void {\n    this.currentState.next(OpenBankingState.Processing);\n    this.fetchAccountData();\n  }\n\n  private onConsentRejected(status: ConsentStatus): void {\n    this._isConsentError = true;\n    this._consentErrorStatus = status.status;\n    this._errorMessage = ''; // Clear to use translated message\n    this.currentState.next(OpenBankingState.Error);\n  }\n\n  private fetchAccountData(): void {\n    this.setLoadingState(true);\n\n    // First, fetch accounts\n    this.openBankingService.getAccounts(this.consentId, this.selectedBank || 'DEFAULT').subscribe({\n      next: (accounts) => {\n        this.accounts = accounts;\n\n        // Select all accounts for transaction fetching\n        this.selectedAccounts = accounts.map(a => a.accountId);\n\n        // Fetch transactions for each account\n        this.fetchTransactionsForAccounts();\n      },\n      error: (error) => {\n        this.handleError(error);\n        this.setLoadingState(false);\n      },\n    });\n  }\n\n  private fetchTransactionsForAccounts(): void {\n    if (this.selectedAccounts.length === 0) {\n      this.completeProcess();\n      return;\n    }\n\n    const transactionRequests = this.selectedAccounts.map(accountId => {\n      // Calculate date range based on bank-specific limits\n      const bankCode = this.selectedBank || 'DEFAULT';\n      const maxDays = this.openBankingService.getMaxTransactionPeriod(bankCode);\n\n      const toDate = new Date();\n      const fromDate = new Date();\n\n      // If unlimited (-1) or very long period, use 365 days as practical limit\n      // Otherwise use the bank's specific limit\n      const daysToFetch = maxDays === -1 ? 365 : Math.min(maxDays, 365);\n      fromDate.setDate(fromDate.getDate() - daysToFetch);\n\n      return this.openBankingService.getTransactions({\n        consentId: this.consentId,\n        bankCode: bankCode,\n        accountId: accountId,\n        fromDate: fromDate.toISOString(),\n        toDate: toDate.toISOString(),\n        page: 1,\n        pageSize: 50\n      });\n    });\n\n    // Execute all transaction requests\n    Promise.all(transactionRequests.map(req => req.toPromise()))\n      .then(results => {\n        // Store transaction pages\n        results.forEach((page, index) => {\n          if (page) {\n            this.transactionPages.set(this.selectedAccounts[index], page);\n            // Flatten all transactions into single array\n            this.transactions.push(...page.items);\n          }\n        });\n\n        this.completeProcess();\n      })\n      .catch(error => {\n        console.error('Error fetching transactions:', error);\n        // Even if transactions fail, we can continue with accounts\n        this.completeProcess();\n      });\n  }\n\n  private completeProcess(): void {\n    // Store data in form model\n    this.storeAccountData();\n\n    this.currentState.next(OpenBankingState.Complete);\n    this.setLoadingState(false);\n\n    // Auto-proceed to next step after a short delay\n    setTimeout(() => {\n      this.onEvent('success');\n    }, 2000);\n  }\n\n  private storeAccountData(): void {\n    // Store the fetched data in the form model for use in later steps\n    const group = this.form as FormGroup;\n\n    // Add accounts and transactions to form\n    group.addControl('openBankingAccounts', new FormControl(this.accounts));\n    group.addControl('openBankingTransactions', new FormControl(this.transactions));\n    group.addControl('openBankingConsentId', new FormControl(this.consentId));\n    group.addControl('openBankingBankCode', new FormControl(this.selectedBank));\n\n    // Store account summary\n    const accountSummary = {\n      totalAccounts: this.accounts.length,\n      totalBalance: this.accounts.reduce((sum, acc) => sum + acc.balance, 0),\n      currencies: [...new Set(this.accounts.map(a => a.currency))],\n      transactionCount: this.transactions.length,\n      oldestTransaction: this.transactions.length > 0\n        ? this.transactions.reduce((oldest, t) =>\n            new Date(t.bookingDate || t.transactionDate || '') < new Date(oldest.bookingDate || oldest.transactionDate || '') ? t : oldest\n          ).bookingDate\n        : null,\n      newestTransaction: this.transactions.length > 0\n        ? this.transactions.reduce((newest, t) =>\n            new Date(t.bookingDate || t.transactionDate || '') > new Date(newest.bookingDate || newest.transactionDate || '') ? t : newest\n          ).bookingDate\n        : null\n    };\n\n    group.addControl('openBankingAccountSummary', new FormControl(accountSummary));\n\n    // Populate the step's controls for SaveAppData\n    this.populateStepControls();\n  }\n\n  private populateStepControls(): void {\n    // Find and populate the controls in the step sections\n    if (!this.stepData?.sections) return;\n\n    // Get email from config mapping or use empty string\n    const email = this.getFieldValue(this.config.emailCollected) || '';\n\n    // Get the first account as the primary account\n    const primaryAccount = this.accounts.length > 0 ? this.accounts[0] : null;\n\n    // Calculate total balance\n    const totalBalance = this.accounts.reduce((sum, acc) => sum + acc.balance, 0);\n\n    // Calculate transaction period in months\n    const months = this.getTransactionPeriodMonths();\n\n    // Map of control identifiers to values\n    const controlValues: { [key: string]: any } = {\n      'OpenBankingConsentStatus': true, // Consent was successfully validated\n      'OpenBankingEmail': email,\n      'OpenBankingAccountNumber': primaryAccount?.iban || '',\n      'OpenBankingAccountHolder': primaryAccount?.accountHolderName || '',\n      'OpenBankingBankCode': this.selectedBank || '',\n      'OpenBankingConsentId': this.consentId,\n      'OpenBankingAccountsCount': this.accounts.length,\n      'OpenBankingAccountsData': JSON.stringify(this.accounts),\n      'OpenBankingTotalBalance': totalBalance,\n      'OpenBankingTransactionsCount': this.transactions.length,\n      'OpenBankingTransactionsData': JSON.stringify(this.transactions),\n      'OpenBankingTransactionsPeriod': months\n    };\n\n    // Add all control values to the form\n    const group = this.form as FormGroup;\n    Object.keys(controlValues).forEach(controlId => {\n      if (!group.contains(controlId)) {\n        group.addControl(controlId, new FormControl(controlValues[controlId]));\n      } else {\n        group.get(controlId)?.setValue(controlValues[controlId]);\n      }\n    });\n\n    // The values are now stored in the form model and will be saved with SaveAppData\n    // When the form is submitted, these values will be associated with the step's controls\n  }\n\n  retry(): void {\n    this._errorMessage = '';\n    this._isConsentError = false;\n    this._consentErrorStatus = '';\n    this.accounts = [];\n    this.transactions = [];\n    this.selectedAccounts = [];\n    this.transactionPages.clear();\n    this.consentId = '';\n    this.selectedBank = '';\n    this.selectedBankObject = undefined;\n    this.banks = [];\n    this.authorizationUrl = null;\n\n    // Reset to initial state - user needs to click Next again\n    this.currentState.next(OpenBankingState.Initial);\n  }\n\n  private handleError(error: HttpErrorResponse): void {\n    console.error('OpenBanking error:', error);\n    this._isConsentError = false;\n    this._consentErrorStatus = '';\n    this._errorMessage = error.error?.message || error.message || 'An error occurred. Please try again.';\n    this.currentState.next(OpenBankingState.Error);\n  }\n\n  private getTranslatedMessage(translations: any[] | undefined, defaultMessage: string = ''): string {\n    if (!translations || translations.length === 0) return defaultMessage;\n\n    const langIso = this.props?.['langIso'] || this.langIso || 'en';\n    const translation = translations.find(t => t.languageIso === langIso);\n    return translation?.value || translations[0]?.value || defaultMessage;\n  }\n\n  get initialMessage(): string {\n    return this.getTranslatedMessage(\n      this.config.initialMessageTranslations,\n      'To continue, we need to verify your bank account information. Click Next to proceed.'\n    );\n  }\n\n  get waitingMessage(): string {\n    return this.getTranslatedMessage(\n      this.config.waitingMessageTranslations,\n      'Please wait while we retrieve your account information...'\n    );\n  }\n\n  get completionMessage(): string {\n    return this.getTranslatedMessage(\n      this.config.completionMessageTranslations,\n      'Account information successfully retrieved.'\n    );\n  }\n\n  get retryButtonLabel(): string {\n    return this.getTranslatedMessage(\n      this.config.retryButtonLabelTranslations,\n      'Retry'\n    );\n  }\n\n  get errorMessage(): string {\n    // If it's a consent error, use the translated consent error message\n    if (this._isConsentError) {\n      const message = this.getTranslatedMessage(\n        this.config.consentErrorMessageTranslations,\n        'You must provide consent to continue. Please try again.'\n      );\n      return this._consentErrorStatus ? `${message} (Status: ${this._consentErrorStatus})` : message;\n    }\n    // Otherwise return the stored error message (from API or default)\n    return this._errorMessage || 'An error occurred. Please try again.';\n  }\n\n  // Button translations\n  get nextButtonLabel(): string {\n    return this.getTranslatedMessage(\n      this.config.nextButtonTranslations,\n      'Next'\n    );\n  }\n\n  get continueButtonLabel(): string {\n    return this.getTranslatedMessage(\n      this.config.continueButtonTranslations,\n      'Continue'\n    );\n  }\n\n  // UI label translations\n  get selectBankTitle(): string {\n    return this.getTranslatedMessage(\n      this.config.selectBankTranslations,\n      'Select your bank'\n    );\n  }\n\n  get consentTitle(): string {\n    return this.getTranslatedMessage(\n      this.config.consentTitleTranslations,\n      'Bank Authorization'\n    );\n  }\n\n  get consentDescription(): string {\n    return this.getTranslatedMessage(\n      this.config.consentDescriptionTranslations,\n      'Please complete the authorization process with your bank'\n    );\n  }\n\n  get waitingForAuthorizationText(): string {\n    return this.getTranslatedMessage(\n      this.config.waitingForAuthorizationTranslations,\n      'Waiting for authorization...'\n    );\n  }\n\n  get popupWindowMessage(): string {\n    return this.getTranslatedMessage(\n      this.config.popupWindowMessageTranslations,\n      'A new window has opened for bank authorization. Please complete the process there.'\n    );\n  }\n\n  get popupBlockerMessage(): string {\n    return this.getTranslatedMessage(\n      this.config.popupBlockerMessageTranslations,\n      'If the window didn\\'t open, please check your popup blocker settings.'\n    );\n  }\n\n  get accountsRetrievedLabel(): string {\n    return this.getTranslatedMessage(\n      this.config.accountsRetrievedTranslations,\n      'Accounts retrieved'\n    );\n  }\n\n  get transactionsRetrievedLabel(): string {\n    return this.getTranslatedMessage(\n      this.config.transactionsRetrievedTranslations,\n      'Transactions retrieved'\n    );\n  }\n\n  get accountsSummaryText(): string {\n    const count = this.accounts.length;\n    const template = this.getTranslatedMessage(\n      this.config.accountsFoundTranslations,\n      '{0} account(s) found'\n    );\n    return template.replace('{0}', count.toString());\n  }\n\n  get transactionsSummaryText(): string {\n    const count = this.transactions.length;\n    const months = this.getTransactionPeriodMonths();\n    const template = this.getTranslatedMessage(\n      this.config.transactionsSummaryTranslations,\n      '{0} transaction(s) retrieved for the last {1} month(s)'\n    );\n    return template.replace('{0}', count.toString()).replace('{1}', months.toString());\n  }\n\n  private getTransactionPeriodMonths(): number {\n    if (!this.selectedBank) return 3;\n\n    const maxDays = this.openBankingService.getMaxTransactionPeriod(this.selectedBank);\n    const daysToFetch = maxDays === -1 ? 365 : Math.min(maxDays, 365);\n    return Math.round(daysToFetch / 30);\n  }\n\n  get isInitialState(): boolean {\n    return this.currentState.value === OpenBankingState.Initial;\n  }\n\n  get isBankSelectionState(): boolean {\n    return this.currentState.value === OpenBankingState.BankSelection;\n  }\n\n  get isConsentState(): boolean {\n    return this.currentState.value === OpenBankingState.Consent;\n  }\n\n  get isProcessingState(): boolean {\n    return this.currentState.value === OpenBankingState.Processing;\n  }\n\n  get isErrorState(): boolean {\n    return this.currentState.value === OpenBankingState.Error;\n  }\n\n  get isCompleteState(): boolean {\n    return this.currentState.value === OpenBankingState.Complete;\n  }\n\n  private setLoadingState(loading: boolean): void {\n    this.isLoading = loading;\n    this.isLoadingSubject.next(loading);\n    this.cdr.detectChanges();\n  }\n\n  ngOnDestroy(): void {\n    this.pollSubscription?.unsubscribe();\n    this.isLoadingSubject.complete();\n  }\n}","<div class=\"open-banking-container\">\n  <!-- Initial State -->\n  <div *ngIf=\"isInitialState\" class=\"state-initial\">\n    <div class=\"message-container\">\n      <p class=\"message\">{{ initialMessage }}</p>\n      <button mat-raised-button color=\"primary\" (click)=\"proceedToNextStep()\" [disabled]=\"isLoadingSubject | async\">\n        {{ nextButtonLabel }}\n      </button>\n    </div>\n  </div>\n\n  <!-- Bank Selection State -->\n  <div *ngIf=\"isBankSelectionState\" class=\"state-bank-selection\">\n    <h3 class=\"section-title\">{{ selectBankTitle }}</h3>\n\n    <!-- Bank Grid -->\n    <div class=\"banks-grid\" *ngIf=\"banks.length > 0\">\n      <div\n        *ngFor=\"let bank of banks\"\n        class=\"bank-card\"\n        [class.selected]=\"selectedBank === bank.bankCode\"\n        (click)=\"selectBank(bank)\">\n        <div class=\"bank-logo\" *ngIf=\"bank.logoUrl\">\n          <img [src]=\"bank.logoUrl\" [alt]=\"bank.displayName\">\n        </div>\n        <div class=\"bank-info\">\n          <h4>{{ bank.displayName }}</h4>\n          <span class=\"bank-code\">{{ bank.bankCode }}</span>\n        </div>\n        <mat-icon class=\"check-icon\" *ngIf=\"selectedBank === bank.bankCode\">check_circle</mat-icon>\n      </div>\n    </div>\n\n    <button\n      mat-raised-button\n      color=\"primary\"\n      (click)=\"proceedToNextStep()\"\n      [disabled]=\"!selectedBank || (isLoadingSubject | async)\"\n      class=\"m-t-16\">\n      {{ continueButtonLabel }}\n    </button>\n  </div>\n\n  <!-- Consent State -->\n  <div *ngIf=\"isConsentState\" class=\"state-consent\">\n    <div class=\"consent-header\">\n      <h3>{{ consentTitle }}</h3>\n      <p>{{ consentDescription }}</p>\n    </div>\n\n    <!-- Show iframe only if we have an authorization URL -->\n    <div class=\"consent-container\" *ngIf=\"authorizationUrl\">\n      <iframe\n        [src]=\"authorizationUrl\"\n        class=\"consent-iframe\"\n        frameborder=\"0\"\n        allowfullscreen\n        sandbox=\"allow-forms allow-scripts allow-same-origin allow-popups allow-popups-to-escape-sandbox allow-top-navigation\">\n      </iframe>\n    </div>\n\n    <!-- Show waiting message when consent is in popup or iframe failed -->\n    <div class=\"consent-waiting\" *ngIf=\"!authorizationUrl\">\n      <mat-spinner diameter=\"40\"></mat-spinner>\n      <p>{{ waitingForAuthorizationText }}</p>\n      <p class=\"consent-popup-hint\">\n        <mat-icon>open_in_new</mat-icon>\n        <span>{{ popupWindowMessage }}</span>\n      </p>\n      <p class=\"consent-popup-note\">\n        {{ popupBlockerMessage }}\n      </p>\n    </div>\n  </div>\n\n  <!-- Processing State -->\n  <div *ngIf=\"isProcessingState\" class=\"state-processing\">\n    <div class=\"loading-container\">\n      <mat-spinner diameter=\"50\"></mat-spinner>\n      <p class=\"loading-message\">{{ waitingMessage }}</p>\n      <div class=\"progress-details\">\n        <p *ngIf=\"accounts.length > 0\">\n          {{ accountsRetrievedLabel }}: {{ accounts.length }}\n        </p>\n        <p *ngIf=\"transactions.length > 0\">\n          {{ transactionsRetrievedLabel }}: {{ transactions.length }}\n        </p>\n      </div>\n    </div>\n  </div>\n\n  <!-- Error State -->\n  <div *ngIf=\"isErrorState\" class=\"state-error\">\n    <div class=\"error-container\">\n      <mat-icon color=\"warn\" class=\"error-icon\">error_outline</mat-icon>\n      <p class=\"error-message\">{{ errorMessage }}</p>\n      <button mat-raised-button color=\"primary\" (click)=\"retry()\">\n        {{ retryButtonLabel }}\n      </button>\n    </div>\n  </div>\n\n  <!-- Complete State -->\n  <div *ngIf=\"isCompleteState\" class=\"state-complete\">\n    <div class=\"success-container\">\n      <mat-icon color=\"primary\" class=\"success-icon\">check_circle</mat-icon>\n      <p class=\"success-message\">{{ completionMessage }}</p>\n\n      <!-- Display summary of accounts found -->\n      <div class=\"accounts-summary\" *ngIf=\"accounts.length > 0\">\n        <h4>{{ accountsSummaryText }}</h4>\n        <div class=\"accounts-list\">\n          <div class=\"account-card\" *ngFor=\"let account of accounts\">\n            <div class=\"account-header\">\n              <mat-icon class=\"account-icon\">account_balance</mat-icon>\n              <div class=\"account-info\">\n                <div class=\"account-name\">{{ account.name || 'Account' }}</div>\n                <div class=\"account-iban\">{{ account.iban }}</div>\n              </div>\n            </div>\n            <div class=\"account-details\">\n              <div class=\"account-balance\">\n                <span class=\"balance-amount\">{{ account.balance | currency:account.currency:'symbol':'1.2-2' }}</span>\n              </div>\n              <div class=\"account-holder\">{{ account.accountHolderName }}</div>\n            </div>\n          </div>\n        </div>\n\n        <div class=\"transaction-summary\" *ngIf=\"transactions.length > 0\">\n          <p>{{ transactionsSummaryText }}</p>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>"]}