valtech-components 2.0.451 → 2.0.453

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 (61) hide show
  1. package/esm2022/lib/components/organisms/tabbed-content/tabbed-content.component.mjs +170 -0
  2. package/esm2022/lib/components/organisms/tabbed-content/types.mjs +2 -0
  3. package/esm2022/lib/components/templates/page-content/page-content.component.mjs +11 -11
  4. package/esm2022/lib/components/templates/page-template/page-template.component.mjs +3 -5
  5. package/esm2022/lib/services/auth/auth-state.service.mjs +173 -0
  6. package/esm2022/lib/services/auth/auth.service.mjs +454 -0
  7. package/esm2022/lib/services/auth/config.mjs +76 -0
  8. package/esm2022/lib/services/auth/guards.mjs +194 -0
  9. package/esm2022/lib/services/auth/index.mjs +70 -0
  10. package/esm2022/lib/services/auth/interceptor.mjs +98 -0
  11. package/esm2022/lib/services/auth/storage.service.mjs +141 -0
  12. package/esm2022/lib/services/auth/sync.service.mjs +149 -0
  13. package/esm2022/lib/services/auth/token.service.mjs +113 -0
  14. package/esm2022/lib/services/auth/types.mjs +29 -0
  15. package/esm2022/lib/services/firebase/config.mjs +108 -0
  16. package/esm2022/lib/services/firebase/firebase.service.mjs +288 -0
  17. package/esm2022/lib/services/firebase/firestore-collection.mjs +254 -0
  18. package/esm2022/lib/services/firebase/firestore.service.mjs +509 -0
  19. package/esm2022/lib/services/firebase/index.mjs +49 -0
  20. package/esm2022/lib/services/firebase/messaging.service.mjs +512 -0
  21. package/esm2022/lib/services/firebase/shared-config.mjs +138 -0
  22. package/esm2022/lib/services/firebase/storage.service.mjs +422 -0
  23. package/esm2022/lib/services/firebase/types.mjs +8 -0
  24. package/esm2022/lib/services/firebase/utils/path-builder.mjs +195 -0
  25. package/esm2022/lib/services/firebase/utils/query-builder.mjs +302 -0
  26. package/esm2022/lib/services/link-processor.service.mjs +61 -43
  27. package/esm2022/lib/services/modal/modal.service.mjs +8 -9
  28. package/esm2022/lib/services/navigation.service.mjs +11 -11
  29. package/esm2022/public-api.mjs +23 -4
  30. package/fesm2022/valtech-components.mjs +4599 -102
  31. package/fesm2022/valtech-components.mjs.map +1 -1
  32. package/lib/components/organisms/tabbed-content/tabbed-content.component.d.ts +65 -0
  33. package/lib/components/organisms/tabbed-content/types.d.ts +53 -0
  34. package/lib/components/templates/page-content/page-content.component.d.ts +3 -0
  35. package/lib/services/auth/auth-state.service.d.ts +85 -0
  36. package/lib/services/auth/auth.service.d.ts +146 -0
  37. package/lib/services/auth/config.d.ts +38 -0
  38. package/lib/services/auth/guards.d.ts +123 -0
  39. package/lib/services/auth/index.d.ts +63 -0
  40. package/lib/services/auth/interceptor.d.ts +22 -0
  41. package/lib/services/auth/storage.service.d.ts +48 -0
  42. package/lib/services/auth/sync.service.d.ts +49 -0
  43. package/lib/services/auth/token.service.d.ts +51 -0
  44. package/lib/services/auth/types.d.ts +315 -0
  45. package/lib/services/firebase/config.d.ts +49 -0
  46. package/lib/services/firebase/firebase.service.d.ts +140 -0
  47. package/lib/services/firebase/firestore-collection.d.ts +175 -0
  48. package/lib/services/firebase/firestore.service.d.ts +304 -0
  49. package/lib/services/firebase/index.d.ts +39 -0
  50. package/lib/services/firebase/messaging.service.d.ts +263 -0
  51. package/lib/services/firebase/shared-config.d.ts +126 -0
  52. package/lib/services/firebase/storage.service.d.ts +206 -0
  53. package/lib/services/firebase/types.d.ts +281 -0
  54. package/lib/services/firebase/utils/path-builder.d.ts +132 -0
  55. package/lib/services/firebase/utils/query-builder.d.ts +210 -0
  56. package/lib/services/modal/modal.service.d.ts +2 -0
  57. package/lib/services/navigation.service.d.ts +4 -4
  58. package/package.json +3 -1
  59. package/public-api.d.ts +9 -0
  60. package/fesm2022/valtech-components-simple-modal-content.component-DQhEgUmS.mjs +0 -136
  61. package/fesm2022/valtech-components-simple-modal-content.component-DQhEgUmS.mjs.map +0 -1
@@ -0,0 +1,170 @@
1
+ import { Component, Input, Output, EventEmitter, signal, computed, ChangeDetectionStrategy, } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { SegmentControlComponent } from '../../molecules/segment-control/segment-control.component';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ /**
7
+ * val-tabbed-content
8
+ *
9
+ * A container component that combines segment navigation with dynamic content panels.
10
+ * Uses segment-control internally for tab navigation and renders the associated
11
+ * template for the active tab.
12
+ *
13
+ * @example Basic usage with templates
14
+ * ```html
15
+ * <ng-template #catalogTemplate>
16
+ * <div>Catalog Content</div>
17
+ * </ng-template>
18
+ * <ng-template #settingsTemplate>
19
+ * <div>Settings Content</div>
20
+ * </ng-template>
21
+ *
22
+ * <val-tabbed-content [props]="{
23
+ * tabs: [
24
+ * { value: 'catalog', label: 'Catalog', icon: 'layers-outline', template: catalogTemplate },
25
+ * { value: 'settings', label: 'Settings', icon: 'settings-outline', template: settingsTemplate }
26
+ * ],
27
+ * selectedTab: 'catalog',
28
+ * scrollable: true,
29
+ * animated: true
30
+ * }" (tabChange)="onTabChange($event)"></val-tabbed-content>
31
+ * ```
32
+ *
33
+ * @input props: TabbedContentMetadata - Configuration for the tabbed content
34
+ * @output tabChange: string - Emits the selected tab value when changed
35
+ */
36
+ export class TabbedContentComponent {
37
+ constructor() {
38
+ /**
39
+ * Emits when the active tab changes.
40
+ */
41
+ this.tabChange = new EventEmitter();
42
+ /** Currently selected tab value */
43
+ this.selectedValue = signal('');
44
+ /** Whether a transition is in progress */
45
+ this.isTransitioning = signal(false);
46
+ /** Computed animation duration string */
47
+ this.animationDuration = computed(() => `${this.props.animationDuration || 300}ms`);
48
+ /** Computed segment control props derived from tabs config */
49
+ this.segmentControlProps = computed(() => {
50
+ const options = this.props.tabs.map(tab => ({
51
+ value: tab.value,
52
+ label: tab.label,
53
+ icon: tab.icon,
54
+ disabled: tab.disabled,
55
+ layout: tab.layout || 'icon-top',
56
+ }));
57
+ return {
58
+ options,
59
+ value: this.selectedValue(),
60
+ color: this.props.color || 'primary',
61
+ scrollable: this.props.scrollable ?? false,
62
+ swipeGesture: this.props.swipeGesture ?? true,
63
+ mode: this.props.mode,
64
+ };
65
+ });
66
+ /** Computed active tab object */
67
+ this.activeTab = computed(() => {
68
+ return this.props.tabs.find(tab => tab.value === this.selectedValue());
69
+ });
70
+ }
71
+ ngOnInit() {
72
+ // Set initial selected tab
73
+ const initialValue = this.props.selectedTab || this.props.tabs[0]?.value || '';
74
+ this.selectedValue.set(initialValue);
75
+ }
76
+ /**
77
+ * Handles segment change events.
78
+ */
79
+ onSegmentChange(value) {
80
+ if (value === this.selectedValue()) {
81
+ return;
82
+ }
83
+ // Trigger transition animation
84
+ if (this.props.animated !== false) {
85
+ this.isTransitioning.set(true);
86
+ // Reset transition state after animation completes
87
+ setTimeout(() => {
88
+ this.selectedValue.set(value);
89
+ this.isTransitioning.set(false);
90
+ this.tabChange.emit(value);
91
+ }, (this.props.animationDuration || 300) / 2);
92
+ }
93
+ else {
94
+ this.selectedValue.set(value);
95
+ this.tabChange.emit(value);
96
+ }
97
+ }
98
+ /**
99
+ * Creates the context object for the template outlet.
100
+ */
101
+ getTemplateContext(tab) {
102
+ const index = this.props.tabs.findIndex(t => t.value === tab.value);
103
+ return {
104
+ $implicit: tab.value,
105
+ tab,
106
+ index,
107
+ };
108
+ }
109
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TabbedContentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
110
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: TabbedContentComponent, isStandalone: true, selector: "val-tabbed-content", inputs: { props: "props" }, outputs: { tabChange: "tabChange" }, ngImport: i0, template: `
111
+ <div
112
+ class="tabbed-content"
113
+ [class]="props.cssClass"
114
+ [style.--animation-duration]="animationDuration()"
115
+ >
116
+ <!-- Segment Control Navigation -->
117
+ <val-segment-control
118
+ [props]="segmentControlProps()"
119
+ (segmentChange)="onSegmentChange($event)"
120
+ ></val-segment-control>
121
+
122
+ <!-- Tab Content Panel -->
123
+ <div
124
+ class="tabbed-content__panel"
125
+ [class.tabbed-content__panel--animated]="props.animated !== false"
126
+ [class.tabbed-content__panel--transitioning]="isTransitioning()"
127
+ >
128
+ @if (activeTab(); as tab) {
129
+ <ng-container
130
+ *ngTemplateOutlet="tab.template; context: getTemplateContext(tab)"
131
+ ></ng-container>
132
+ }
133
+ </div>
134
+ </div>
135
+ `, isInline: true, styles: [".tabbed-content{display:flex;flex-direction:column;width:100%}.tabbed-content val-segment-control{margin-bottom:1rem}.tabbed-content val-segment-control ion-segment{--background: var(--ion-color-light);border-radius:12px;padding:4px}.tabbed-content__panel{width:100%;min-height:100px}.tabbed-content__panel--animated{animation:fadeIn var(--animation-duration, .3s) ease-out}.tabbed-content__panel--transitioning{opacity:0;animation:fadeOut calc(var(--animation-duration, .3s) / 2) ease-out forwards}@keyframes fadeIn{0%{opacity:0;transform:translateY(10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeOut{0%{opacity:1;transform:translateY(0)}to{opacity:0;transform:translateY(-10px)}}:host-context(.dark) .tabbed-content val-segment-control ion-segment,:host-context([data-theme=dark]) .tabbed-content val-segment-control ion-segment{--background: var(--ion-color-dark-tint)}@media (max-width: 576px){.tabbed-content val-segment-control ion-segment{padding:2px}.tabbed-content val-segment-control ion-segment-button{min-width:auto;padding:8px 12px}.tabbed-content val-segment-control ion-segment-button ion-label{font-size:.75rem}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: SegmentControlComponent, selector: "val-segment-control", inputs: ["props"], outputs: ["segmentChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
136
+ }
137
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TabbedContentComponent, decorators: [{
138
+ type: Component,
139
+ args: [{ selector: 'val-tabbed-content', standalone: true, imports: [CommonModule, SegmentControlComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `
140
+ <div
141
+ class="tabbed-content"
142
+ [class]="props.cssClass"
143
+ [style.--animation-duration]="animationDuration()"
144
+ >
145
+ <!-- Segment Control Navigation -->
146
+ <val-segment-control
147
+ [props]="segmentControlProps()"
148
+ (segmentChange)="onSegmentChange($event)"
149
+ ></val-segment-control>
150
+
151
+ <!-- Tab Content Panel -->
152
+ <div
153
+ class="tabbed-content__panel"
154
+ [class.tabbed-content__panel--animated]="props.animated !== false"
155
+ [class.tabbed-content__panel--transitioning]="isTransitioning()"
156
+ >
157
+ @if (activeTab(); as tab) {
158
+ <ng-container
159
+ *ngTemplateOutlet="tab.template; context: getTemplateContext(tab)"
160
+ ></ng-container>
161
+ }
162
+ </div>
163
+ </div>
164
+ `, styles: [".tabbed-content{display:flex;flex-direction:column;width:100%}.tabbed-content val-segment-control{margin-bottom:1rem}.tabbed-content val-segment-control ion-segment{--background: var(--ion-color-light);border-radius:12px;padding:4px}.tabbed-content__panel{width:100%;min-height:100px}.tabbed-content__panel--animated{animation:fadeIn var(--animation-duration, .3s) ease-out}.tabbed-content__panel--transitioning{opacity:0;animation:fadeOut calc(var(--animation-duration, .3s) / 2) ease-out forwards}@keyframes fadeIn{0%{opacity:0;transform:translateY(10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeOut{0%{opacity:1;transform:translateY(0)}to{opacity:0;transform:translateY(-10px)}}:host-context(.dark) .tabbed-content val-segment-control ion-segment,:host-context([data-theme=dark]) .tabbed-content val-segment-control ion-segment{--background: var(--ion-color-dark-tint)}@media (max-width: 576px){.tabbed-content val-segment-control ion-segment{padding:2px}.tabbed-content val-segment-control ion-segment-button{min-width:auto;padding:8px 12px}.tabbed-content val-segment-control ion-segment-button ion-label{font-size:.75rem}}\n"] }]
165
+ }], propDecorators: { props: [{
166
+ type: Input
167
+ }], tabChange: [{
168
+ type: Output
169
+ }] } });
170
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabbed-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/tabbed-content/tabbed-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  signal,\n  computed,\n  ChangeDetectionStrategy,\n  OnInit,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { SegmentControlComponent } from '../../molecules/segment-control/segment-control.component';\nimport { SegmentControlMetadata, SegmentOption } from '../../molecules/segment-control/types';\nimport { TabbedContentMetadata, TabbedContentTab, TabbedContentContext } from './types';\n\n/**\n * val-tabbed-content\n *\n * A container component that combines segment navigation with dynamic content panels.\n * Uses segment-control internally for tab navigation and renders the associated\n * template for the active tab.\n *\n * @example Basic usage with templates\n * ```html\n * <ng-template #catalogTemplate>\n *   <div>Catalog Content</div>\n * </ng-template>\n * <ng-template #settingsTemplate>\n *   <div>Settings Content</div>\n * </ng-template>\n *\n * <val-tabbed-content [props]=\"{\n *   tabs: [\n *     { value: 'catalog', label: 'Catalog', icon: 'layers-outline', template: catalogTemplate },\n *     { value: 'settings', label: 'Settings', icon: 'settings-outline', template: settingsTemplate }\n *   ],\n *   selectedTab: 'catalog',\n *   scrollable: true,\n *   animated: true\n * }\" (tabChange)=\"onTabChange($event)\"></val-tabbed-content>\n * ```\n *\n * @input props: TabbedContentMetadata - Configuration for the tabbed content\n * @output tabChange: string - Emits the selected tab value when changed\n */\n@Component({\n  selector: 'val-tabbed-content',\n  standalone: true,\n  imports: [CommonModule, SegmentControlComponent],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `\n    <div\n      class=\"tabbed-content\"\n      [class]=\"props.cssClass\"\n      [style.--animation-duration]=\"animationDuration()\"\n    >\n      <!-- Segment Control Navigation -->\n      <val-segment-control\n        [props]=\"segmentControlProps()\"\n        (segmentChange)=\"onSegmentChange($event)\"\n      ></val-segment-control>\n\n      <!-- Tab Content Panel -->\n      <div\n        class=\"tabbed-content__panel\"\n        [class.tabbed-content__panel--animated]=\"props.animated !== false\"\n        [class.tabbed-content__panel--transitioning]=\"isTransitioning()\"\n      >\n        @if (activeTab(); as tab) {\n          <ng-container\n            *ngTemplateOutlet=\"tab.template; context: getTemplateContext(tab)\"\n          ></ng-container>\n        }\n      </div>\n    </div>\n  `,\n  styleUrls: ['./tabbed-content.component.scss'],\n})\nexport class TabbedContentComponent implements OnInit {\n  /**\n   * Configuration object for the tabbed content.\n   */\n  @Input() props!: TabbedContentMetadata;\n\n  /**\n   * Emits when the active tab changes.\n   */\n  @Output() tabChange = new EventEmitter<string>();\n\n  /** Currently selected tab value */\n  private selectedValue = signal<string>('');\n\n  /** Whether a transition is in progress */\n  isTransitioning = signal<boolean>(false);\n\n  /** Computed animation duration string */\n  animationDuration = computed(() => `${this.props.animationDuration || 300}ms`);\n\n  /** Computed segment control props derived from tabs config */\n  segmentControlProps = computed<SegmentControlMetadata>(() => {\n    const options: SegmentOption[] = this.props.tabs.map(tab => ({\n      value: tab.value,\n      label: tab.label,\n      icon: tab.icon,\n      disabled: tab.disabled,\n      layout: tab.layout || 'icon-top',\n    }));\n\n    return {\n      options,\n      value: this.selectedValue(),\n      color: this.props.color || 'primary',\n      scrollable: this.props.scrollable ?? false,\n      swipeGesture: this.props.swipeGesture ?? true,\n      mode: this.props.mode,\n    };\n  });\n\n  /** Computed active tab object */\n  activeTab = computed<TabbedContentTab | undefined>(() => {\n    return this.props.tabs.find(tab => tab.value === this.selectedValue());\n  });\n\n  ngOnInit(): void {\n    // Set initial selected tab\n    const initialValue = this.props.selectedTab || this.props.tabs[0]?.value || '';\n    this.selectedValue.set(initialValue);\n  }\n\n  /**\n   * Handles segment change events.\n   */\n  onSegmentChange(value: string): void {\n    if (value === this.selectedValue()) {\n      return;\n    }\n\n    // Trigger transition animation\n    if (this.props.animated !== false) {\n      this.isTransitioning.set(true);\n\n      // Reset transition state after animation completes\n      setTimeout(() => {\n        this.selectedValue.set(value);\n        this.isTransitioning.set(false);\n        this.tabChange.emit(value);\n      }, (this.props.animationDuration || 300) / 2);\n    } else {\n      this.selectedValue.set(value);\n      this.tabChange.emit(value);\n    }\n  }\n\n  /**\n   * Creates the context object for the template outlet.\n   */\n  getTemplateContext(tab: TabbedContentTab): TabbedContentContext {\n    const index = this.props.tabs.findIndex(t => t.value === tab.value);\n    return {\n      $implicit: tab.value,\n      tab,\n      index,\n    };\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL3RhYmJlZC1jb250ZW50L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUZW1wbGF0ZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29sb3IgfSBmcm9tICdAaW9uaWMvY29yZSc7XG5cbi8qKlxuICogQ29udGV4dCBwYXNzZWQgdG8gZWFjaCB0YWIncyB0ZW1wbGF0ZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBUYWJiZWRDb250ZW50Q29udGV4dCB7XG4gIC8qKiBUaGUgdmFsdWUgb2YgdGhlIGFjdGl2ZSB0YWIgKi9cbiAgJGltcGxpY2l0OiBzdHJpbmc7XG4gIC8qKiBUaGUgZnVsbCB0YWIgY29uZmlndXJhdGlvbiAqL1xuICB0YWI6IFRhYmJlZENvbnRlbnRUYWI7XG4gIC8qKiBJbmRleCBvZiB0aGUgdGFiICovXG4gIGluZGV4OiBudW1iZXI7XG59XG5cbi8qKlxuICogQ29uZmlndXJhdGlvbiBmb3IgYSBzaW5nbGUgdGFiLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRhYmJlZENvbnRlbnRUYWIge1xuICAvKiogVW5pcXVlIGlkZW50aWZpZXIgZm9yIHRoZSB0YWIgKi9cbiAgdmFsdWU6IHN0cmluZztcbiAgLyoqIERpc3BsYXkgbGFiZWwgZm9yIHRoZSB0YWIgYnV0dG9uICovXG4gIGxhYmVsPzogc3RyaW5nO1xuICAvKiogSWNvbiBuYW1lIChJb25pY29ucykgKi9cbiAgaWNvbj86IHN0cmluZztcbiAgLyoqIFdoZXRoZXIgdGhlIHRhYiBpcyBkaXNhYmxlZCAqL1xuICBkaXNhYmxlZD86IGJvb2xlYW47XG4gIC8qKiBMYXlvdXQgZGlyZWN0aW9uIGZvciBpY29uIGFuZCBsYWJlbCAqL1xuICBsYXlvdXQ/OiAnaWNvbi1zdGFydCcgfCAnaWNvbi1lbmQnIHwgJ2ljb24tdG9wJyB8ICdpY29uLWJvdHRvbSc7XG4gIC8qKiBUZW1wbGF0ZSB0byByZW5kZXIgd2hlbiB0aGlzIHRhYiBpcyBhY3RpdmUgKi9cbiAgdGVtcGxhdGU6IFRlbXBsYXRlUmVmPFRhYmJlZENvbnRlbnRDb250ZXh0Pjtcbn1cblxuLyoqXG4gKiBNZXRhZGF0YSBmb3IgdGhlIHRhYmJlZC1jb250ZW50IGNvbXBvbmVudC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBUYWJiZWRDb250ZW50TWV0YWRhdGEge1xuICAvKiogQXJyYXkgb2YgdGFiIGNvbmZpZ3VyYXRpb25zICovXG4gIHRhYnM6IFRhYmJlZENvbnRlbnRUYWJbXTtcbiAgLyoqIEluaXRpYWxseSBzZWxlY3RlZCB0YWIgdmFsdWUgKGRlZmF1bHRzIHRvIGZpcnN0IHRhYikgKi9cbiAgc2VsZWN0ZWRUYWI/OiBzdHJpbmc7XG4gIC8qKiBDb2xvciB0aGVtZSBmb3IgdGhlIHNlZ21lbnQgY29udHJvbCAqL1xuICBjb2xvcj86IENvbG9yO1xuICAvKiogQWxsb3cgaG9yaXpvbnRhbCBzY3JvbGxpbmcgZm9yIG1hbnkgdGFicyAqL1xuICBzY3JvbGxhYmxlPzogYm9vbGVhbjtcbiAgLyoqIEVuYWJsZSBzd2lwZSBnZXN0dXJlIHRvIGNoYW5nZSB0YWJzIChpT1Mgb25seSkgKi9cbiAgc3dpcGVHZXN0dXJlPzogYm9vbGVhbjtcbiAgLyoqIFZpc3VhbCBtb2RlIHN0eWxlICovXG4gIG1vZGU/OiAnaW9zJyB8ICdtZCc7XG4gIC8qKiBFbmFibGUgZmFkZSBhbmltYXRpb24gb24gdGFiIGNoYW5nZSAqL1xuICBhbmltYXRlZD86IGJvb2xlYW47XG4gIC8qKiBBbmltYXRpb24gZHVyYXRpb24gaW4gbWlsbGlzZWNvbmRzICovXG4gIGFuaW1hdGlvbkR1cmF0aW9uPzogbnVtYmVyO1xuICAvKiogQWRkaXRpb25hbCBDU1MgY2xhc3MgZm9yIHRoZSBjb250YWluZXIgKi9cbiAgY3NzQ2xhc3M/OiBzdHJpbmc7XG59XG4iXX0=
@@ -1,12 +1,12 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
2
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
3
3
  import { IonContent } from '@ionic/angular/standalone';
4
4
  import { HeaderComponent } from '../../organisms/header/header.component';
5
- import { ThemeService } from '../../../services/theme.service';
6
- import { NavigationService } from '../../../services/navigation.service';
7
5
  import { resolveColor } from '../../../shared/utils/styles';
8
6
  import * as i0 from "@angular/core";
9
- import * as i1 from "@angular/common";
7
+ import * as i1 from "../../../services/theme.service";
8
+ import * as i2 from "../../../services/navigation.service";
9
+ import * as i3 from "@angular/common";
10
10
  /**
11
11
  * val-page-content
12
12
  *
@@ -34,9 +34,9 @@ import * as i1 from "@angular/common";
34
34
  * @output onHeaderClick - Emits when a header action is clicked
35
35
  */
36
36
  export class PageContentComponent {
37
- constructor() {
38
- this.theme = inject(ThemeService);
39
- this.nav = inject(NavigationService);
37
+ constructor(theme, nav) {
38
+ this.theme = theme;
39
+ this.nav = nav;
40
40
  /**
41
41
  * Page content configuration.
42
42
  */
@@ -108,7 +108,7 @@ export class PageContentComponent {
108
108
  this.nav.navigateByUrl(this.props.homeRoute);
109
109
  }
110
110
  }
111
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageContentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
111
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageContentComponent, deps: [{ token: i1.ThemeService }, { token: i2.NavigationService }], target: i0.ɵɵFactoryTarget.Component }); }
112
112
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: PageContentComponent, isStandalone: true, selector: "val-page-content", inputs: { props: "props" }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
113
113
  <div class="ion-page">
114
114
  <val-header
@@ -128,7 +128,7 @@ export class PageContentComponent {
128
128
  </ion-content>
129
129
  <ng-content select="[extra-footer]"></ng-content>
130
130
  </div>
131
- `, isInline: true, styles: ["main{min-height:60vh}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: HeaderComponent, selector: "val-header", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }] }); }
131
+ `, isInline: true, styles: ["main{min-height:60vh}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: HeaderComponent, selector: "val-header", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }] }); }
132
132
  }
133
133
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageContentComponent, decorators: [{
134
134
  type: Component,
@@ -152,9 +152,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
152
152
  <ng-content select="[extra-footer]"></ng-content>
153
153
  </div>
154
154
  `, styles: ["main{min-height:60vh}\n"] }]
155
- }], propDecorators: { props: [{
155
+ }], ctorParameters: () => [{ type: i1.ThemeService }, { type: i2.NavigationService }], propDecorators: { props: [{
156
156
  type: Input
157
157
  }], onHeaderClick: [{
158
158
  type: Output
159
159
  }] } });
160
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <ng-content select=\"[content]\"></ng-content>\n        </main>\n        <ng-content select=\"[footer]\"></ng-content>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageContentComponent {\n  private theme = inject(ThemeService);\n  private nav = inject(NavigationService);\n\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   */\n  get headerProps() {\n    return this.props.header || this.defaultHeader;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
160
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACvE,OAAO,EAAE,UAAU,EAAE,MAAM,2BAA2B,CAAC;AACvD,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAI1E,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;;;;;AAE5D;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AA+BH,MAAM,OAAO,oBAAoB;IAM/B,YACU,KAAmB,EACnB,GAAsB;QADtB,UAAK,GAAL,KAAK,CAAc;QACnB,QAAG,GAAH,GAAG,CAAmB;QAPhC;;WAEG;QACM,UAAK,GAAwB,EAAE,CAAC;QAOzC;;WAEG;QACO,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;QAErD;;WAEG;QACc,kBAAa,GAAG;YAC/B,QAAQ,EAAE,IAAI;YACd,WAAW,EAAE,IAAI;YACjB,OAAO,EAAE;gBACP,QAAQ,EAAE,KAAK;gBACf,WAAW,EAAE,IAAI;gBACjB,SAAS,EAAE,MAAe;gBAC1B,QAAQ,EAAE,IAAI;gBACd,KAAK,EAAE,EAAE;gBACT,OAAO,EAAE;oBACP;wBACE,KAAK,EAAE,aAAa;wBACpB,WAAW,EAAE,EAAE;wBACf,QAAQ,EAAE,MAAe;wBACzB,IAAI,EAAE,OAAgB;wBACtB,KAAK,EAAE;4BACL,KAAK,EAAE,EAAE;4BACT,GAAG,EAAE,aAAa;4BAClB,GAAG,EAAE,aAAa;4BAClB,IAAI,EAAE,KAAc;4BACpB,MAAM,EAAE,KAAK;4BACb,QAAQ,EAAE,KAAK;4BACf,IAAI,EAAE,OAAgB;4BACtB,OAAO,EAAE,KAAK;4BACd,IAAI,EAAE,IAAI;yBACX;qBACF;iBACF;aACF;SACF,CAAC;IAvCC,CAAC;IAyCJ;;OAEG;IACH,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,aAAa,CAAC;IACjD,CAAC;IAED;;OAEG;IACH,aAAa;QACX,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACtB,OAAO,6BAA6B,CAAC;QACvC,CAAC;QAED,MAAM,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC;QACjC,IAAI,CAAC,EAAE,EAAE,CAAC;YACR,OAAO,6BAA6B,CAAC;QACvC,CAAC;QAED,OAAO,YAAY,CAAC,EAAE,CAAC,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,oBAAoB,CAAC,KAAa;QAChC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAE/B,4DAA4D;QAC5D,IAAI,KAAK,KAAK,aAAa,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACpD,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC;+GAnFU,oBAAoB;mGAApB,oBAAoB,qJA1BrB;;;;;;;;;;;;;;;;;;;GAmBT,gGApBS,YAAY,oHAAE,eAAe,gGAAE,UAAU;;4FA2BxC,oBAAoB;kBA9BhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,EAAE,eAAe,EAAE,UAAU,CAAC,YAC1C;;;;;;;;;;;;;;;;;;;GAmBT;iHAWQ,KAAK;sBAAb,KAAK;gBAUI,aAAa;sBAAtB,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <ng-content select=\"[content]\"></ng-content>\n        </main>\n        <ng-content select=\"[footer]\"></ng-content>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageContentComponent {\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  constructor(\n    private theme: ThemeService,\n    private nav: NavigationService\n  ) {}\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   */\n  get headerProps() {\n    return this.props.header || this.defaultHeader;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
@@ -72,7 +72,6 @@ export class PageTemplateComponent {
72
72
  limit: props.descriptionLimit || 180,
73
73
  content: props.pageDescription,
74
74
  color: props.descriptionColor || 'dark',
75
- expandText: 'more'
76
75
  }"
77
76
  />
78
77
  </div>
@@ -88,7 +87,7 @@ export class PageTemplateComponent {
88
87
  <val-button
89
88
  class="back-button"
90
89
  [props]="{
91
- text: props.backButtonText || 'Back',
90
+ text: props.backButtonText || 'Volver',
92
91
  color: 'dark',
93
92
  size: 'small',
94
93
  type: 'button',
@@ -138,7 +137,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
138
137
  limit: props.descriptionLimit || 180,
139
138
  content: props.pageDescription,
140
139
  color: props.descriptionColor || 'dark',
141
- expandText: 'more'
142
140
  }"
143
141
  />
144
142
  </div>
@@ -154,7 +152,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
154
152
  <val-button
155
153
  class="back-button"
156
154
  [props]="{
157
- text: props.backButtonText || 'Back',
155
+ text: props.backButtonText || 'Volver',
158
156
  color: 'dark',
159
157
  size: 'small',
160
158
  type: 'button',
@@ -178,4 +176,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
178
176
  }], onBack: [{
179
177
  type: Output
180
178
  }] } });
181
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZS10ZW1wbGF0ZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtdGVtcGxhdGUvcGFnZS10ZW1wbGF0ZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMvQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRSxVQUFVLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNyRyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwyREFBMkQsQ0FBQztBQUNwRyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUNBQXFDLENBQUM7O0FBR3RFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBOEZILE1BQU0sT0FBTyxxQkFBcUI7SUE3RmxDO1FBOEZVLFFBQUcsR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFcEM7O1dBRUc7UUFDTSxVQUFLLEdBQXlCLEVBQUUsQ0FBQztRQUUxQzs7V0FFRztRQUNPLFdBQU0sR0FBRyxJQUFJLFlBQVksRUFBUSxDQUFDO0tBUzdDO0lBUEM7O09BRUc7SUFDSCxVQUFVO1FBQ1IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNuQixJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQ2xCLENBQUM7K0dBbkJVLHFCQUFxQjttR0FBckIscUJBQXFCLHdJQS9FdEI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FvRFQsb1NBOURDLFlBQVksK0JBQ1osU0FBUyxvR0FDVCxVQUFVLG1GQUNWLFFBQVEsaUZBQ1IsdUJBQXVCLG1GQUN2QixPQUFPLHdFQUNQLE1BQU0sb0RBQ04sTUFBTSxrVEFDTixlQUFlOzs0RkFpRk4scUJBQXFCO2tCQTdGakMsU0FBUzsrQkFDRSxtQkFBbUIsY0FDakIsSUFBSSxXQUNQO3dCQUNQLFlBQVk7d0JBQ1osU0FBUzt3QkFDVCxVQUFVO3dCQUNWLFFBQVE7d0JBQ1IsdUJBQXVCO3dCQUN2QixPQUFPO3dCQUNQLE1BQU07d0JBQ04sTUFBTTt3QkFDTixlQUFlO3FCQUNoQixZQUNTOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0RUOzhCQWlDUSxLQUFLO3NCQUFiLEtBQUs7Z0JBS0ksTUFBTTtzQkFBZixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBpbmplY3QsIElucHV0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5hdkNvbnRyb2xsZXIgfSBmcm9tICdAaW9uaWMvYW5ndWxhcic7XG5pbXBvcnQgeyBJb25Db2wsIElvbkdyaWQsIElvbkhlYWRlciwgSW9uUm93LCBJb25UaXRsZSwgSW9uVG9vbGJhciB9IGZyb20gJ0Bpb25pYy9hbmd1bGFyL3N0YW5kYWxvbmUnO1xuaW1wb3J0IHsgRXhwYW5kYWJsZVRleHRDb21wb25lbnQgfSBmcm9tICcuLi8uLi9tb2xlY3VsZXMvZXhwYW5kYWJsZS10ZXh0L2V4cGFuZGFibGUtdGV4dC5jb21wb25lbnQnO1xuaW1wb3J0IHsgQnV0dG9uQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvYnV0dG9uL2J1dHRvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgUGFnZVRlbXBsYXRlTWV0YWRhdGEgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiB2YWwtcGFnZS10ZW1wbGF0ZVxuICpcbiAqIEEgcGFnZSB0ZW1wbGF0ZSBjb21wb25lbnQgd2l0aCB0aXRsZSwgZXhwYW5kYWJsZSBkZXNjcmlwdGlvbixcbiAqIGNvbnRlbnQgcHJvamVjdGlvbiwgYW5kIG9wdGlvbmFsIGJhY2sgbmF2aWdhdGlvbiBidXR0b24uXG4gKlxuICogQGV4YW1wbGVcbiAqIDx2YWwtcGFnZS10ZW1wbGF0ZVxuICogICBbcHJvcHNdPVwie1xuICogICAgIHBhZ2VUaXRsZTogJ0dldHRpbmcgU3RhcnRlZCcsXG4gKiAgICAgcGFnZURlc2NyaXB0aW9uOiAnTGVhcm4gaG93IHRvIHVzZSBvdXIgY29tcG9uZW50cy4uLicsXG4gKiAgICAgc2hvd0JhY2tCdXR0b246IHRydWVcbiAqICAgfVwiXG4gKiA+XG4gKiAgIDxkaXYgZXh0cmEtZGVzY3JpcHRpb24+XG4gKiAgICAgPHA+QWRkaXRpb25hbCBpbmZvIGhlcmU8L3A+XG4gKiAgIDwvZGl2PlxuICpcbiAqICAgPCEtLSBNYWluIGNvbnRlbnQgLS0+XG4gKiAgIDxteS1jb250ZW50PjwvbXktY29udGVudD5cbiAqXG4gKiAgIDxkaXYgZXh0cmEtZm9vdGVyPlxuICogICAgIDxwPkZvb3RlciBjb250ZW50PC9wPlxuICogICA8L2Rpdj5cbiAqIDwvdmFsLXBhZ2UtdGVtcGxhdGU+XG4gKlxuICogQGlucHV0IHByb3BzIC0gUGFnZSB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uXG4gKiBAb3V0cHV0IG9uQmFjayAtIEVtaXRzIHdoZW4gYmFjayBidXR0b24gaXMgY2xpY2tlZFxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtcGFnZS10ZW1wbGF0ZScsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgSW9uSGVhZGVyLFxuICAgIElvblRvb2xiYXIsXG4gICAgSW9uVGl0bGUsXG4gICAgRXhwYW5kYWJsZVRleHRDb21wb25lbnQsXG4gICAgSW9uR3JpZCxcbiAgICBJb25Sb3csXG4gICAgSW9uQ29sLFxuICAgIEJ1dHRvbkNvbXBvbmVudCxcbiAgXSxcbiAgdGVtcGxhdGU6IGBcbiAgICBAaWYgKHByb3BzLnBhZ2VUaXRsZSkge1xuICAgICAgPGlvbi1oZWFkZXIgW2NsYXNzLmlvbi1uby1ib3JkZXJdPVwidHJ1ZVwiPlxuICAgICAgICA8aW9uLXRvb2xiYXIgc3R5bGU9XCItLWJhY2tncm91bmQ6IHRyYW5zcGFyZW50O1wiPlxuICAgICAgICAgIDxpb24tdGl0bGUgY2xhc3M9XCJwYWdlLXRpdGxlXCIgc2l6ZT1cImxhcmdlXCI+e3sgcHJvcHMucGFnZVRpdGxlIH19PC9pb24tdGl0bGU+XG4gICAgICAgIDwvaW9uLXRvb2xiYXI+XG4gICAgICA8L2lvbi1oZWFkZXI+XG4gICAgfVxuICAgIDxpb24tZ3JpZD5cbiAgICAgIDxpb24tcm93IGNsYXNzPVwiaW9uLWp1c3RpZnktY29udGVudC1jZW50ZXIgZGVzY3JpcHRpb24tcm93XCI+XG4gICAgICAgIDxpb24tY29sIHNpemU9XCIxMlwiIHNpemUtbWQ9XCIxMFwiIHNpemUtbGc9XCI4XCI+XG4gICAgICAgICAgQGlmIChwcm9wcy5wYWdlRGVzY3JpcHRpb24pIHtcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJkZXNjcmlwdGlvbi1jb250YWluZXJcIj5cbiAgICAgICAgICAgICAgPHZhbC1leHBhbmRhYmxlLXRleHRcbiAgICAgICAgICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgICAgICAgICAgbGltaXQ6IHByb3BzLmRlc2NyaXB0aW9uTGltaXQgfHwgMTgwLFxuICAgICAgICAgICAgICAgICAgY29udGVudDogcHJvcHMucGFnZURlc2NyaXB0aW9uLFxuICAgICAgICAgICAgICAgICAgY29sb3I6IHByb3BzLmRlc2NyaXB0aW9uQ29sb3IgfHwgJ2RhcmsnLFxuICAgICAgICAgICAgICAgICAgZXhwYW5kVGV4dDogJ21vcmUnXG4gICAgICAgICAgICAgICAgfVwiXG4gICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICB9XG4gICAgICAgICAgPG5nLWNvbnRlbnQgc2VsZWN0PVwiW2V4dHJhLWRlc2NyaXB0aW9uXVwiPjwvbmctY29udGVudD5cbiAgICAgICAgPC9pb24tY29sPlxuICAgICAgPC9pb24tcm93PlxuICAgICAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuICAgICAgPG5nLWNvbnRlbnQgc2VsZWN0PVwiW2V4dHJhLWZvb3Rlcl1cIj48L25nLWNvbnRlbnQ+XG4gICAgICBAaWYgKHByb3BzLnNob3dCYWNrQnV0dG9uKSB7XG4gICAgICAgIDxpb24tcm93IGNsYXNzPVwiaW9uLWp1c3RpZnktY29udGVudC1jZW50ZXIgYmFjay1yb3dcIj5cbiAgICAgICAgICA8aW9uLWNvbCBzaXplPVwiMTJcIiBzaXplLW1kPVwiMTBcIiBzaXplLWxnPVwiOFwiPlxuICAgICAgICAgICAgPHZhbC1idXR0b25cbiAgICAgICAgICAgICAgY2xhc3M9XCJiYWNrLWJ1dHRvblwiXG4gICAgICAgICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgICAgICAgdGV4dDogcHJvcHMuYmFja0J1dHRvblRleHQgfHwgJ0JhY2snLFxuICAgICAgICAgICAgICAgIGNvbG9yOiAnZGFyaycsXG4gICAgICAgICAgICAgICAgc2l6ZTogJ3NtYWxsJyxcbiAgICAgICAgICAgICAgICB0eXBlOiAnYnV0dG9uJyxcbiAgICAgICAgICAgICAgICBzdGF0ZTogJ0VOQUJMRUQnLFxuICAgICAgICAgICAgICAgIGZpbGw6ICdvdXRsaW5lJyxcbiAgICAgICAgICAgICAgICBzaGFwZTogJ3JvdW5kJyxcbiAgICAgICAgICAgICAgICBpY29uOiB7XG4gICAgICAgICAgICAgICAgICBuYW1lOiAnYXJyb3ctYmFjay1vdXRsaW5lJyxcbiAgICAgICAgICAgICAgICAgIHNsb3Q6ICdzdGFydCdcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cIlxuICAgICAgICAgICAgICAob25DbGljayk9XCJoYW5kbGVCYWNrKClcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L2lvbi1jb2w+XG4gICAgICAgIDwvaW9uLXJvdz5cbiAgICAgIH1cbiAgICA8L2lvbi1ncmlkPlxuICBgLFxuICBzdHlsZXM6IGBcbiAgICAucGFnZS10aXRsZSB7XG4gICAgICBtYXJnaW4tbGVmdDogLTRweDtcbiAgICAgIHBhZGRpbmc6IDA7XG4gICAgICBmb250LXNpemU6IDIuNXJlbTtcbiAgICAgIGZvbnQtd2VpZ2h0OiA4MDA7XG4gICAgfVxuXG4gICAgLmRlc2NyaXB0aW9uLXJvdyB7XG4gICAgICBtYXJnaW4tYm90dG9tOiAxNnB4O1xuICAgIH1cblxuICAgIC5kZXNjcmlwdGlvbi1jb250YWluZXIge1xuICAgICAgbWFyZ2luLXRvcDogMXJlbTtcbiAgICB9XG5cbiAgICAuYmFjay1yb3cge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMTZweDtcbiAgICB9XG5cbiAgICAuYmFjay1idXR0b24ge1xuICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICBtYXJnaW46IDFyZW0gMDtcbiAgICB9XG4gIGAsXG59KVxuZXhwb3J0IGNsYXNzIFBhZ2VUZW1wbGF0ZUNvbXBvbmVudCB7XG4gIHByaXZhdGUgbmF2ID0gaW5qZWN0KE5hdkNvbnRyb2xsZXIpO1xuXG4gIC8qKlxuICAgKiBQYWdlIHRlbXBsYXRlIGNvbmZpZ3VyYXRpb24uXG4gICAqL1xuICBASW5wdXQoKSBwcm9wczogUGFnZVRlbXBsYXRlTWV0YWRhdGEgPSB7fTtcblxuICAvKipcbiAgICogRW1pdHMgd2hlbiB0aGUgYmFjayBidXR0b24gaXMgY2xpY2tlZC5cbiAgICovXG4gIEBPdXRwdXQoKSBvbkJhY2sgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG5cbiAgLyoqXG4gICAqIEhhbmRsZXMgYmFjayBuYXZpZ2F0aW9uLlxuICAgKi9cbiAgaGFuZGxlQmFjaygpOiB2b2lkIHtcbiAgICB0aGlzLm9uQmFjay5lbWl0KCk7XG4gICAgdGhpcy5uYXYuYmFjaygpO1xuICB9XG59XG4iXX0=
179
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZS10ZW1wbGF0ZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtdGVtcGxhdGUvcGFnZS10ZW1wbGF0ZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMvQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRSxVQUFVLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNyRyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwyREFBMkQsQ0FBQztBQUNwRyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUNBQXFDLENBQUM7O0FBR3RFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBNkZILE1BQU0sT0FBTyxxQkFBcUI7SUE1RmxDO1FBNkZVLFFBQUcsR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFcEM7O1dBRUc7UUFDTSxVQUFLLEdBQXlCLEVBQUUsQ0FBQztRQUUxQzs7V0FFRztRQUNPLFdBQU0sR0FBRyxJQUFJLFlBQVksRUFBUSxDQUFDO0tBUzdDO0lBUEM7O09BRUc7SUFDSCxVQUFVO1FBQ1IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNuQixJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQ2xCLENBQUM7K0dBbkJVLHFCQUFxQjttR0FBckIscUJBQXFCLHdJQTlFdEI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1EVCxvU0E3REMsWUFBWSwrQkFDWixTQUFTLG9HQUNULFVBQVUsbUZBQ1YsUUFBUSxpRkFDUix1QkFBdUIsbUZBQ3ZCLE9BQU8sd0VBQ1AsTUFBTSxvREFDTixNQUFNLGtUQUNOLGVBQWU7OzRGQWdGTixxQkFBcUI7a0JBNUZqQyxTQUFTOytCQUNFLG1CQUFtQixjQUNqQixJQUFJLFdBQ1A7d0JBQ1AsWUFBWTt3QkFDWixTQUFTO3dCQUNULFVBQVU7d0JBQ1YsUUFBUTt3QkFDUix1QkFBdUI7d0JBQ3ZCLE9BQU87d0JBQ1AsTUFBTTt3QkFDTixNQUFNO3dCQUNOLGVBQWU7cUJBQ2hCLFlBQ1M7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1EVDs4QkFpQ1EsS0FBSztzQkFBYixLQUFLO2dCQUtJLE1BQU07c0JBQWYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgaW5qZWN0LCBJbnB1dCwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOYXZDb250cm9sbGVyIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXInO1xuaW1wb3J0IHsgSW9uQ29sLCBJb25HcmlkLCBJb25IZWFkZXIsIElvblJvdywgSW9uVGl0bGUsIElvblRvb2xiYXIgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IEV4cGFuZGFibGVUZXh0Q29tcG9uZW50IH0gZnJvbSAnLi4vLi4vbW9sZWN1bGVzL2V4cGFuZGFibGUtdGV4dC9leHBhbmRhYmxlLXRleHQuY29tcG9uZW50JztcbmltcG9ydCB7IEJ1dHRvbkNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2J1dHRvbi9idXR0b24uY29tcG9uZW50JztcbmltcG9ydCB7IFBhZ2VUZW1wbGF0ZU1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogdmFsLXBhZ2UtdGVtcGxhdGVcbiAqXG4gKiBBIHBhZ2UgdGVtcGxhdGUgY29tcG9uZW50IHdpdGggdGl0bGUsIGV4cGFuZGFibGUgZGVzY3JpcHRpb24sXG4gKiBjb250ZW50IHByb2plY3Rpb24sIGFuZCBvcHRpb25hbCBiYWNrIG5hdmlnYXRpb24gYnV0dG9uLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLXBhZ2UtdGVtcGxhdGVcbiAqICAgW3Byb3BzXT1cIntcbiAqICAgICBwYWdlVGl0bGU6ICdHZXR0aW5nIFN0YXJ0ZWQnLFxuICogICAgIHBhZ2VEZXNjcmlwdGlvbjogJ0xlYXJuIGhvdyB0byB1c2Ugb3VyIGNvbXBvbmVudHMuLi4nLFxuICogICAgIHNob3dCYWNrQnV0dG9uOiB0cnVlXG4gKiAgIH1cIlxuICogPlxuICogICA8ZGl2IGV4dHJhLWRlc2NyaXB0aW9uPlxuICogICAgIDxwPkFkZGl0aW9uYWwgaW5mbyBoZXJlPC9wPlxuICogICA8L2Rpdj5cbiAqXG4gKiAgIDwhLS0gTWFpbiBjb250ZW50IC0tPlxuICogICA8bXktY29udGVudD48L215LWNvbnRlbnQ+XG4gKlxuICogICA8ZGl2IGV4dHJhLWZvb3Rlcj5cbiAqICAgICA8cD5Gb290ZXIgY29udGVudDwvcD5cbiAqICAgPC9kaXY+XG4gKiA8L3ZhbC1wYWdlLXRlbXBsYXRlPlxuICpcbiAqIEBpbnB1dCBwcm9wcyAtIFBhZ2UgdGVtcGxhdGUgY29uZmlndXJhdGlvblxuICogQG91dHB1dCBvbkJhY2sgLSBFbWl0cyB3aGVuIGJhY2sgYnV0dG9uIGlzIGNsaWNrZWRcbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLXBhZ2UtdGVtcGxhdGUnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbXG4gICAgQ29tbW9uTW9kdWxlLFxuICAgIElvbkhlYWRlcixcbiAgICBJb25Ub29sYmFyLFxuICAgIElvblRpdGxlLFxuICAgIEV4cGFuZGFibGVUZXh0Q29tcG9uZW50LFxuICAgIElvbkdyaWQsXG4gICAgSW9uUm93LFxuICAgIElvbkNvbCxcbiAgICBCdXR0b25Db21wb25lbnQsXG4gIF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgQGlmIChwcm9wcy5wYWdlVGl0bGUpIHtcbiAgICAgIDxpb24taGVhZGVyIFtjbGFzcy5pb24tbm8tYm9yZGVyXT1cInRydWVcIj5cbiAgICAgICAgPGlvbi10b29sYmFyIHN0eWxlPVwiLS1iYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDtcIj5cbiAgICAgICAgICA8aW9uLXRpdGxlIGNsYXNzPVwicGFnZS10aXRsZVwiIHNpemU9XCJsYXJnZVwiPnt7IHByb3BzLnBhZ2VUaXRsZSB9fTwvaW9uLXRpdGxlPlxuICAgICAgICA8L2lvbi10b29sYmFyPlxuICAgICAgPC9pb24taGVhZGVyPlxuICAgIH1cbiAgICA8aW9uLWdyaWQ+XG4gICAgICA8aW9uLXJvdyBjbGFzcz1cImlvbi1qdXN0aWZ5LWNvbnRlbnQtY2VudGVyIGRlc2NyaXB0aW9uLXJvd1wiPlxuICAgICAgICA8aW9uLWNvbCBzaXplPVwiMTJcIiBzaXplLW1kPVwiMTBcIiBzaXplLWxnPVwiOFwiPlxuICAgICAgICAgIEBpZiAocHJvcHMucGFnZURlc2NyaXB0aW9uKSB7XG4gICAgICAgICAgICA8ZGl2IGNsYXNzPVwiZGVzY3JpcHRpb24tY29udGFpbmVyXCI+XG4gICAgICAgICAgICAgIDx2YWwtZXhwYW5kYWJsZS10ZXh0XG4gICAgICAgICAgICAgICAgW3Byb3BzXT1cIntcbiAgICAgICAgICAgICAgICAgIGxpbWl0OiBwcm9wcy5kZXNjcmlwdGlvbkxpbWl0IHx8IDE4MCxcbiAgICAgICAgICAgICAgICAgIGNvbnRlbnQ6IHByb3BzLnBhZ2VEZXNjcmlwdGlvbixcbiAgICAgICAgICAgICAgICAgIGNvbG9yOiBwcm9wcy5kZXNjcmlwdGlvbkNvbG9yIHx8ICdkYXJrJyxcbiAgICAgICAgICAgICAgICB9XCJcbiAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgIH1cbiAgICAgICAgICA8bmctY29udGVudCBzZWxlY3Q9XCJbZXh0cmEtZGVzY3JpcHRpb25dXCI+PC9uZy1jb250ZW50PlxuICAgICAgICA8L2lvbi1jb2w+XG4gICAgICA8L2lvbi1yb3c+XG4gICAgICA8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG4gICAgICA8bmctY29udGVudCBzZWxlY3Q9XCJbZXh0cmEtZm9vdGVyXVwiPjwvbmctY29udGVudD5cbiAgICAgIEBpZiAocHJvcHMuc2hvd0JhY2tCdXR0b24pIHtcbiAgICAgICAgPGlvbi1yb3cgY2xhc3M9XCJpb24tanVzdGlmeS1jb250ZW50LWNlbnRlciBiYWNrLXJvd1wiPlxuICAgICAgICAgIDxpb24tY29sIHNpemU9XCIxMlwiIHNpemUtbWQ9XCIxMFwiIHNpemUtbGc9XCI4XCI+XG4gICAgICAgICAgICA8dmFsLWJ1dHRvblxuICAgICAgICAgICAgICBjbGFzcz1cImJhY2stYnV0dG9uXCJcbiAgICAgICAgICAgICAgW3Byb3BzXT1cIntcbiAgICAgICAgICAgICAgICB0ZXh0OiBwcm9wcy5iYWNrQnV0dG9uVGV4dCB8fCAnVm9sdmVyJyxcbiAgICAgICAgICAgICAgICBjb2xvcjogJ2RhcmsnLFxuICAgICAgICAgICAgICAgIHNpemU6ICdzbWFsbCcsXG4gICAgICAgICAgICAgICAgdHlwZTogJ2J1dHRvbicsXG4gICAgICAgICAgICAgICAgc3RhdGU6ICdFTkFCTEVEJyxcbiAgICAgICAgICAgICAgICBmaWxsOiAnb3V0bGluZScsXG4gICAgICAgICAgICAgICAgc2hhcGU6ICdyb3VuZCcsXG4gICAgICAgICAgICAgICAgaWNvbjoge1xuICAgICAgICAgICAgICAgICAgbmFtZTogJ2Fycm93LWJhY2stb3V0bGluZScsXG4gICAgICAgICAgICAgICAgICBzbG90OiAnc3RhcnQnXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XCJcbiAgICAgICAgICAgICAgKG9uQ2xpY2spPVwiaGFuZGxlQmFjaygpXCJcbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9pb24tY29sPlxuICAgICAgICA8L2lvbi1yb3c+XG4gICAgICB9XG4gICAgPC9pb24tZ3JpZD5cbiAgYCxcbiAgc3R5bGVzOiBgXG4gICAgLnBhZ2UtdGl0bGUge1xuICAgICAgbWFyZ2luLWxlZnQ6IC00cHg7XG4gICAgICBwYWRkaW5nOiAwO1xuICAgICAgZm9udC1zaXplOiAyLjVyZW07XG4gICAgICBmb250LXdlaWdodDogODAwO1xuICAgIH1cblxuICAgIC5kZXNjcmlwdGlvbi1yb3cge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMTZweDtcbiAgICB9XG5cbiAgICAuZGVzY3JpcHRpb24tY29udGFpbmVyIHtcbiAgICAgIG1hcmdpbi10b3A6IDFyZW07XG4gICAgfVxuXG4gICAgLmJhY2stcm93IHtcbiAgICAgIG1hcmdpbi1ib3R0b206IDE2cHg7XG4gICAgfVxuXG4gICAgLmJhY2stYnV0dG9uIHtcbiAgICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgICAgbWFyZ2luOiAxcmVtIDA7XG4gICAgfVxuICBgLFxufSlcbmV4cG9ydCBjbGFzcyBQYWdlVGVtcGxhdGVDb21wb25lbnQge1xuICBwcml2YXRlIG5hdiA9IGluamVjdChOYXZDb250cm9sbGVyKTtcblxuICAvKipcbiAgICogUGFnZSB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uLlxuICAgKi9cbiAgQElucHV0KCkgcHJvcHM6IFBhZ2VUZW1wbGF0ZU1ldGFkYXRhID0ge307XG5cbiAgLyoqXG4gICAqIEVtaXRzIHdoZW4gdGhlIGJhY2sgYnV0dG9uIGlzIGNsaWNrZWQuXG4gICAqL1xuICBAT3V0cHV0KCkgb25CYWNrID0gbmV3IEV2ZW50RW1pdHRlcjx2b2lkPigpO1xuXG4gIC8qKlxuICAgKiBIYW5kbGVzIGJhY2sgbmF2aWdhdGlvbi5cbiAgICovXG4gIGhhbmRsZUJhY2soKTogdm9pZCB7XG4gICAgdGhpcy5vbkJhY2suZW1pdCgpO1xuICAgIHRoaXMubmF2LmJhY2soKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,173 @@
1
+ import { Injectable, signal, computed } from '@angular/core';
2
+ import { INITIAL_AUTH_STATE, INITIAL_MFA_STATE, } from './types';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * Servicio para manejo de estado de autenticación con Angular Signals.
6
+ * Proporciona estado reactivo inmutable.
7
+ */
8
+ export class AuthStateService {
9
+ constructor() {
10
+ // Estado interno (mutable solo dentro del servicio)
11
+ this._state = signal(INITIAL_AUTH_STATE);
12
+ this._mfaPending = signal(INITIAL_MFA_STATE);
13
+ // =============================================
14
+ // Signals públicos (readonly)
15
+ // =============================================
16
+ /** Estado completo de autenticación */
17
+ this.state = this._state.asReadonly();
18
+ /** Estado de MFA pendiente */
19
+ this.mfaPending = this._mfaPending.asReadonly();
20
+ /** Usuario está autenticado */
21
+ this.isAuthenticated = computed(() => this._state().isAuthenticated);
22
+ /** Estado de carga */
23
+ this.isLoading = computed(() => this._state().isLoading);
24
+ /** Token de acceso */
25
+ this.accessToken = computed(() => this._state().accessToken);
26
+ /** Roles del usuario */
27
+ this.roles = computed(() => this._state().roles);
28
+ /** Permisos del usuario */
29
+ this.permissions = computed(() => this._state().permissions);
30
+ /** Usuario es super admin */
31
+ this.isSuperAdmin = computed(() => this._state().isSuperAdmin);
32
+ /** Error actual */
33
+ this.error = computed(() => this._state().error);
34
+ /** Información del usuario */
35
+ this.user = computed(() => {
36
+ const state = this._state();
37
+ if (!state.isAuthenticated || !state.userId) {
38
+ return null;
39
+ }
40
+ return {
41
+ userId: state.userId,
42
+ email: state.email || '',
43
+ roles: state.roles,
44
+ permissions: state.permissions,
45
+ isSuperAdmin: state.isSuperAdmin,
46
+ };
47
+ });
48
+ }
49
+ // =============================================
50
+ // Métodos de actualización
51
+ // =============================================
52
+ /**
53
+ * Establece el estado de carga.
54
+ */
55
+ setLoading(isLoading) {
56
+ this._state.update((s) => ({ ...s, isLoading }));
57
+ }
58
+ /**
59
+ * Establece el estado de autenticación exitosa.
60
+ */
61
+ setAuthenticated(data) {
62
+ this._state.set({
63
+ isAuthenticated: true,
64
+ isLoading: false,
65
+ accessToken: data.accessToken,
66
+ refreshToken: data.refreshToken,
67
+ userId: data.userId || null,
68
+ email: data.email || null,
69
+ roles: data.roles,
70
+ permissions: data.permissions,
71
+ isSuperAdmin: data.isSuperAdmin,
72
+ expiresAt: data.expiresAt,
73
+ error: null,
74
+ });
75
+ }
76
+ /**
77
+ * Actualiza solo el access token (después de refresh).
78
+ */
79
+ updateAccessToken(accessToken, expiresIn) {
80
+ const expiresAt = Date.now() + expiresIn * 1000;
81
+ this._state.update((s) => ({
82
+ ...s,
83
+ accessToken,
84
+ expiresAt,
85
+ }));
86
+ }
87
+ /**
88
+ * Actualiza los permisos.
89
+ */
90
+ updatePermissions(roles, permissions, isSuperAdmin) {
91
+ this._state.update((s) => ({
92
+ ...s,
93
+ roles,
94
+ permissions,
95
+ isSuperAdmin,
96
+ }));
97
+ }
98
+ /**
99
+ * Establece un error de autenticación.
100
+ */
101
+ setError(error) {
102
+ this._state.update((s) => ({
103
+ ...s,
104
+ error,
105
+ isLoading: false,
106
+ }));
107
+ }
108
+ /**
109
+ * Limpia el error.
110
+ */
111
+ clearError() {
112
+ this._state.update((s) => ({
113
+ ...s,
114
+ error: null,
115
+ }));
116
+ }
117
+ /**
118
+ * Establece estado de MFA pendiente.
119
+ */
120
+ setMFAPending(mfaState) {
121
+ this._mfaPending.set(mfaState);
122
+ }
123
+ /**
124
+ * Limpia el estado de MFA pendiente.
125
+ */
126
+ clearMFAPending() {
127
+ this._mfaPending.set(INITIAL_MFA_STATE);
128
+ }
129
+ /**
130
+ * Resetea todo el estado a valores iniciales.
131
+ */
132
+ reset() {
133
+ this._state.set(INITIAL_AUTH_STATE);
134
+ this._mfaPending.set(INITIAL_MFA_STATE);
135
+ }
136
+ /**
137
+ * Restaura estado desde datos almacenados.
138
+ */
139
+ restoreFromStorage(stored) {
140
+ if (stored.accessToken) {
141
+ this._state.set({
142
+ isAuthenticated: true,
143
+ isLoading: false,
144
+ accessToken: stored.accessToken,
145
+ refreshToken: stored.refreshToken || null,
146
+ userId: null, // Se extraerá del token
147
+ email: null, // Se extraerá del token
148
+ roles: stored.roles || [],
149
+ permissions: stored.permissions || [],
150
+ isSuperAdmin: stored.isSuperAdmin || false,
151
+ expiresAt: stored.expiresAt || null,
152
+ error: null,
153
+ });
154
+ }
155
+ }
156
+ /**
157
+ * Actualiza el userId y email (después de parsear el token).
158
+ */
159
+ updateUserInfo(userId, email) {
160
+ this._state.update((s) => ({
161
+ ...s,
162
+ userId,
163
+ email,
164
+ }));
165
+ }
166
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
167
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthStateService, providedIn: 'root' }); }
168
+ }
169
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthStateService, decorators: [{
170
+ type: Injectable,
171
+ args: [{ providedIn: 'root' }]
172
+ }] });
173
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth-state.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, computed } from '@angular/core';\nimport {\n  AuthState,\n  AuthUser,\n  AuthError,\n  MFAPendingState,\n  INITIAL_AUTH_STATE,\n  INITIAL_MFA_STATE,\n  StoredAuthState,\n} from './types';\n\n/**\n * Servicio para manejo de estado de autenticación con Angular Signals.\n * Proporciona estado reactivo inmutable.\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthStateService {\n  // Estado interno (mutable solo dentro del servicio)\n  private _state = signal<AuthState>(INITIAL_AUTH_STATE);\n  private _mfaPending = signal<MFAPendingState>(INITIAL_MFA_STATE);\n\n  // =============================================\n  // Signals públicos (readonly)\n  // =============================================\n\n  /** Estado completo de autenticación */\n  readonly state = this._state.asReadonly();\n\n  /** Estado de MFA pendiente */\n  readonly mfaPending = this._mfaPending.asReadonly();\n\n  /** Usuario está autenticado */\n  readonly isAuthenticated = computed(() => this._state().isAuthenticated);\n\n  /** Estado de carga */\n  readonly isLoading = computed(() => this._state().isLoading);\n\n  /** Token de acceso */\n  readonly accessToken = computed(() => this._state().accessToken);\n\n  /** Roles del usuario */\n  readonly roles = computed(() => this._state().roles);\n\n  /** Permisos del usuario */\n  readonly permissions = computed(() => this._state().permissions);\n\n  /** Usuario es super admin */\n  readonly isSuperAdmin = computed(() => this._state().isSuperAdmin);\n\n  /** Error actual */\n  readonly error = computed(() => this._state().error);\n\n  /** Información del usuario */\n  readonly user = computed<AuthUser | null>(() => {\n    const state = this._state();\n    if (!state.isAuthenticated || !state.userId) {\n      return null;\n    }\n    return {\n      userId: state.userId,\n      email: state.email || '',\n      roles: state.roles,\n      permissions: state.permissions,\n      isSuperAdmin: state.isSuperAdmin,\n    };\n  });\n\n  // =============================================\n  // Métodos de actualización\n  // =============================================\n\n  /**\n   * Establece el estado de carga.\n   */\n  setLoading(isLoading: boolean): void {\n    this._state.update((s) => ({ ...s, isLoading }));\n  }\n\n  /**\n   * Establece el estado de autenticación exitosa.\n   */\n  setAuthenticated(data: {\n    accessToken: string;\n    refreshToken: string;\n    userId?: string;\n    email?: string;\n    roles: string[];\n    permissions: string[];\n    isSuperAdmin: boolean;\n    expiresAt: number;\n  }): void {\n    this._state.set({\n      isAuthenticated: true,\n      isLoading: false,\n      accessToken: data.accessToken,\n      refreshToken: data.refreshToken,\n      userId: data.userId || null,\n      email: data.email || null,\n      roles: data.roles,\n      permissions: data.permissions,\n      isSuperAdmin: data.isSuperAdmin,\n      expiresAt: data.expiresAt,\n      error: null,\n    });\n  }\n\n  /**\n   * Actualiza solo el access token (después de refresh).\n   */\n  updateAccessToken(accessToken: string, expiresIn: number): void {\n    const expiresAt = Date.now() + expiresIn * 1000;\n    this._state.update((s) => ({\n      ...s,\n      accessToken,\n      expiresAt,\n    }));\n  }\n\n  /**\n   * Actualiza los permisos.\n   */\n  updatePermissions(\n    roles: string[],\n    permissions: string[],\n    isSuperAdmin: boolean\n  ): void {\n    this._state.update((s) => ({\n      ...s,\n      roles,\n      permissions,\n      isSuperAdmin,\n    }));\n  }\n\n  /**\n   * Establece un error de autenticación.\n   */\n  setError(error: AuthError): void {\n    this._state.update((s) => ({\n      ...s,\n      error,\n      isLoading: false,\n    }));\n  }\n\n  /**\n   * Limpia el error.\n   */\n  clearError(): void {\n    this._state.update((s) => ({\n      ...s,\n      error: null,\n    }));\n  }\n\n  /**\n   * Establece estado de MFA pendiente.\n   */\n  setMFAPending(mfaState: MFAPendingState): void {\n    this._mfaPending.set(mfaState);\n  }\n\n  /**\n   * Limpia el estado de MFA pendiente.\n   */\n  clearMFAPending(): void {\n    this._mfaPending.set(INITIAL_MFA_STATE);\n  }\n\n  /**\n   * Resetea todo el estado a valores iniciales.\n   */\n  reset(): void {\n    this._state.set(INITIAL_AUTH_STATE);\n    this._mfaPending.set(INITIAL_MFA_STATE);\n  }\n\n  /**\n   * Restaura estado desde datos almacenados.\n   */\n  restoreFromStorage(stored: Partial<StoredAuthState>): void {\n    if (stored.accessToken) {\n      this._state.set({\n        isAuthenticated: true,\n        isLoading: false,\n        accessToken: stored.accessToken,\n        refreshToken: stored.refreshToken || null,\n        userId: null, // Se extraerá del token\n        email: null, // Se extraerá del token\n        roles: stored.roles || [],\n        permissions: stored.permissions || [],\n        isSuperAdmin: stored.isSuperAdmin || false,\n        expiresAt: stored.expiresAt || null,\n        error: null,\n      });\n    }\n  }\n\n  /**\n   * Actualiza el userId y email (después de parsear el token).\n   */\n  updateUserInfo(userId: string, email: string): void {\n    this._state.update((s) => ({\n      ...s,\n      userId,\n      email,\n    }));\n  }\n}\n"]}