valtech-components 2.0.807 → 2.0.809

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.
@@ -0,0 +1,42 @@
1
+ /**
2
+ * UpdateBanner Types
3
+ *
4
+ * Contenido i18n del banner de actualización de la app.
5
+ */
6
+ /**
7
+ * Namespace i18n usado por `val-update-banner`.
8
+ * Las apps consumidoras pueden sobreescribir estas keys registrando
9
+ * contenido bajo este namespace en `I18nService`.
10
+ */
11
+ export const UPDATE_BANNER_I18N_NAMESPACE = 'UpdateBanner';
12
+ /**
13
+ * Textos por defecto del banner (fallback si la app no registra el namespace).
14
+ * Soporta es / en / pt.
15
+ */
16
+ export const UPDATE_BANNER_DEFAULT_CONTENT = {
17
+ es: {
18
+ availableTitle: 'Hay una versión nueva disponible',
19
+ availableMessage: 'Actualiza para obtener las últimas mejoras.',
20
+ requiredTitle: 'Debes actualizar para continuar',
21
+ requiredMessage: 'Esta versión ya no es compatible. Actualiza para seguir.',
22
+ updateAction: 'Actualizar',
23
+ dismissAction: 'Cerrar',
24
+ },
25
+ en: {
26
+ availableTitle: 'A new version is available',
27
+ availableMessage: 'Update to get the latest improvements.',
28
+ requiredTitle: 'You must update to continue',
29
+ requiredMessage: 'This version is no longer supported. Please update.',
30
+ updateAction: 'Update',
31
+ dismissAction: 'Close',
32
+ },
33
+ pt: {
34
+ availableTitle: 'Há uma nova versão disponível',
35
+ availableMessage: 'Atualize para obter as últimas melhorias.',
36
+ requiredTitle: 'Você precisa atualizar para continuar',
37
+ requiredMessage: 'Esta versão não é mais compatível. Atualize para seguir.',
38
+ updateAction: 'Atualizar',
39
+ dismissAction: 'Fechar',
40
+ },
41
+ };
42
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3VwZGF0ZS1iYW5uZXIvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7R0FJRztBQUVIOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyxjQUFjLENBQUM7QUFFM0Q7OztHQUdHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sNkJBQTZCLEdBQUc7SUFDM0MsRUFBRSxFQUFFO1FBQ0YsY0FBYyxFQUFFLGtDQUFrQztRQUNsRCxnQkFBZ0IsRUFBRSw2Q0FBNkM7UUFDL0QsYUFBYSxFQUFFLGlDQUFpQztRQUNoRCxlQUFlLEVBQUUsMERBQTBEO1FBQzNFLFlBQVksRUFBRSxZQUFZO1FBQzFCLGFBQWEsRUFBRSxRQUFRO0tBQ3hCO0lBQ0QsRUFBRSxFQUFFO1FBQ0YsY0FBYyxFQUFFLDRCQUE0QjtRQUM1QyxnQkFBZ0IsRUFBRSx3Q0FBd0M7UUFDMUQsYUFBYSxFQUFFLDZCQUE2QjtRQUM1QyxlQUFlLEVBQUUscURBQXFEO1FBQ3RFLFlBQVksRUFBRSxRQUFRO1FBQ3RCLGFBQWEsRUFBRSxPQUFPO0tBQ3ZCO0lBQ0QsRUFBRSxFQUFFO1FBQ0YsY0FBYyxFQUFFLCtCQUErQjtRQUMvQyxnQkFBZ0IsRUFBRSwyQ0FBMkM7UUFDN0QsYUFBYSxFQUFFLHVDQUF1QztRQUN0RCxlQUFlLEVBQUUsMERBQTBEO1FBQzNFLFlBQVksRUFBRSxXQUFXO1FBQ3pCLGFBQWEsRUFBRSxRQUFRO0tBQ3hCO0NBQ08sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVXBkYXRlQmFubmVyIFR5cGVzXG4gKlxuICogQ29udGVuaWRvIGkxOG4gZGVsIGJhbm5lciBkZSBhY3R1YWxpemFjacOzbiBkZSBsYSBhcHAuXG4gKi9cblxuLyoqXG4gKiBOYW1lc3BhY2UgaTE4biB1c2FkbyBwb3IgYHZhbC11cGRhdGUtYmFubmVyYC5cbiAqIExhcyBhcHBzIGNvbnN1bWlkb3JhcyBwdWVkZW4gc29icmVlc2NyaWJpciBlc3RhcyBrZXlzIHJlZ2lzdHJhbmRvXG4gKiBjb250ZW5pZG8gYmFqbyBlc3RlIG5hbWVzcGFjZSBlbiBgSTE4blNlcnZpY2VgLlxuICovXG5leHBvcnQgY29uc3QgVVBEQVRFX0JBTk5FUl9JMThOX05BTUVTUEFDRSA9ICdVcGRhdGVCYW5uZXInO1xuXG4vKipcbiAqIFRleHRvcyBwb3IgZGVmZWN0byBkZWwgYmFubmVyIChmYWxsYmFjayBzaSBsYSBhcHAgbm8gcmVnaXN0cmEgZWwgbmFtZXNwYWNlKS5cbiAqIFNvcG9ydGEgZXMgLyBlbiAvIHB0LlxuICovXG5leHBvcnQgY29uc3QgVVBEQVRFX0JBTk5FUl9ERUZBVUxUX0NPTlRFTlQgPSB7XG4gIGVzOiB7XG4gICAgYXZhaWxhYmxlVGl0bGU6ICdIYXkgdW5hIHZlcnNpw7NuIG51ZXZhIGRpc3BvbmlibGUnLFxuICAgIGF2YWlsYWJsZU1lc3NhZ2U6ICdBY3R1YWxpemEgcGFyYSBvYnRlbmVyIGxhcyDDumx0aW1hcyBtZWpvcmFzLicsXG4gICAgcmVxdWlyZWRUaXRsZTogJ0RlYmVzIGFjdHVhbGl6YXIgcGFyYSBjb250aW51YXInLFxuICAgIHJlcXVpcmVkTWVzc2FnZTogJ0VzdGEgdmVyc2nDs24geWEgbm8gZXMgY29tcGF0aWJsZS4gQWN0dWFsaXphIHBhcmEgc2VndWlyLicsXG4gICAgdXBkYXRlQWN0aW9uOiAnQWN0dWFsaXphcicsXG4gICAgZGlzbWlzc0FjdGlvbjogJ0NlcnJhcicsXG4gIH0sXG4gIGVuOiB7XG4gICAgYXZhaWxhYmxlVGl0bGU6ICdBIG5ldyB2ZXJzaW9uIGlzIGF2YWlsYWJsZScsXG4gICAgYXZhaWxhYmxlTWVzc2FnZTogJ1VwZGF0ZSB0byBnZXQgdGhlIGxhdGVzdCBpbXByb3ZlbWVudHMuJyxcbiAgICByZXF1aXJlZFRpdGxlOiAnWW91IG11c3QgdXBkYXRlIHRvIGNvbnRpbnVlJyxcbiAgICByZXF1aXJlZE1lc3NhZ2U6ICdUaGlzIHZlcnNpb24gaXMgbm8gbG9uZ2VyIHN1cHBvcnRlZC4gUGxlYXNlIHVwZGF0ZS4nLFxuICAgIHVwZGF0ZUFjdGlvbjogJ1VwZGF0ZScsXG4gICAgZGlzbWlzc0FjdGlvbjogJ0Nsb3NlJyxcbiAgfSxcbiAgcHQ6IHtcbiAgICBhdmFpbGFibGVUaXRsZTogJ0jDoSB1bWEgbm92YSB2ZXJzw6NvIGRpc3BvbsOtdmVsJyxcbiAgICBhdmFpbGFibGVNZXNzYWdlOiAnQXR1YWxpemUgcGFyYSBvYnRlciBhcyDDumx0aW1hcyBtZWxob3JpYXMuJyxcbiAgICByZXF1aXJlZFRpdGxlOiAnVm9jw6ogcHJlY2lzYSBhdHVhbGl6YXIgcGFyYSBjb250aW51YXInLFxuICAgIHJlcXVpcmVkTWVzc2FnZTogJ0VzdGEgdmVyc8OjbyBuw6NvIMOpIG1haXMgY29tcGF0w612ZWwuIEF0dWFsaXplIHBhcmEgc2VndWlyLicsXG4gICAgdXBkYXRlQWN0aW9uOiAnQXR1YWxpemFyJyxcbiAgICBkaXNtaXNzQWN0aW9uOiAnRmVjaGFyJyxcbiAgfSxcbn0gYXMgY29uc3Q7XG4iXX0=
@@ -1,112 +1,243 @@
1
1
  /**
2
2
  * UpdateBannerComponent
3
3
  *
4
- * Componente que muestra un banner cuando hay una nueva versión de la aplicación disponible.
5
- * Se integra automáticamente con AppConfigService para detectar actualizaciones.
4
+ * Banner que avisa al usuario cuando hay una versión nueva de la app.
5
+ * Consume `AppVersionService` (no toca `SwUpdate` ni `AppConfigService`
6
+ * directamente) y soporta dos modos:
7
+ *
8
+ * - `update-available` — banner descartable: mensaje + botón "Actualizar"
9
+ * + botón cerrar.
10
+ * - `update-required` — overlay bloqueante no-descartable (hard gate):
11
+ * mensaje + botón "Actualizar". Sin botón cerrar.
12
+ *
13
+ * Cuando el estado es `up-to-date` (o el usuario descartó el banner opcional)
14
+ * no renderiza nada.
6
15
  */
7
16
  import { CommonModule } from '@angular/common';
8
- import { Component, computed, inject, signal } from '@angular/core';
9
- import { BannerComponent } from '../../organisms/banner/banner.component';
10
- import { AppConfigService } from '../../../services/app-config/app-config.service';
17
+ import { Component, computed, inject } from '@angular/core';
11
18
  import { I18nService } from '../../../services/i18n';
19
+ import { AppVersionService } from '../../../services/app-version/app-version.service';
20
+ import { ButtonComponent } from '../../atoms/button/button.component';
21
+ import { IconComponent } from '../../atoms/icon/icon.component';
22
+ import { TextComponent } from '../../atoms/text/text.component';
23
+ import { UPDATE_BANNER_DEFAULT_CONTENT, UPDATE_BANNER_I18N_NAMESPACE } from './types';
12
24
  import * as i0 from "@angular/core";
13
25
  /**
14
26
  * val-update-banner
15
27
  *
16
- * Banner de actualización que aparece cuando hay una nueva versión disponible.
17
- * Se cierra automáticamente cuando el usuario hace clic en "Actualizar" o en cerrar.
28
+ * Avisa al usuario de una versión nueva. Colócalo una sola vez, alto en el
29
+ * árbol de la app (p.ej. en `app.component` por encima del router outlet)
30
+ * para que sea visible en cualquier ruta.
18
31
  *
19
32
  * @example
20
33
  * ```html
21
- * <!-- Uso básico -->
22
34
  * <val-update-banner />
23
- *
24
- * <!-- El banner solo aparece si hay actualización disponible -->
25
35
  * ```
26
36
  */
27
37
  export class UpdateBannerComponent {
28
38
  constructor() {
29
- this.appConfig = inject(AppConfigService);
39
+ this.version = inject(AppVersionService);
30
40
  this.i18n = inject(I18nService);
31
41
  /**
32
- * Signal para controlar si el banner fue descartado por el usuario.
42
+ * True cuando el banner está en modo obligatorio (hard gate, no descartable).
33
43
  */
34
- this.dismissed = signal(false);
44
+ this.required = computed(() => this.version.status() === 'update-required');
35
45
  /**
36
- * Indica si el banner debe mostrarse.
46
+ * True cuando el banner debe renderizarse:
47
+ * - siempre que el estado sea `update-required`, o
48
+ * - si es `update-available` y el usuario no lo descartó.
37
49
  */
38
- this.show = computed(() => this.appConfig.hasUpdate() && !this.dismissed());
50
+ this.visible = computed(() => {
51
+ const status = this.version.status();
52
+ if (status === 'update-required')
53
+ return true;
54
+ if (status === 'update-available')
55
+ return !this.version.dismissed();
56
+ return false;
57
+ });
39
58
  /**
40
- * Props del banner calculadas reactivamente.
59
+ * Textos resolvidos del banner. Reactivo a idioma y al estado required.
60
+ * Usa el namespace `UpdateBanner` de `I18nService` con fallback embebido.
41
61
  */
42
- this.bannerProps = computed(() => ({
43
- color: 'primary',
44
- bordered: false,
45
- closable: true,
46
- mode: 'center',
47
- alignment: 'center',
48
- padding: '0.5rem 1rem',
49
- content: {
50
- position: 'center',
51
- title: {
52
- content: this.i18n.t('clickToUpdate', 'AppConfig'),
53
- size: 'small',
54
- color: 'light',
55
- bold: false,
56
- },
57
- },
58
- actions: {
59
- position: 'center',
60
- columned: false,
61
- buttons: [
62
- {
63
- token: 'update',
64
- text: this.i18n.t('updateNow', 'AppConfig'),
65
- color: 'light',
66
- fill: 'outline',
67
- size: 'small',
68
- state: 'ENABLED',
69
- type: 'button',
70
- },
71
- ],
72
- },
73
- }));
62
+ this.t = computed(() => {
63
+ this.i18n.lang(); // track para reactividad
64
+ const isRequired = this.required();
65
+ return {
66
+ title: this.tr(isRequired ? 'requiredTitle' : 'availableTitle'),
67
+ message: this.tr(isRequired ? 'requiredMessage' : 'availableMessage'),
68
+ updateAction: this.tr('updateAction'),
69
+ dismissAction: this.tr('dismissAction'),
70
+ };
71
+ });
74
72
  }
75
- /**
76
- * Maneja el clic en acciones del banner.
77
- */
78
- onAction(token) {
79
- if (token === 'update') {
80
- window.location.reload();
81
- }
73
+ /** Aplica la actualización (activa el SW si existe + recarga). */
74
+ onUpdate() {
75
+ void this.version.applyUpdate();
76
+ }
77
+ /** Descarta el banner opcional. No-op en modo obligatorio. */
78
+ onDismiss() {
79
+ this.version.dismiss();
82
80
  }
83
81
  /**
84
- * Descarta el banner temporalmente (hasta que se recargue la página).
82
+ * Resuelve una key i18n del namespace `UpdateBanner` con fallback al
83
+ * contenido por defecto embebido (es / en / pt).
85
84
  */
86
- dismiss() {
87
- this.dismissed.set(true);
85
+ tr(key) {
86
+ const value = this.i18n.t(key, UPDATE_BANNER_I18N_NAMESPACE);
87
+ if (value && !value.startsWith('['))
88
+ return value;
89
+ const lang = this.i18n.lang();
90
+ const pack = UPDATE_BANNER_DEFAULT_CONTENT[lang] ??
91
+ UPDATE_BANNER_DEFAULT_CONTENT.es;
92
+ return pack[key];
88
93
  }
89
94
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UpdateBannerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
90
95
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UpdateBannerComponent, isStandalone: true, selector: "val-update-banner", ngImport: i0, template: `
91
- @if (show()) {
92
- <val-banner
93
- [props]="bannerProps()"
94
- (onClick)="onAction($event)"
95
- (onClose)="dismiss()"
96
- />
96
+ @if (visible()) {
97
+ <div
98
+ class="val-update-banner"
99
+ [class.val-update-banner--required]="required()"
100
+ role="alert"
101
+ [attr.aria-live]="required() ? 'assertive' : 'polite'"
102
+ >
103
+ @if (required()) {
104
+ <div class="val-update-banner__backdrop"></div>
105
+ }
106
+
107
+ <div class="val-update-banner__panel">
108
+ <div class="val-update-banner__icon">
109
+ <val-icon
110
+ [props]="{
111
+ name: required() ? 'warning-outline' : 'cloud-download-outline',
112
+ size: 'large',
113
+ color: required() ? 'warning' : 'primary',
114
+ }"
115
+ />
116
+ </div>
117
+
118
+ <div class="val-update-banner__body">
119
+ <val-text
120
+ [props]="{
121
+ content: t().title,
122
+ size: 'medium',
123
+ bold: true,
124
+ color: 'dark',
125
+ }"
126
+ />
127
+ <val-text
128
+ [props]="{
129
+ content: t().message,
130
+ size: 'small',
131
+ bold: false,
132
+ color: 'medium',
133
+ }"
134
+ />
135
+ </div>
136
+
137
+ <div class="val-update-banner__actions">
138
+ <val-button
139
+ [props]="{
140
+ text: t().updateAction,
141
+ color: 'primary',
142
+ fill: 'solid',
143
+ size: 'small',
144
+ state: 'ENABLED',
145
+ type: 'button',
146
+ }"
147
+ (onClick)="onUpdate()"
148
+ />
149
+ @if (!required()) {
150
+ <val-button
151
+ [props]="{
152
+ text: t().dismissAction,
153
+ color: 'medium',
154
+ fill: 'clear',
155
+ size: 'small',
156
+ state: 'ENABLED',
157
+ type: 'button',
158
+ }"
159
+ (onClick)="onDismiss()"
160
+ />
161
+ }
162
+ </div>
163
+ </div>
164
+ </div>
97
165
  }
98
- `, isInline: true, styles: [":host{display:block}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: BannerComponent, selector: "val-banner", inputs: ["props"], outputs: ["onClick", "onClose"] }] }); }
166
+ `, isInline: true, styles: [":host{display:contents}.val-update-banner{position:fixed;top:0;left:0;right:0;z-index:1100;display:flex;justify-content:center;padding:calc(12px + env(safe-area-inset-top,0px)) 16px 12px;pointer-events:none}.val-update-banner__backdrop{position:fixed;inset:0;z-index:-1;background:#0000008c;backdrop-filter:blur(2px);-webkit-backdrop-filter:blur(2px);pointer-events:auto}.val-update-banner__panel{pointer-events:auto;display:flex;flex-direction:row;align-items:center;gap:12px;width:100%;max-width:640px;padding:12px 16px;background:var(--ion-background-color, #fff);border:1px solid var(--val-border-color, rgba(0, 0, 0, .08));border-radius:var(--val-border-radius, 12px);box-shadow:0 4px 24px #0000001f;animation:val-update-banner-in .25s ease-out}.val-update-banner--required{align-items:center;height:100%}.val-update-banner--required .val-update-banner__panel{flex-direction:column;text-align:center;max-width:420px}.val-update-banner__icon{flex:0 0 auto;display:inline-flex;align-items:center}.val-update-banner__body{flex:1 1 auto;display:flex;flex-direction:column;gap:2px;min-width:0}.val-update-banner__actions{flex:0 0 auto;display:flex;flex-direction:row;align-items:center;gap:4px}.val-update-banner--required .val-update-banner__actions{margin-top:8px}@keyframes val-update-banner-in{0%{opacity:0;transform:translateY(-12px)}to{opacity:1;transform:translateY(0)}}@media (max-width: 540px){.val-update-banner__panel{flex-direction:column;text-align:center}.val-update-banner__actions{width:100%;justify-content:center}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: TextComponent, selector: "val-text", inputs: ["props"] }, { kind: "component", type: ButtonComponent, selector: "val-button", inputs: ["preset", "props"], outputs: ["onClick"] }, { kind: "component", type: IconComponent, selector: "val-icon", inputs: ["props"] }] }); }
99
167
  }
100
168
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UpdateBannerComponent, decorators: [{
101
169
  type: Component,
102
- args: [{ selector: 'val-update-banner', standalone: true, imports: [CommonModule, BannerComponent], template: `
103
- @if (show()) {
104
- <val-banner
105
- [props]="bannerProps()"
106
- (onClick)="onAction($event)"
107
- (onClose)="dismiss()"
108
- />
170
+ args: [{ selector: 'val-update-banner', standalone: true, imports: [CommonModule, TextComponent, ButtonComponent, IconComponent], template: `
171
+ @if (visible()) {
172
+ <div
173
+ class="val-update-banner"
174
+ [class.val-update-banner--required]="required()"
175
+ role="alert"
176
+ [attr.aria-live]="required() ? 'assertive' : 'polite'"
177
+ >
178
+ @if (required()) {
179
+ <div class="val-update-banner__backdrop"></div>
180
+ }
181
+
182
+ <div class="val-update-banner__panel">
183
+ <div class="val-update-banner__icon">
184
+ <val-icon
185
+ [props]="{
186
+ name: required() ? 'warning-outline' : 'cloud-download-outline',
187
+ size: 'large',
188
+ color: required() ? 'warning' : 'primary',
189
+ }"
190
+ />
191
+ </div>
192
+
193
+ <div class="val-update-banner__body">
194
+ <val-text
195
+ [props]="{
196
+ content: t().title,
197
+ size: 'medium',
198
+ bold: true,
199
+ color: 'dark',
200
+ }"
201
+ />
202
+ <val-text
203
+ [props]="{
204
+ content: t().message,
205
+ size: 'small',
206
+ bold: false,
207
+ color: 'medium',
208
+ }"
209
+ />
210
+ </div>
211
+
212
+ <div class="val-update-banner__actions">
213
+ <val-button
214
+ [props]="{
215
+ text: t().updateAction,
216
+ color: 'primary',
217
+ fill: 'solid',
218
+ size: 'small',
219
+ state: 'ENABLED',
220
+ type: 'button',
221
+ }"
222
+ (onClick)="onUpdate()"
223
+ />
224
+ @if (!required()) {
225
+ <val-button
226
+ [props]="{
227
+ text: t().dismissAction,
228
+ color: 'medium',
229
+ fill: 'clear',
230
+ size: 'small',
231
+ state: 'ENABLED',
232
+ type: 'button',
233
+ }"
234
+ (onClick)="onDismiss()"
235
+ />
236
+ }
237
+ </div>
238
+ </div>
239
+ </div>
109
240
  }
110
- `, styles: [":host{display:block}\n"] }]
241
+ `, styles: [":host{display:contents}.val-update-banner{position:fixed;top:0;left:0;right:0;z-index:1100;display:flex;justify-content:center;padding:calc(12px + env(safe-area-inset-top,0px)) 16px 12px;pointer-events:none}.val-update-banner__backdrop{position:fixed;inset:0;z-index:-1;background:#0000008c;backdrop-filter:blur(2px);-webkit-backdrop-filter:blur(2px);pointer-events:auto}.val-update-banner__panel{pointer-events:auto;display:flex;flex-direction:row;align-items:center;gap:12px;width:100%;max-width:640px;padding:12px 16px;background:var(--ion-background-color, #fff);border:1px solid var(--val-border-color, rgba(0, 0, 0, .08));border-radius:var(--val-border-radius, 12px);box-shadow:0 4px 24px #0000001f;animation:val-update-banner-in .25s ease-out}.val-update-banner--required{align-items:center;height:100%}.val-update-banner--required .val-update-banner__panel{flex-direction:column;text-align:center;max-width:420px}.val-update-banner__icon{flex:0 0 auto;display:inline-flex;align-items:center}.val-update-banner__body{flex:1 1 auto;display:flex;flex-direction:column;gap:2px;min-width:0}.val-update-banner__actions{flex:0 0 auto;display:flex;flex-direction:row;align-items:center;gap:4px}.val-update-banner--required .val-update-banner__actions{margin-top:8px}@keyframes val-update-banner-in{0%{opacity:0;transform:translateY(-12px)}to{opacity:1;transform:translateY(0)}}@media (max-width: 540px){.val-update-banner__panel{flex-direction:column;text-align:center}.val-update-banner__actions{width:100%;justify-content:center}}\n"] }]
111
242
  }] });
112
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBkYXRlLWJhbm5lci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3VwZGF0ZS1iYW5uZXIvdXBkYXRlLWJhbm5lci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7O0dBS0c7QUFFSCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVwRSxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0seUNBQXlDLENBQUM7QUFDMUUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0saURBQWlELENBQUM7QUFDbkYsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHdCQUF3QixDQUFDOztBQUdyRDs7Ozs7Ozs7Ozs7OztHQWFHO0FBc0JILE1BQU0sT0FBTyxxQkFBcUI7SUFyQmxDO1FBc0JVLGNBQVMsR0FBRyxNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztRQUNyQyxTQUFJLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBRW5DOztXQUVHO1FBQ0ssY0FBUyxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUVsQzs7V0FFRztRQUNNLFNBQUksR0FBRyxRQUFRLENBQ3RCLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQ3RELENBQUM7UUFFRjs7V0FFRztRQUNNLGdCQUFXLEdBQUcsUUFBUSxDQUFpQixHQUFHLEVBQUUsQ0FBQyxDQUFDO1lBQ3JELEtBQUssRUFBRSxTQUFTO1lBQ2hCLFFBQVEsRUFBRSxLQUFLO1lBQ2YsUUFBUSxFQUFFLElBQUk7WUFDZCxJQUFJLEVBQUUsUUFBUTtZQUNkLFNBQVMsRUFBRSxRQUFRO1lBQ25CLE9BQU8sRUFBRSxhQUFhO1lBQ3RCLE9BQU8sRUFBRTtnQkFDUCxRQUFRLEVBQUUsUUFBUTtnQkFDbEIsS0FBSyxFQUFFO29CQUNMLE9BQU8sRUFBRSxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxlQUFlLEVBQUUsV0FBVyxDQUFDO29CQUNsRCxJQUFJLEVBQUUsT0FBTztvQkFDYixLQUFLLEVBQUUsT0FBTztvQkFDZCxJQUFJLEVBQUUsS0FBSztpQkFDWjthQUNGO1lBQ0QsT0FBTyxFQUFFO2dCQUNQLFFBQVEsRUFBRSxRQUFRO2dCQUNsQixRQUFRLEVBQUUsS0FBSztnQkFDZixPQUFPLEVBQUU7b0JBQ1A7d0JBQ0UsS0FBSyxFQUFFLFFBQVE7d0JBQ2YsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLFdBQVcsRUFBRSxXQUFXLENBQUM7d0JBQzNDLEtBQUssRUFBRSxPQUFPO3dCQUNkLElBQUksRUFBRSxTQUFTO3dCQUNmLElBQUksRUFBRSxPQUFPO3dCQUNiLEtBQUssRUFBRSxTQUFTO3dCQUNoQixJQUFJLEVBQUUsUUFBUTtxQkFDZjtpQkFDRjthQUNGO1NBQ0YsQ0FBQyxDQUFDLENBQUM7S0FpQkw7SUFmQzs7T0FFRztJQUNILFFBQVEsQ0FBQyxLQUFhO1FBQ3BCLElBQUksS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQ3ZCLE1BQU0sQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDM0IsQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNILE9BQU87UUFDTCxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMzQixDQUFDOytHQWxFVSxxQkFBcUI7bUdBQXJCLHFCQUFxQiw2RUFqQnRCOzs7Ozs7OztHQVFULCtGQVRTLFlBQVksK0JBQUUsZUFBZTs7NEZBa0I1QixxQkFBcUI7a0JBckJqQyxTQUFTOytCQUNFLG1CQUFtQixjQUNqQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsZUFBZSxDQUFDLFlBQzlCOzs7Ozs7OztHQVFUIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBVcGRhdGVCYW5uZXJDb21wb25lbnRcbiAqXG4gKiBDb21wb25lbnRlIHF1ZSBtdWVzdHJhIHVuIGJhbm5lciBjdWFuZG8gaGF5IHVuYSBudWV2YSB2ZXJzacOzbiBkZSBsYSBhcGxpY2FjacOzbiBkaXNwb25pYmxlLlxuICogU2UgaW50ZWdyYSBhdXRvbcOhdGljYW1lbnRlIGNvbiBBcHBDb25maWdTZXJ2aWNlIHBhcmEgZGV0ZWN0YXIgYWN0dWFsaXphY2lvbmVzLlxuICovXG5cbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIGNvbXB1dGVkLCBpbmplY3QsIHNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQgeyBCYW5uZXJDb21wb25lbnQgfSBmcm9tICcuLi8uLi9vcmdhbmlzbXMvYmFubmVyL2Jhbm5lci5jb21wb25lbnQnO1xuaW1wb3J0IHsgQXBwQ29uZmlnU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2FwcC1jb25maWcvYXBwLWNvbmZpZy5zZXJ2aWNlJztcbmltcG9ydCB7IEkxOG5TZXJ2aWNlIH0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvaTE4bic7XG5pbXBvcnQgeyBCYW5uZXJNZXRhZGF0YSB9IGZyb20gJy4uLy4uL29yZ2FuaXNtcy9iYW5uZXIvdHlwZXMnO1xuXG4vKipcbiAqIHZhbC11cGRhdGUtYmFubmVyXG4gKlxuICogQmFubmVyIGRlIGFjdHVhbGl6YWNpw7NuIHF1ZSBhcGFyZWNlIGN1YW5kbyBoYXkgdW5hIG51ZXZhIHZlcnNpw7NuIGRpc3BvbmlibGUuXG4gKiBTZSBjaWVycmEgYXV0b23DoXRpY2FtZW50ZSBjdWFuZG8gZWwgdXN1YXJpbyBoYWNlIGNsaWMgZW4gXCJBY3R1YWxpemFyXCIgbyBlbiBjZXJyYXIuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYGh0bWxcbiAqIDwhLS0gVXNvIGLDoXNpY28gLS0+XG4gKiA8dmFsLXVwZGF0ZS1iYW5uZXIgLz5cbiAqXG4gKiA8IS0tIEVsIGJhbm5lciBzb2xvIGFwYXJlY2Ugc2kgaGF5IGFjdHVhbGl6YWNpw7NuIGRpc3BvbmlibGUgLS0+XG4gKiBgYGBcbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLXVwZGF0ZS1iYW5uZXInLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBCYW5uZXJDb21wb25lbnRdLFxuICB0ZW1wbGF0ZTogYFxuICAgIEBpZiAoc2hvdygpKSB7XG4gICAgICA8dmFsLWJhbm5lclxuICAgICAgICBbcHJvcHNdPVwiYmFubmVyUHJvcHMoKVwiXG4gICAgICAgIChvbkNsaWNrKT1cIm9uQWN0aW9uKCRldmVudClcIlxuICAgICAgICAob25DbG9zZSk9XCJkaXNtaXNzKClcIlxuICAgICAgLz5cbiAgICB9XG4gIGAsXG4gIHN0eWxlczogW1xuICAgIGBcbiAgICAgIDpob3N0IHtcbiAgICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICB9XG4gICAgYCxcbiAgXSxcbn0pXG5leHBvcnQgY2xhc3MgVXBkYXRlQmFubmVyQ29tcG9uZW50IHtcbiAgcHJpdmF0ZSBhcHBDb25maWcgPSBpbmplY3QoQXBwQ29uZmlnU2VydmljZSk7XG4gIHByaXZhdGUgaTE4biA9IGluamVjdChJMThuU2VydmljZSk7XG5cbiAgLyoqXG4gICAqIFNpZ25hbCBwYXJhIGNvbnRyb2xhciBzaSBlbCBiYW5uZXIgZnVlIGRlc2NhcnRhZG8gcG9yIGVsIHVzdWFyaW8uXG4gICAqL1xuICBwcml2YXRlIGRpc21pc3NlZCA9IHNpZ25hbChmYWxzZSk7XG5cbiAgLyoqXG4gICAqIEluZGljYSBzaSBlbCBiYW5uZXIgZGViZSBtb3N0cmFyc2UuXG4gICAqL1xuICByZWFkb25seSBzaG93ID0gY29tcHV0ZWQoXG4gICAgKCkgPT4gdGhpcy5hcHBDb25maWcuaGFzVXBkYXRlKCkgJiYgIXRoaXMuZGlzbWlzc2VkKClcbiAgKTtcblxuICAvKipcbiAgICogUHJvcHMgZGVsIGJhbm5lciBjYWxjdWxhZGFzIHJlYWN0aXZhbWVudGUuXG4gICAqL1xuICByZWFkb25seSBiYW5uZXJQcm9wcyA9IGNvbXB1dGVkPEJhbm5lck1ldGFkYXRhPigoKSA9PiAoe1xuICAgIGNvbG9yOiAncHJpbWFyeScsXG4gICAgYm9yZGVyZWQ6IGZhbHNlLFxuICAgIGNsb3NhYmxlOiB0cnVlLFxuICAgIG1vZGU6ICdjZW50ZXInLFxuICAgIGFsaWdubWVudDogJ2NlbnRlcicsXG4gICAgcGFkZGluZzogJzAuNXJlbSAxcmVtJyxcbiAgICBjb250ZW50OiB7XG4gICAgICBwb3NpdGlvbjogJ2NlbnRlcicsXG4gICAgICB0aXRsZToge1xuICAgICAgICBjb250ZW50OiB0aGlzLmkxOG4udCgnY2xpY2tUb1VwZGF0ZScsICdBcHBDb25maWcnKSxcbiAgICAgICAgc2l6ZTogJ3NtYWxsJyxcbiAgICAgICAgY29sb3I6ICdsaWdodCcsXG4gICAgICAgIGJvbGQ6IGZhbHNlLFxuICAgICAgfSxcbiAgICB9LFxuICAgIGFjdGlvbnM6IHtcbiAgICAgIHBvc2l0aW9uOiAnY2VudGVyJyxcbiAgICAgIGNvbHVtbmVkOiBmYWxzZSxcbiAgICAgIGJ1dHRvbnM6IFtcbiAgICAgICAge1xuICAgICAgICAgIHRva2VuOiAndXBkYXRlJyxcbiAgICAgICAgICB0ZXh0OiB0aGlzLmkxOG4udCgndXBkYXRlTm93JywgJ0FwcENvbmZpZycpLFxuICAgICAgICAgIGNvbG9yOiAnbGlnaHQnLFxuICAgICAgICAgIGZpbGw6ICdvdXRsaW5lJyxcbiAgICAgICAgICBzaXplOiAnc21hbGwnLFxuICAgICAgICAgIHN0YXRlOiAnRU5BQkxFRCcsXG4gICAgICAgICAgdHlwZTogJ2J1dHRvbicsXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH0sXG4gIH0pKTtcblxuICAvKipcbiAgICogTWFuZWphIGVsIGNsaWMgZW4gYWNjaW9uZXMgZGVsIGJhbm5lci5cbiAgICovXG4gIG9uQWN0aW9uKHRva2VuOiBzdHJpbmcpOiB2b2lkIHtcbiAgICBpZiAodG9rZW4gPT09ICd1cGRhdGUnKSB7XG4gICAgICB3aW5kb3cubG9jYXRpb24ucmVsb2FkKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIERlc2NhcnRhIGVsIGJhbm5lciB0ZW1wb3JhbG1lbnRlIChoYXN0YSBxdWUgc2UgcmVjYXJndWUgbGEgcMOhZ2luYSkuXG4gICAqL1xuICBkaXNtaXNzKCk6IHZvaWQge1xuICAgIHRoaXMuZGlzbWlzc2VkLnNldCh0cnVlKTtcbiAgfVxufVxuIl19
243
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBkYXRlLWJhbm5lci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3VwZGF0ZS1iYW5uZXIvdXBkYXRlLWJhbm5lci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7O0dBY0c7QUFFSCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRTVELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUNyRCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxtREFBbUQsQ0FBQztBQUN0RixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDdEUsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ2hFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUNoRSxPQUFPLEVBQUUsNkJBQTZCLEVBQUUsNEJBQTRCLEVBQUUsTUFBTSxTQUFTLENBQUM7O0FBRXRGOzs7Ozs7Ozs7OztHQVdHO0FBK0VILE1BQU0sT0FBTyxxQkFBcUI7SUE5RWxDO1FBK0VtQixZQUFPLEdBQUcsTUFBTSxDQUFDLGlCQUFpQixDQUFDLENBQUM7UUFDcEMsU0FBSSxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUU1Qzs7V0FFRztRQUNNLGFBQVEsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsS0FBSyxpQkFBaUIsQ0FBQyxDQUFDO1FBRWhGOzs7O1dBSUc7UUFDTSxZQUFPLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUMvQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLE1BQU0sRUFBRSxDQUFDO1lBQ3JDLElBQUksTUFBTSxLQUFLLGlCQUFpQjtnQkFBRSxPQUFPLElBQUksQ0FBQztZQUM5QyxJQUFJLE1BQU0sS0FBSyxrQkFBa0I7Z0JBQUUsT0FBTyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDcEUsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDLENBQUMsQ0FBQztRQUVIOzs7V0FHRztRQUNNLE1BQUMsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1lBQ3pCLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQyx5QkFBeUI7WUFDM0MsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1lBQ25DLE9BQU87Z0JBQ0wsS0FBSyxFQUFFLElBQUksQ0FBQyxFQUFFLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxlQUFlLENBQUMsQ0FBQyxDQUFDLGdCQUFnQixDQUFDO2dCQUMvRCxPQUFPLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLGlCQUFpQixDQUFDLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztnQkFDckUsWUFBWSxFQUFFLElBQUksQ0FBQyxFQUFFLENBQUMsY0FBYyxDQUFDO2dCQUNyQyxhQUFhLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxlQUFlLENBQUM7YUFDeEMsQ0FBQztRQUNKLENBQUMsQ0FBQyxDQUFDO0tBMEJKO0lBeEJDLGtFQUFrRTtJQUNsRSxRQUFRO1FBQ04sS0FBSyxJQUFJLENBQUMsT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQ2xDLENBQUM7SUFFRCw4REFBOEQ7SUFDOUQsU0FBUztRQUNQLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDekIsQ0FBQztJQUVEOzs7T0FHRztJQUNLLEVBQUUsQ0FBQyxHQUFrRDtRQUMzRCxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxHQUFHLEVBQUUsNEJBQTRCLENBQUMsQ0FBQztRQUM3RCxJQUFJLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDO1lBQUUsT0FBTyxLQUFLLENBQUM7UUFFbEQsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUM5QixNQUFNLElBQUksR0FDUiw2QkFBNkIsQ0FBQyxJQUFrRCxDQUFDO1lBQ2pGLDZCQUE2QixDQUFDLEVBQUUsQ0FBQztRQUNuQyxPQUFPLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNuQixDQUFDOytHQTNEVSxxQkFBcUI7bUdBQXJCLHFCQUFxQiw2RUExRXRCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVFVCx1a0RBeEVTLFlBQVksK0JBQUUsYUFBYSx3RUFBRSxlQUFlLDBHQUFFLGFBQWE7OzRGQTJFMUQscUJBQXFCO2tCQTlFakMsU0FBUzsrQkFDRSxtQkFBbUIsY0FDakIsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsYUFBYSxDQUFDLFlBQzVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVFVCIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVXBkYXRlQmFubmVyQ29tcG9uZW50XG4gKlxuICogQmFubmVyIHF1ZSBhdmlzYSBhbCB1c3VhcmlvIGN1YW5kbyBoYXkgdW5hIHZlcnNpw7NuIG51ZXZhIGRlIGxhIGFwcC5cbiAqIENvbnN1bWUgYEFwcFZlcnNpb25TZXJ2aWNlYCAobm8gdG9jYSBgU3dVcGRhdGVgIG5pIGBBcHBDb25maWdTZXJ2aWNlYFxuICogZGlyZWN0YW1lbnRlKSB5IHNvcG9ydGEgZG9zIG1vZG9zOlxuICpcbiAqIC0gYHVwZGF0ZS1hdmFpbGFibGVgIOKAlCBiYW5uZXIgZGVzY2FydGFibGU6IG1lbnNhamUgKyBib3TDs24gXCJBY3R1YWxpemFyXCJcbiAqICAgKyBib3TDs24gY2VycmFyLlxuICogLSBgdXBkYXRlLXJlcXVpcmVkYCDigJQgb3ZlcmxheSBibG9xdWVhbnRlIG5vLWRlc2NhcnRhYmxlIChoYXJkIGdhdGUpOlxuICogICBtZW5zYWplICsgYm90w7NuIFwiQWN0dWFsaXphclwiLiBTaW4gYm90w7NuIGNlcnJhci5cbiAqXG4gKiBDdWFuZG8gZWwgZXN0YWRvIGVzIGB1cC10by1kYXRlYCAobyBlbCB1c3VhcmlvIGRlc2NhcnTDsyBlbCBiYW5uZXIgb3BjaW9uYWwpXG4gKiBubyByZW5kZXJpemEgbmFkYS5cbiAqL1xuXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBjb21wdXRlZCwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbmltcG9ydCB7IEkxOG5TZXJ2aWNlIH0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvaTE4bic7XG5pbXBvcnQgeyBBcHBWZXJzaW9uU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2FwcC12ZXJzaW9uL2FwcC12ZXJzaW9uLnNlcnZpY2UnO1xuaW1wb3J0IHsgQnV0dG9uQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvYnV0dG9uL2J1dHRvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgSWNvbkNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2ljb24vaWNvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgVGV4dENvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL3RleHQvdGV4dC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVVBEQVRFX0JBTk5FUl9ERUZBVUxUX0NPTlRFTlQsIFVQREFURV9CQU5ORVJfSTE4Tl9OQU1FU1BBQ0UgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiB2YWwtdXBkYXRlLWJhbm5lclxuICpcbiAqIEF2aXNhIGFsIHVzdWFyaW8gZGUgdW5hIHZlcnNpw7NuIG51ZXZhLiBDb2zDs2NhbG8gdW5hIHNvbGEgdmV6LCBhbHRvIGVuIGVsXG4gKiDDoXJib2wgZGUgbGEgYXBwIChwLmVqLiBlbiBgYXBwLmNvbXBvbmVudGAgcG9yIGVuY2ltYSBkZWwgcm91dGVyIG91dGxldClcbiAqIHBhcmEgcXVlIHNlYSB2aXNpYmxlIGVuIGN1YWxxdWllciBydXRhLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGBodG1sXG4gKiA8dmFsLXVwZGF0ZS1iYW5uZXIgLz5cbiAqIGBgYFxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtdXBkYXRlLWJhbm5lcicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFRleHRDb21wb25lbnQsIEJ1dHRvbkNvbXBvbmVudCwgSWNvbkNvbXBvbmVudF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgQGlmICh2aXNpYmxlKCkpIHtcbiAgICAgIDxkaXZcbiAgICAgICAgY2xhc3M9XCJ2YWwtdXBkYXRlLWJhbm5lclwiXG4gICAgICAgIFtjbGFzcy52YWwtdXBkYXRlLWJhbm5lci0tcmVxdWlyZWRdPVwicmVxdWlyZWQoKVwiXG4gICAgICAgIHJvbGU9XCJhbGVydFwiXG4gICAgICAgIFthdHRyLmFyaWEtbGl2ZV09XCJyZXF1aXJlZCgpID8gJ2Fzc2VydGl2ZScgOiAncG9saXRlJ1wiXG4gICAgICA+XG4gICAgICAgIEBpZiAocmVxdWlyZWQoKSkge1xuICAgICAgICAgIDxkaXYgY2xhc3M9XCJ2YWwtdXBkYXRlLWJhbm5lcl9fYmFja2Ryb3BcIj48L2Rpdj5cbiAgICAgICAgfVxuXG4gICAgICAgIDxkaXYgY2xhc3M9XCJ2YWwtdXBkYXRlLWJhbm5lcl9fcGFuZWxcIj5cbiAgICAgICAgICA8ZGl2IGNsYXNzPVwidmFsLXVwZGF0ZS1iYW5uZXJfX2ljb25cIj5cbiAgICAgICAgICAgIDx2YWwtaWNvblxuICAgICAgICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgICAgICAgIG5hbWU6IHJlcXVpcmVkKCkgPyAnd2FybmluZy1vdXRsaW5lJyA6ICdjbG91ZC1kb3dubG9hZC1vdXRsaW5lJyxcbiAgICAgICAgICAgICAgICBzaXplOiAnbGFyZ2UnLFxuICAgICAgICAgICAgICAgIGNvbG9yOiByZXF1aXJlZCgpID8gJ3dhcm5pbmcnIDogJ3ByaW1hcnknLFxuICAgICAgICAgICAgICB9XCJcbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9kaXY+XG5cbiAgICAgICAgICA8ZGl2IGNsYXNzPVwidmFsLXVwZGF0ZS1iYW5uZXJfX2JvZHlcIj5cbiAgICAgICAgICAgIDx2YWwtdGV4dFxuICAgICAgICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgICAgICAgIGNvbnRlbnQ6IHQoKS50aXRsZSxcbiAgICAgICAgICAgICAgICBzaXplOiAnbWVkaXVtJyxcbiAgICAgICAgICAgICAgICBib2xkOiB0cnVlLFxuICAgICAgICAgICAgICAgIGNvbG9yOiAnZGFyaycsXG4gICAgICAgICAgICAgIH1cIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICAgIDx2YWwtdGV4dFxuICAgICAgICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgICAgICAgIGNvbnRlbnQ6IHQoKS5tZXNzYWdlLFxuICAgICAgICAgICAgICAgIHNpemU6ICdzbWFsbCcsXG4gICAgICAgICAgICAgICAgYm9sZDogZmFsc2UsXG4gICAgICAgICAgICAgICAgY29sb3I6ICdtZWRpdW0nLFxuICAgICAgICAgICAgICB9XCJcbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9kaXY+XG5cbiAgICAgICAgICA8ZGl2IGNsYXNzPVwidmFsLXVwZGF0ZS1iYW5uZXJfX2FjdGlvbnNcIj5cbiAgICAgICAgICAgIDx2YWwtYnV0dG9uXG4gICAgICAgICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgICAgICAgdGV4dDogdCgpLnVwZGF0ZUFjdGlvbixcbiAgICAgICAgICAgICAgICBjb2xvcjogJ3ByaW1hcnknLFxuICAgICAgICAgICAgICAgIGZpbGw6ICdzb2xpZCcsXG4gICAgICAgICAgICAgICAgc2l6ZTogJ3NtYWxsJyxcbiAgICAgICAgICAgICAgICBzdGF0ZTogJ0VOQUJMRUQnLFxuICAgICAgICAgICAgICAgIHR5cGU6ICdidXR0b24nLFxuICAgICAgICAgICAgICB9XCJcbiAgICAgICAgICAgICAgKG9uQ2xpY2spPVwib25VcGRhdGUoKVwiXG4gICAgICAgICAgICAvPlxuICAgICAgICAgICAgQGlmICghcmVxdWlyZWQoKSkge1xuICAgICAgICAgICAgICA8dmFsLWJ1dHRvblxuICAgICAgICAgICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgICAgICAgICB0ZXh0OiB0KCkuZGlzbWlzc0FjdGlvbixcbiAgICAgICAgICAgICAgICAgIGNvbG9yOiAnbWVkaXVtJyxcbiAgICAgICAgICAgICAgICAgIGZpbGw6ICdjbGVhcicsXG4gICAgICAgICAgICAgICAgICBzaXplOiAnc21hbGwnLFxuICAgICAgICAgICAgICAgICAgc3RhdGU6ICdFTkFCTEVEJyxcbiAgICAgICAgICAgICAgICAgIHR5cGU6ICdidXR0b24nLFxuICAgICAgICAgICAgICAgIH1cIlxuICAgICAgICAgICAgICAgIChvbkNsaWNrKT1cIm9uRGlzbWlzcygpXCJcbiAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgIH1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICB9XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL3VwZGF0ZS1iYW5uZXIuY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgVXBkYXRlQmFubmVyQ29tcG9uZW50IHtcbiAgcHJpdmF0ZSByZWFkb25seSB2ZXJzaW9uID0gaW5qZWN0KEFwcFZlcnNpb25TZXJ2aWNlKTtcbiAgcHJpdmF0ZSByZWFkb25seSBpMThuID0gaW5qZWN0KEkxOG5TZXJ2aWNlKTtcblxuICAvKipcbiAgICogVHJ1ZSBjdWFuZG8gZWwgYmFubmVyIGVzdMOhIGVuIG1vZG8gb2JsaWdhdG9yaW8gKGhhcmQgZ2F0ZSwgbm8gZGVzY2FydGFibGUpLlxuICAgKi9cbiAgcmVhZG9ubHkgcmVxdWlyZWQgPSBjb21wdXRlZCgoKSA9PiB0aGlzLnZlcnNpb24uc3RhdHVzKCkgPT09ICd1cGRhdGUtcmVxdWlyZWQnKTtcblxuICAvKipcbiAgICogVHJ1ZSBjdWFuZG8gZWwgYmFubmVyIGRlYmUgcmVuZGVyaXphcnNlOlxuICAgKiAtIHNpZW1wcmUgcXVlIGVsIGVzdGFkbyBzZWEgYHVwZGF0ZS1yZXF1aXJlZGAsIG9cbiAgICogLSBzaSBlcyBgdXBkYXRlLWF2YWlsYWJsZWAgeSBlbCB1c3VhcmlvIG5vIGxvIGRlc2NhcnTDsy5cbiAgICovXG4gIHJlYWRvbmx5IHZpc2libGUgPSBjb21wdXRlZCgoKSA9PiB7XG4gICAgY29uc3Qgc3RhdHVzID0gdGhpcy52ZXJzaW9uLnN0YXR1cygpO1xuICAgIGlmIChzdGF0dXMgPT09ICd1cGRhdGUtcmVxdWlyZWQnKSByZXR1cm4gdHJ1ZTtcbiAgICBpZiAoc3RhdHVzID09PSAndXBkYXRlLWF2YWlsYWJsZScpIHJldHVybiAhdGhpcy52ZXJzaW9uLmRpc21pc3NlZCgpO1xuICAgIHJldHVybiBmYWxzZTtcbiAgfSk7XG5cbiAgLyoqXG4gICAqIFRleHRvcyByZXNvbHZpZG9zIGRlbCBiYW5uZXIuIFJlYWN0aXZvIGEgaWRpb21hIHkgYWwgZXN0YWRvIHJlcXVpcmVkLlxuICAgKiBVc2EgZWwgbmFtZXNwYWNlIGBVcGRhdGVCYW5uZXJgIGRlIGBJMThuU2VydmljZWAgY29uIGZhbGxiYWNrIGVtYmViaWRvLlxuICAgKi9cbiAgcmVhZG9ubHkgdCA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICB0aGlzLmkxOG4ubGFuZygpOyAvLyB0cmFjayBwYXJhIHJlYWN0aXZpZGFkXG4gICAgY29uc3QgaXNSZXF1aXJlZCA9IHRoaXMucmVxdWlyZWQoKTtcbiAgICByZXR1cm4ge1xuICAgICAgdGl0bGU6IHRoaXMudHIoaXNSZXF1aXJlZCA/ICdyZXF1aXJlZFRpdGxlJyA6ICdhdmFpbGFibGVUaXRsZScpLFxuICAgICAgbWVzc2FnZTogdGhpcy50cihpc1JlcXVpcmVkID8gJ3JlcXVpcmVkTWVzc2FnZScgOiAnYXZhaWxhYmxlTWVzc2FnZScpLFxuICAgICAgdXBkYXRlQWN0aW9uOiB0aGlzLnRyKCd1cGRhdGVBY3Rpb24nKSxcbiAgICAgIGRpc21pc3NBY3Rpb246IHRoaXMudHIoJ2Rpc21pc3NBY3Rpb24nKSxcbiAgICB9O1xuICB9KTtcblxuICAvKiogQXBsaWNhIGxhIGFjdHVhbGl6YWNpw7NuIChhY3RpdmEgZWwgU1cgc2kgZXhpc3RlICsgcmVjYXJnYSkuICovXG4gIG9uVXBkYXRlKCk6IHZvaWQge1xuICAgIHZvaWQgdGhpcy52ZXJzaW9uLmFwcGx5VXBkYXRlKCk7XG4gIH1cblxuICAvKiogRGVzY2FydGEgZWwgYmFubmVyIG9wY2lvbmFsLiBOby1vcCBlbiBtb2RvIG9ibGlnYXRvcmlvLiAqL1xuICBvbkRpc21pc3MoKTogdm9pZCB7XG4gICAgdGhpcy52ZXJzaW9uLmRpc21pc3MoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXN1ZWx2ZSB1bmEga2V5IGkxOG4gZGVsIG5hbWVzcGFjZSBgVXBkYXRlQmFubmVyYCBjb24gZmFsbGJhY2sgYWxcbiAgICogY29udGVuaWRvIHBvciBkZWZlY3RvIGVtYmViaWRvIChlcyAvIGVuIC8gcHQpLlxuICAgKi9cbiAgcHJpdmF0ZSB0cihrZXk6IGtleW9mIHR5cGVvZiBVUERBVEVfQkFOTkVSX0RFRkFVTFRfQ09OVEVOVC5lcyk6IHN0cmluZyB7XG4gICAgY29uc3QgdmFsdWUgPSB0aGlzLmkxOG4udChrZXksIFVQREFURV9CQU5ORVJfSTE4Tl9OQU1FU1BBQ0UpO1xuICAgIGlmICh2YWx1ZSAmJiAhdmFsdWUuc3RhcnRzV2l0aCgnWycpKSByZXR1cm4gdmFsdWU7XG5cbiAgICBjb25zdCBsYW5nID0gdGhpcy5pMThuLmxhbmcoKTtcbiAgICBjb25zdCBwYWNrID1cbiAgICAgIFVQREFURV9CQU5ORVJfREVGQVVMVF9DT05URU5UW2xhbmcgYXMga2V5b2YgdHlwZW9mIFVQREFURV9CQU5ORVJfREVGQVVMVF9DT05URU5UXSA/P1xuICAgICAgVVBEQVRFX0JBTk5FUl9ERUZBVUxUX0NPTlRFTlQuZXM7XG4gICAgcmV0dXJuIHBhY2tba2V5XTtcbiAgfVxufVxuIl19
@@ -0,0 +1,187 @@
1
+ /**
2
+ * AppVersionService
3
+ *
4
+ * Verificación de versión de la app para PWA / web.
5
+ *
6
+ * Unifica dos señales para detectar actualizaciones:
7
+ *
8
+ * 1. `SwUpdate` (`@angular/service-worker`) — el evento `VERSION_READY` indica
9
+ * que un bundle nuevo fue descargado y está listo para activarse. Solo está
10
+ * disponible cuando el service worker está registrado (build de producción).
11
+ * Además se chequea proactivamente cada `checkIntervalMs` y cada vez que la
12
+ * app vuelve a foreground (`visibilitychange`).
13
+ *
14
+ * 2. `AppConfigService` (config remoto en Firestore) — expone `version`
15
+ * (última versión publicada) y `minVersion` (versión mínima soportada).
16
+ * Se usa para la política de actualización obligatoria (hard gate).
17
+ *
18
+ * Ambas dependencias se inyectan como opcionales: el servicio degrada con
19
+ * gracia si el SW no está habilitado (dev mode) o si `AppConfigService` no
20
+ * está wireado en la app.
21
+ */
22
+ import { DOCUMENT } from '@angular/common';
23
+ import { DestroyRef, Injectable, computed, inject, signal } from '@angular/core';
24
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
25
+ import { SwUpdate } from '@angular/service-worker';
26
+ import { filter, interval } from 'rxjs';
27
+ import { AppConfigService } from '../app-config/app-config.service';
28
+ import { VALTECH_APP_VERSION } from './config';
29
+ import { DEFAULT_APP_VERSION_SERVICE_CONFIG } from './types';
30
+ import * as i0 from "@angular/core";
31
+ /**
32
+ * Compara dos versiones semver. Retorna true si `a` es estrictamente menor
33
+ * que `b`. Tolerante a longitudes distintas y a partes no numéricas.
34
+ */
35
+ function isVersionLower(a, b) {
36
+ const pa = a.split('.').map(n => Number(n) || 0);
37
+ const pb = b.split('.').map(n => Number(n) || 0);
38
+ const len = Math.max(pa.length, pb.length);
39
+ for (let i = 0; i < len; i++) {
40
+ const x = pa[i] || 0;
41
+ const y = pb[i] || 0;
42
+ if (x < y)
43
+ return true;
44
+ if (x > y)
45
+ return false;
46
+ }
47
+ return false;
48
+ }
49
+ export class AppVersionService {
50
+ constructor() {
51
+ /** SwUpdate solo existe si el SW está registrado (prod build). */
52
+ this.swUpdate = inject(SwUpdate, { optional: true });
53
+ /** AppConfigService es opcional: la app puede no wirearlo. */
54
+ this.appConfig = inject(AppConfigService, { optional: true });
55
+ this.serviceConfig = inject(VALTECH_APP_VERSION, {
56
+ optional: true,
57
+ });
58
+ this.document = inject(DOCUMENT);
59
+ this.destroyRef = inject(DestroyRef);
60
+ /** True cuando el SW reportó un bundle nuevo listo (`VERSION_READY`). */
61
+ this.swUpdateReady = signal(false);
62
+ /**
63
+ * True cuando el usuario descartó el banner de actualización opcional.
64
+ * El banner obligatorio (`update-required`) ignora este flag.
65
+ */
66
+ this._dismissed = signal(false);
67
+ this.dismissed = this._dismissed.asReadonly();
68
+ // ===========================================================================
69
+ // SIGNALS PÚBLICOS
70
+ // ===========================================================================
71
+ /**
72
+ * Versión local del build actual de la app.
73
+ * Proviene del provider `provideValtechAppVersion({ currentVersion })`.
74
+ */
75
+ this.currentVersion = computed(() => this.serviceConfig?.currentVersion ?? '0.0.0');
76
+ /**
77
+ * Última versión publicada según el config remoto.
78
+ * `null` si `AppConfigService` no está disponible o aún no cargó.
79
+ */
80
+ this.latestVersion = computed(() => this.appConfig?.remoteVersion() ?? null);
81
+ /**
82
+ * Estado de la versión:
83
+ * - `update-required` — la versión local quedó por debajo de `minVersion`
84
+ * del config remoto. Hard gate. Solo si `AppConfigService` tiene datos.
85
+ * - `update-available` — el SW descargó un bundle nuevo (`VERSION_READY`)
86
+ * o `AppConfigService.hasUpdate` es true.
87
+ * - `up-to-date` — en cualquier otro caso.
88
+ */
89
+ this.status = computed(() => {
90
+ // 1. Hard gate — minVersion del config remoto.
91
+ if (this.appConfig) {
92
+ const minVersion = this.appConfig.appConfig()?.minVersion;
93
+ if (minVersion && isVersionLower(this.currentVersion(), minVersion)) {
94
+ return 'update-required';
95
+ }
96
+ }
97
+ // 2. Update opcional — SW bundle listo o config remoto más nuevo.
98
+ if (this.swUpdateReady() || this.appConfig?.hasUpdate()) {
99
+ return 'update-available';
100
+ }
101
+ return 'up-to-date';
102
+ });
103
+ this.wireServiceWorker();
104
+ }
105
+ // ===========================================================================
106
+ // MÉTODOS PÚBLICOS
107
+ // ===========================================================================
108
+ /**
109
+ * Aplica la actualización pendiente.
110
+ *
111
+ * Si el SW está habilitado, activa la versión descargada y recarga; si no,
112
+ * solo recarga la página (la web obtiene el bundle nuevo del servidor).
113
+ */
114
+ async applyUpdate() {
115
+ try {
116
+ if (this.swUpdate?.isEnabled) {
117
+ await this.swUpdate.activateUpdate();
118
+ }
119
+ }
120
+ catch (err) {
121
+ console.warn('[AppVersionService] activateUpdate failed:', err);
122
+ }
123
+ finally {
124
+ this.document.defaultView?.location.reload();
125
+ }
126
+ }
127
+ /**
128
+ * Descarta el banner de actualización opcional.
129
+ * No tiene efecto sobre el estado `update-required` (hard gate).
130
+ */
131
+ dismiss() {
132
+ this._dismissed.set(true);
133
+ }
134
+ /**
135
+ * Fuerza un chequeo de actualización del service worker.
136
+ * No-op si el SW no está habilitado.
137
+ */
138
+ async checkNow() {
139
+ if (!this.swUpdate?.isEnabled)
140
+ return;
141
+ try {
142
+ await this.swUpdate.checkForUpdate();
143
+ }
144
+ catch (err) {
145
+ console.warn('[AppVersionService] checkForUpdate failed:', err);
146
+ }
147
+ }
148
+ // ===========================================================================
149
+ // PRIVADO
150
+ // ===========================================================================
151
+ /**
152
+ * Suscribe `versionUpdates`, programa chequeos periódicos y reacciona al
153
+ * volver la app a foreground. Solo actúa si el SW está habilitado.
154
+ */
155
+ wireServiceWorker() {
156
+ if (!this.swUpdate?.isEnabled)
157
+ return;
158
+ // Bundle nuevo descargado y listo.
159
+ this.swUpdate.versionUpdates
160
+ .pipe(filter(evt => evt.type === 'VERSION_READY'), takeUntilDestroyed(this.destroyRef))
161
+ .subscribe(() => this.swUpdateReady.set(true));
162
+ // Chequeo periódico.
163
+ const intervalMs = this.serviceConfig?.checkIntervalMs ?? DEFAULT_APP_VERSION_SERVICE_CONFIG.checkIntervalMs;
164
+ interval(intervalMs)
165
+ .pipe(takeUntilDestroyed(this.destroyRef))
166
+ .subscribe(() => void this.checkNow());
167
+ // Chequeo al volver la app a foreground.
168
+ const onVisible = () => {
169
+ if (this.document.visibilityState === 'visible') {
170
+ void this.checkNow();
171
+ }
172
+ };
173
+ this.document.addEventListener('visibilitychange', onVisible);
174
+ this.destroyRef.onDestroy(() => {
175
+ this.document.removeEventListener('visibilitychange', onVisible);
176
+ });
177
+ // Chequeo inicial.
178
+ void this.checkNow();
179
+ }
180
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AppVersionService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
181
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AppVersionService, providedIn: 'root' }); }
182
+ }
183
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AppVersionService, decorators: [{
184
+ type: Injectable,
185
+ args: [{ providedIn: 'root' }]
186
+ }], ctorParameters: () => [] });
187
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLXZlcnNpb24uc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2VydmljZXMvYXBwLXZlcnNpb24vYXBwLXZlcnNpb24uc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FvQkc7QUFFSCxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDM0MsT0FBTyxFQUFFLFVBQVUsRUFBRSxVQUFVLEVBQVUsUUFBUSxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekYsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDaEUsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ25ELE9BQU8sRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRXhDLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGtDQUFrQyxDQUFDO0FBQ3BFLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUMvQyxPQUFPLEVBQW9CLGtDQUFrQyxFQUFFLE1BQU0sU0FBUyxDQUFDOztBQUUvRTs7O0dBR0c7QUFDSCxTQUFTLGNBQWMsQ0FBQyxDQUFTLEVBQUUsQ0FBUztJQUMxQyxNQUFNLEVBQUUsR0FBRyxDQUFDLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUNqRCxNQUFNLEVBQUUsR0FBRyxDQUFDLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUNqRCxN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LEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1FBRWpELHFCQUFxQjtRQUNyQixNQUFNLFVBQVUsR0FDZCxJQUFJLENBQUMsYUFBYSxFQUFFLGVBQWUsSUFBSSxrQ0FBa0MsQ0FBQyxlQUFlLENBQUM7UUFFNUYsUUFBUSxDQUFDLFVBQVUsQ0FBQzthQUNqQixJQUFJLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDO2FBQ3pDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxLQUFLLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBRXpDLHlDQUF5QztRQUN6QyxNQUFNLFNBQVMsR0FBRyxHQUFTLEVBQUU7WUFDM0IsSUFBSSxJQUFJLENBQUMsUUFBUSxDQUFDLGVBQWUsS0FBSyxTQUFTLEVBQUUsQ0FBQztnQkFDaEQsS0FBSyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDdkIsQ0FBQztRQUNILENBQUMsQ0FBQztRQUNGLElBQUksQ0FBQyxRQUFRLENBQUMsZ0JBQWdCLENBQUMsa0JBQWtCLEVBQUUsU0FBUyxDQUFDLENBQUM7UUFDOUQsSUFBSSxDQUFDLFVBQVUsQ0FBQyxTQUFTLENBQUMsR0FBRyxFQUFFO1lBQzdCLElBQUksQ0FBQyxRQUFRLENBQUMsbUJBQW1CLENBQUMsa0JBQWtCLEVBQUUsU0FBUyxDQUFDLENBQUM7UUFDbkUsQ0FBQyxDQUFDLENBQUM7UUFFSCxtQkFBbUI7UUFDbkIsS0FBSyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUM7SUFDdkIsQ0FBQzsrR0F6SlUsaUJBQWlCO21IQUFqQixpQkFBaUIsY0FESixNQUFNOzs0RkFDbkIsaUJBQWlCO2tCQUQ3QixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQXBwVmVyc2lvblNlcnZpY2VcbiAqXG4gKiBWZXJpZmljYWNpw7NuIGRlIHZlcnNpw7NuIGRlIGxhIGFwcCBwYXJhIFBXQSAvIHdlYi5cbiAqXG4gKiBVbmlmaWNhIGRvcyBzZcOxYWxlcyBwYXJhIGRldGVjdGFyIGFjdHVhbGl6YWNpb25lczpcbiAqXG4gKiAxLiBgU3dVcGRhdGVgIChgQGFuZ3VsYXIvc2VydmljZS13b3JrZXJgKSDigJQgZWwgZXZlbnRvIGBWRVJTSU9OX1JFQURZYCBpbmRpY2FcbiAqICAgIHF1ZSB1biBidW5kbGUgbnVldm8gZnVlIGRlc2NhcmdhZG8geSBlc3TDoSBsaXN0byBwYXJhIGFjdGl2YXJzZS4gU29sbyBlc3TDoVxuICogICAgZGlzcG9uaWJsZSBjdWFuZG8gZWwgc2VydmljZSB3b3JrZXIgZXN0w6EgcmVnaXN0cmFkbyAoYnVpbGQgZGUgcHJvZHVjY2nDs24pLlxuICogICAgQWRlbcOhcyBzZSBjaGVxdWVhIHByb2FjdGl2YW1lbnRlIGNhZGEgYGNoZWNrSW50ZXJ2YWxNc2AgeSBjYWRhIHZleiBxdWUgbGFcbiAqICAgIGFwcCB2dWVsdmUgYSBmb3JlZ3JvdW5kIChgdmlzaWJpbGl0eWNoYW5nZWApLlxuICpcbiAqIDIuIGBBcHBDb25maWdTZXJ2aWNlYCAoY29uZmlnIHJlbW90byBlbiBGaXJlc3RvcmUpIOKAlCBleHBvbmUgYHZlcnNpb25gXG4gKiAgICAow7psdGltYSB2ZXJzacOzbiBwdWJsaWNhZGEpIHkgYG1pblZlcnNpb25gICh2ZXJzacOzbiBtw61uaW1hIHNvcG9ydGFkYSkuXG4gKiAgICBTZSB1c2EgcGFyYSBsYSBwb2zDrXRpY2EgZGUgYWN0dWFsaXphY2nDs24gb2JsaWdhdG9yaWEgKGhhcmQgZ2F0ZSkuXG4gKlxuICogQW1iYXMgZGVwZW5kZW5jaWFzIHNlIGlueWVjdGFuIGNvbW8gb3BjaW9uYWxlczogZWwgc2VydmljaW8gZGVncmFkYSBjb25cbiAqIGdyYWNpYSBzaSBlbCBTVyBubyBlc3TDoSBoYWJpbGl0YWRvIChkZXYgbW9kZSkgbyBzaSBgQXBwQ29uZmlnU2VydmljZWAgbm9cbiAqIGVzdMOhIHdpcmVhZG8gZW4gbGEgYXBwLlxuICovXG5cbmltcG9ydCB7IERPQ1VNRU5UIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IERlc3Ryb3lSZWYsIEluamVjdGFibGUsIFNpZ25hbCwgY29tcHV0ZWQsIGluamVjdCwgc2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyB0YWtlVW50aWxEZXN0cm95ZWQgfSBmcm9tICdAYW5ndWxhci9jb3JlL3J4anMtaW50ZXJvcCc7XG5pbXBvcnQgeyBTd1VwZGF0ZSB9IGZyb20gJ0Bhbmd1bGFyL3NlcnZpY2Utd29ya2VyJztcbmltcG9ydCB7IGZpbHRlciwgaW50ZXJ2YWwgfSBmcm9tICdyeGpzJztcblxuaW1wb3J0IHsgQXBwQ29uZmlnU2VydmljZSB9IGZyb20gJy4uL2FwcC1jb25maWcvYXBwLWNvbmZpZy5zZXJ2aWNlJztcbmltcG9ydCB7IFZBTFRFQ0hfQVBQX1ZFUlNJT04gfSBmcm9tICcuL2NvbmZpZyc7XG5pbXBvcnQgeyBBcHBWZXJzaW9uU3RhdHVzLCBERUZBVUxUX0FQUF9WRVJTSU9OX1NFUlZJQ0VfQ09ORklHIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogQ29tcGFyYSBkb3MgdmVyc2lvbmVzIHNlbXZlci4gUmV0b3JuYSB0cnVlIHNpIGBhYCBlcyBlc3RyaWN0YW1lbnRlIG1lbm9yXG4gKiBxdWUgYGJgLiBUb2xlcmFudGUgYSBsb25naXR1ZGVzIGRpc3RpbnRhcyB5IGEgcGFydGVzIG5vIG51bcOpcmljYXMuXG4gKi9cbmZ1bmN0aW9uIGlzVmVyc2lvbkxvd2VyKGE6IHN0cmluZywgYjogc3RyaW5nKTogYm9vbGVhbiB7XG4gIGNvbnN0IHBhID0gYS5zcGxpdCgnLicpLm1hcChuID0+IE51bWJlcihuKSB8fCAwKTtcbiAgY29uc3QgcGIgPSBiLnNwbGl0KCcuJykubWFwKG4gPT4gTnVtYmVyKG4pIHx8IDApO1xuICBjb25zdCBsZW4gPSBNYXRoLm1heChwYS5sZW5ndGgsIHBiLmxlbmd0aCk7XG5cbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBsZW47IGkrKykge1xuICAgIGNvbnN0IHggPSBwYVtpXSB8fCAwO1xuICAgIGNvbnN0IHkgPSBwYltpXSB8fCAwO1xuICAgIGlmICh4IDwgeSkgcmV0dXJuIHRydWU7XG4gICAgaWYgKHggPiB5KSByZXR1cm4gZmFsc2U7XG4gIH1cblxuICByZXR1cm4gZmFsc2U7XG59XG5cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgQXBwVmVyc2lvblNlcnZpY2Uge1xuICAvKiogU3dVcGRhdGUgc29sbyBleGlzdGUgc2kgZWwgU1cgZXN0w6EgcmVnaXN0cmFkbyAocHJvZCBidWlsZCkuICovXG4gIHByaXZhdGUgcmVhZG9ubHkgc3dVcGRhdGUgPSBpbmplY3QoU3dVcGRhdGUsIHsgb3B0aW9uYWw6IHRydWUgfSk7XG4gIC8qKiBBcHBDb25maWdTZXJ2aWNlIGVzIG9wY2lvbmFsOiBsYSBhcHAgcHVlZGUgbm8gd2lyZWFybG8uICovXG4gIHByaXZhdGUgcmVhZG9ubHkgYXBwQ29uZmlnID0gaW5qZWN0KEFwcENvbmZpZ1NlcnZpY2UsIHsgb3B0aW9uYWw6IHRydWUgfSk7XG4gIHByaXZhdGUgcmVhZG9ubHkgc2VydmljZUNvbmZpZyA9IGluamVjdChWQUxURUNIX0FQUF9WRVJTSU9OLCB7XG4gICAgb3B0aW9uYWw6IHRydWUsXG4gIH0pO1xuICBwcml2YXRlIHJlYWRvbmx5IGRvY3VtZW50ID0gaW5qZWN0KERPQ1VNRU5UKTtcbiAgcHJpdmF0ZSByZWFkb25seSBkZXN0cm95UmVmID0gaW5qZWN0KERlc3Ryb3lSZWYpO1xuXG4gIC8qKiBUcnVlIGN1YW5kbyBlbCBTVyByZXBvcnTDsyB1biBidW5kbGUgbnVldm8gbGlzdG8gKGBWRVJTSU9OX1JFQURZYCkuICovXG4gIHByaXZhdGUgcmVhZG9ubHkgc3dVcGRhdGVSZWFkeSA9IHNpZ25hbChmYWxzZSk7XG5cbiAgLyoqXG4gICAqIFRydWUgY3VhbmRvIGVsIHVzdWFyaW8gZGVzY2FydMOzIGVsIGJhbm5lciBkZSBhY3R1YWxpemFjacOzbiBvcGNpb25hbC5cbiAgICogRWwgYmFubmVyIG9ibGlnYXRvcmlvIChgdXBkYXRlLXJlcXVpcmVkYCkgaWdub3JhIGVzdGUgZmxhZy5cbiAgICovXG4gIHByaXZhdGUgcmVhZG9ubHkgX2Rpc21pc3NlZCA9IHNpZ25hbChmYWxzZSk7XG4gIHJlYWRvbmx5IGRpc21pc3NlZDogU2lnbmFsPGJvb2xlYW4+ID0gdGhpcy5fZGlzbWlzc2VkLmFzUmVhZG9ubHkoKTtcblxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gU0lHTkFMUyBQw5pCTElDT1NcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqXG4gICAqIFZlcnNpw7NuIGxvY2FsIGRlbCBidWlsZCBhY3R1YWwgZGUgbGEgYXBwLlxuICAgKiBQcm92aWVuZSBkZWwgcHJvdmlkZXIgYHByb3ZpZGVWYWx0ZWNoQXBwVmVyc2lvbih7IGN1cnJlbnRWZXJzaW9uIH0pYC5cbiAgICovXG4gIHJlYWRvbmx5IGN1cnJlbnRWZXJzaW9uOiBTaWduYWw8c3RyaW5nPiA9IGNvbXB1dGVkKFxuICAgICgpID0+IHRoaXMuc2VydmljZUNvbmZpZz8uY3VycmVudFZlcnNpb24gPz8gJzAuMC4wJ1xuICApO1xuXG4gIC8qKlxuICAgKiDDmmx0aW1hIHZlcnNpw7NuIHB1YmxpY2FkYSBzZWfDum4gZWwgY29uZmlnIHJlbW90by5cbiAgICogYG51bGxgIHNpIGBBcHBDb25maWdTZXJ2aWNlYCBubyBlc3TDoSBkaXNwb25pYmxlIG8gYcO6biBubyBjYXJnw7MuXG4gICAqL1xuICByZWFkb25seSBsYXRlc3RWZXJzaW9uOiBTaWduYWw8c3RyaW5nIHwgbnVsbD4gPSBjb21wdXRlZChcbiAgICAoKSA9PiB0aGlzLmFwcENvbmZpZz8ucmVtb3RlVmVyc2lvbigpID8/IG51bGxcbiAgKTtcblxuICAvKipcbiAgICogRXN0YWRvIGRlIGxhIHZlcnNpw7NuOlxuICAgKiAtIGB1cGRhdGUtcmVxdWlyZWRgIOKAlCBsYSB2ZXJzacOzbiBsb2NhbCBxdWVkw7MgcG9yIGRlYmFqbyBkZSBgbWluVmVyc2lvbmBcbiAgICogICBkZWwgY29uZmlnIHJlbW90by4gSGFyZCBnYXRlLiBTb2xvIHNpIGBBcHBDb25maWdTZXJ2aWNlYCB0aWVuZSBkYXRvcy5cbiAgICogLSBgdXBkYXRlLWF2YWlsYWJsZWAg4oCUIGVsIFNXIGRlc2NhcmfDsyB1biBidW5kbGUgbnVldm8gKGBWRVJTSU9OX1JFQURZYClcbiAgICogICBvIGBBcHBDb25maWdTZXJ2aWNlLmhhc1VwZGF0ZWAgZXMgdHJ1ZS5cbiAgICogLSBgdXAtdG8tZGF0ZWAg4oCUIGVuIGN1YWxxdWllciBvdHJvIGNhc28uXG4gICAqL1xuICByZWFkb25seSBzdGF0dXM6IFNpZ25hbDxBcHBWZXJzaW9uU3RhdHVzPiA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICAvLyAxLiBIYXJkIGdhdGUg4oCUIG1pblZlcnNpb24gZGVsIGNvbmZpZyByZW1vdG8uXG4gICAgaWYgKHRoaXMuYXBwQ29uZmlnKSB7XG4gICAgICBjb25zdCBtaW5WZXJzaW9uID0gdGhpcy5hcHBDb25maWcuYXBwQ29uZmlnKCk/Lm1pblZlcnNpb247XG4gICAgICBpZiAobWluVmVyc2lvbiAmJiBpc1ZlcnNpb25Mb3dlcih0aGlzLmN1cnJlbnRWZXJzaW9uKCksIG1pblZlcnNpb24pKSB7XG4gICAgICAgIHJldHVybiAndXBkYXRlLXJlcXVpcmVkJztcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyAyLiBVcGRhdGUgb3BjaW9uYWwg4oCUIFNXIGJ1bmRsZSBsaXN0byBvIGNvbmZpZyByZW1vdG8gbcOhcyBudWV2by5cbiAgICBpZiAodGhpcy5zd1VwZGF0ZVJlYWR5KCkgfHwgdGhpcy5hcHBDb25maWc/Lmhhc1VwZGF0ZSgpKSB7XG4gICAgICByZXR1cm4gJ3VwZGF0ZS1hdmFpbGFibGUnO1xuICAgIH1cblxuICAgIHJldHVybiAndXAtdG8tZGF0ZSc7XG4gIH0pO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMud2lyZVNlcnZpY2VXb3JrZXIoKTtcbiAgfVxuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICAvLyBNw4lUT0RPUyBQw5pCTElDT1NcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqXG4gICAqIEFwbGljYSBsYSBhY3R1YWxpemFjacOzbiBwZW5kaWVudGUuXG4gICAqXG4gICAqIFNpIGVsIFNXIGVzdMOhIGhhYmlsaXRhZG8sIGFjdGl2YSBsYSB2ZXJzacOzbiBkZXNjYXJnYWRhIHkgcmVjYXJnYTsgc2kgbm8sXG4gICAqIHNvbG8gcmVjYXJnYSBsYSBww6FnaW5hIChsYSB3ZWIgb2J0aWVuZSBlbCBidW5kbGUgbnVldm8gZGVsIHNlcnZpZG9yKS5cbiAgICovXG4gIGFzeW5jIGFwcGx5VXBkYXRlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIHRyeSB7XG4gICAgICBpZiAodGhpcy5zd1VwZGF0ZT8uaXNFbmFibGVkKSB7XG4gICAgICAgIGF3YWl0IHRoaXMuc3dVcGRhdGUuYWN0aXZhdGVVcGRhdGUoKTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgIGNvbnNvbGUud2FybignW0FwcFZlcnNpb25TZXJ2aWNlXSBhY3RpdmF0ZVVwZGF0ZSBmYWlsZWQ6JywgZXJyKTtcbiAgICB9IGZpbmFsbHkge1xuICAgICAgdGhpcy5kb2N1bWVudC5kZWZhdWx0Vmlldz8ubG9jYXRpb24ucmVsb2FkKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIERlc2NhcnRhIGVsIGJhbm5lciBkZSBhY3R1YWxpemFjacOzbiBvcGNpb25hbC5cbiAgICogTm8gdGllbmUgZWZlY3RvIHNvYnJlIGVsIGVzdGFkbyBgdXBkYXRlLXJlcXVpcmVkYCAoaGFyZCBnYXRlKS5cbiAgICovXG4gIGRpc21pc3MoKTogdm9pZCB7XG4gICAgdGhpcy5fZGlzbWlzc2VkLnNldCh0cnVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBGdWVyemEgdW4gY2hlcXVlbyBkZSBhY3R1YWxpemFjacOzbiBkZWwgc2VydmljZSB3b3JrZXIuXG4gICAqIE5vLW9wIHNpIGVsIFNXIG5vIGVzdMOhIGhhYmlsaXRhZG8uXG4gICAqL1xuICBhc3luYyBjaGVja05vdygpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBpZiAoIXRoaXMuc3dVcGRhdGU/LmlzRW5hYmxlZCkgcmV0dXJuO1xuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0aGlzLnN3VXBkYXRlLmNoZWNrRm9yVXBkYXRlKCk7XG4gICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICBjb25zb2xlLndhcm4oJ1tBcHBWZXJzaW9uU2VydmljZV0gY2hlY2tGb3JVcGRhdGUgZmFpbGVkOicsIGVycik7XG4gICAgfVxuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIFBSSVZBRE9cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqXG4gICAqIFN1c2NyaWJlIGB2ZXJzaW9uVXBkYXRlc2AsIHByb2dyYW1hIGNoZXF1ZW9zIHBlcmnDs2RpY29zIHkgcmVhY2Npb25hIGFsXG4gICAqIHZvbHZlciBsYSBhcHAgYSBmb3JlZ3JvdW5kLiBTb2xvIGFjdMO6YSBzaSBlbCBTVyBlc3TDoSBoYWJpbGl0YWRvLlxuICAgKi9cbiAgcHJpdmF0ZSB3aXJlU2VydmljZVdvcmtlcigpOiB2b2lkIHtcbiAgICBpZiAoIXRoaXMuc3dVcGRhdGU/LmlzRW5hYmxlZCkgcmV0dXJuO1xuXG4gICAgLy8gQnVuZGxlIG51ZXZvIGRlc2NhcmdhZG8geSBsaXN0by5cbiAgICB0aGlzLnN3VXBkYXRlLnZlcnNpb25VcGRhdGVzXG4gICAgICAucGlwZShcbiAgICAgICAgZmlsdGVyKGV2dCA9PiBldnQudHlwZSA9PT0gJ1ZFUlNJT05fUkVBRFknKSxcbiAgICAgICAgdGFrZVVudGlsRGVzdHJveWVkKHRoaXMuZGVzdHJveVJlZilcbiAgICAgIClcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4gdGhpcy5zd1VwZGF0ZVJlYWR5LnNldCh0cnVlKSk7XG5cbiAgICAvLyBDaGVxdWVvIHBlcmnDs2RpY28uXG4gICAgY29uc3QgaW50ZXJ2YWxNcyA9XG4gICAgICB0aGlzLnNlcnZpY2VDb25maWc/LmNoZWNrSW50ZXJ2YWxNcyA/PyBERUZBVUxUX0FQUF9WRVJTSU9OX1NFUlZJQ0VfQ09ORklHLmNoZWNrSW50ZXJ2YWxNcztcblxuICAgIGludGVydmFsKGludGVydmFsTXMpXG4gICAgICAucGlwZSh0YWtlVW50aWxEZXN0cm95ZWQodGhpcy5kZXN0cm95UmVmKSlcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4gdm9pZCB0aGlzLmNoZWNrTm93KCkpO1xuXG4gICAgLy8gQ2hlcXVlbyBhbCB2b2x2ZXIgbGEgYXBwIGEgZm9yZWdyb3VuZC5cbiAgICBjb25zdCBvblZpc2libGUgPSAoKTogdm9pZCA9PiB7XG4gICAgICBpZiAodGhpcy5kb2N1bWVudC52aXNpYmlsaXR5U3RhdGUgPT09ICd2aXNpYmxlJykge1xuICAgICAgICB2b2lkIHRoaXMuY2hlY2tOb3coKTtcbiAgICAgIH1cbiAgICB9O1xuICAgIHRoaXMuZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcigndmlzaWJpbGl0eWNoYW5nZScsIG9uVmlzaWJsZSk7XG4gICAgdGhpcy5kZXN0cm95UmVmLm9uRGVzdHJveSgoKSA9PiB7XG4gICAgICB0aGlzLmRvY3VtZW50LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ3Zpc2liaWxpdHljaGFuZ2UnLCBvblZpc2libGUpO1xuICAgIH0pO1xuXG4gICAgLy8gQ2hlcXVlbyBpbmljaWFsLlxuICAgIHZvaWQgdGhpcy5jaGVja05vdygpO1xuICB9XG59XG4iXX0=