valtech-components 2.0.489 → 2.0.491

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.
@@ -1,9 +1,10 @@
1
1
  import { CommonModule } from '@angular/common';
2
2
  import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
3
3
  import { NavController } from '@ionic/angular';
4
- import { IonCol, IonGrid, IonHeader, IonRow, IonTitle, IonToolbar } from '@ionic/angular/standalone';
4
+ import { IonCol, IonGrid, IonRow } from '@ionic/angular/standalone';
5
5
  import { ExpandableTextComponent } from '../../molecules/expandable-text/expandable-text.component';
6
6
  import { ButtonComponent } from '../../atoms/button/button.component';
7
+ import { TitleComponent } from '../../atoms/title/title.component';
7
8
  import * as i0 from "@angular/core";
8
9
  /**
9
10
  * val-page-template
@@ -56,11 +57,16 @@ export class PageTemplateComponent {
56
57
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageTemplateComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
57
58
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageTemplateComponent, isStandalone: true, selector: "val-page-template", inputs: { props: "props" }, outputs: { onBack: "onBack" }, ngImport: i0, template: `
58
59
  @if (props.pageTitle) {
59
- <ion-header [class.ion-no-border]="true">
60
- <ion-toolbar style="--background: transparent;">
61
- <ion-title class="page-title" size="medium">{{ props.pageTitle }}</ion-title>
62
- </ion-toolbar>
63
- </ion-header>
60
+ <div class="page-title-container">
61
+ <val-title
62
+ [props]="{
63
+ content: props.pageTitle,
64
+ size: 'large',
65
+ color: 'dark',
66
+ bold: true
67
+ }"
68
+ />
69
+ </div>
64
70
  }
65
71
  <ion-grid>
66
72
  <ion-row class="ion-justify-content-center description-row">
@@ -105,15 +111,13 @@ export class PageTemplateComponent {
105
111
  </ion-row>
106
112
  }
107
113
  </ion-grid>
108
- `, isInline: true, styles: [".page-title{margin-left:-4px;padding:0;font-size:2.5rem;font-weight:800;text-align:left}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: ExpandableTextComponent, selector: "val-expandable-text", inputs: ["props"] }, { kind: "component", type: IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { kind: "component", type: IonRow, selector: "ion-row" }, { kind: "component", type: IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: ButtonComponent, selector: "val-button", inputs: ["props"], outputs: ["onClick"] }] }); }
114
+ `, isInline: true, styles: [".page-title-container{padding:0 16px;margin-bottom:8px}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: TitleComponent, selector: "val-title", inputs: ["props"] }, { kind: "component", type: ExpandableTextComponent, selector: "val-expandable-text", inputs: ["props"] }, { kind: "component", type: IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { kind: "component", type: IonRow, selector: "ion-row" }, { kind: "component", type: IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: ButtonComponent, selector: "val-button", inputs: ["props"], outputs: ["onClick"] }] }); }
109
115
  }
110
116
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageTemplateComponent, decorators: [{
111
117
  type: Component,
112
118
  args: [{ selector: 'val-page-template', standalone: true, imports: [
113
119
  CommonModule,
114
- IonHeader,
115
- IonToolbar,
116
- IonTitle,
120
+ TitleComponent,
117
121
  ExpandableTextComponent,
118
122
  IonGrid,
119
123
  IonRow,
@@ -121,11 +125,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
121
125
  ButtonComponent,
122
126
  ], template: `
123
127
  @if (props.pageTitle) {
124
- <ion-header [class.ion-no-border]="true">
125
- <ion-toolbar style="--background: transparent;">
126
- <ion-title class="page-title" size="medium">{{ props.pageTitle }}</ion-title>
127
- </ion-toolbar>
128
- </ion-header>
128
+ <div class="page-title-container">
129
+ <val-title
130
+ [props]="{
131
+ content: props.pageTitle,
132
+ size: 'large',
133
+ color: 'dark',
134
+ bold: true
135
+ }"
136
+ />
137
+ </div>
129
138
  }
130
139
  <ion-grid>
131
140
  <ion-row class="ion-justify-content-center description-row">
@@ -170,10 +179,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
170
179
  </ion-row>
171
180
  }
172
181
  </ion-grid>
173
- `, styles: [".page-title{margin-left:-4px;padding:0;font-size:2.5rem;font-weight:800;text-align:left}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"] }]
182
+ `, styles: [".page-title-container{padding:0 16px;margin-bottom:8px}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"] }]
174
183
  }], propDecorators: { props: [{
175
184
  type: Input
176
185
  }], onBack: [{
177
186
  type: Output
178
187
  }] } });
179
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZS10ZW1wbGF0ZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtdGVtcGxhdGUvcGFnZS10ZW1wbGF0ZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMvQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRSxVQUFVLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNyRyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwyREFBMkQsQ0FBQztBQUNwRyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUNBQXFDLENBQUM7O0FBR3RFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBOEZILE1BQU0sT0FBTyxxQkFBcUI7SUE3RmxDO1FBOEZVLFFBQUcsR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFcEM7O1dBRUc7UUFDTSxVQUFLLEdBQXlCLEVBQUUsQ0FBQztRQUUxQzs7V0FFRztRQUNPLFdBQU0sR0FBRyxJQUFJLFlBQVksRUFBUSxDQUFDO0tBUzdDO0lBUEM7O09BRUc7SUFDSCxVQUFVO1FBQ1IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNuQixJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQ2xCLENBQUM7K0dBbkJVLHFCQUFxQjttR0FBckIscUJBQXFCLHdJQS9FdEI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1EVCxvVEE3REMsWUFBWSwrQkFDWixTQUFTLG9HQUNULFVBQVUsbUZBQ1YsUUFBUSxpRkFDUix1QkFBdUIsbUZBQ3ZCLE9BQU8sd0VBQ1AsTUFBTSxvREFDTixNQUFNLGtUQUNOLGVBQWU7OzRGQWlGTixxQkFBcUI7a0JBN0ZqQyxTQUFTOytCQUNFLG1CQUFtQixjQUNqQixJQUFJLFdBQ1A7d0JBQ1AsWUFBWTt3QkFDWixTQUFTO3dCQUNULFVBQVU7d0JBQ1YsUUFBUTt3QkFDUix1QkFBdUI7d0JBQ3ZCLE9BQU87d0JBQ1AsTUFBTTt3QkFDTixNQUFNO3dCQUNOLGVBQWU7cUJBQ2hCLFlBQ1M7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1EVDs4QkFrQ1EsS0FBSztzQkFBYixLQUFLO2dCQUtJLE1BQU07c0JBQWYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgaW5qZWN0LCBJbnB1dCwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOYXZDb250cm9sbGVyIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXInO1xuaW1wb3J0IHsgSW9uQ29sLCBJb25HcmlkLCBJb25IZWFkZXIsIElvblJvdywgSW9uVGl0bGUsIElvblRvb2xiYXIgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IEV4cGFuZGFibGVUZXh0Q29tcG9uZW50IH0gZnJvbSAnLi4vLi4vbW9sZWN1bGVzL2V4cGFuZGFibGUtdGV4dC9leHBhbmRhYmxlLXRleHQuY29tcG9uZW50JztcbmltcG9ydCB7IEJ1dHRvbkNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2J1dHRvbi9idXR0b24uY29tcG9uZW50JztcbmltcG9ydCB7IFBhZ2VUZW1wbGF0ZU1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogdmFsLXBhZ2UtdGVtcGxhdGVcbiAqXG4gKiBBIHBhZ2UgdGVtcGxhdGUgY29tcG9uZW50IHdpdGggdGl0bGUsIGV4cGFuZGFibGUgZGVzY3JpcHRpb24sXG4gKiBjb250ZW50IHByb2plY3Rpb24sIGFuZCBvcHRpb25hbCBiYWNrIG5hdmlnYXRpb24gYnV0dG9uLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLXBhZ2UtdGVtcGxhdGVcbiAqICAgW3Byb3BzXT1cIntcbiAqICAgICBwYWdlVGl0bGU6ICdHZXR0aW5nIFN0YXJ0ZWQnLFxuICogICAgIHBhZ2VEZXNjcmlwdGlvbjogJ0xlYXJuIGhvdyB0byB1c2Ugb3VyIGNvbXBvbmVudHMuLi4nLFxuICogICAgIHNob3dCYWNrQnV0dG9uOiB0cnVlXG4gKiAgIH1cIlxuICogPlxuICogICA8ZGl2IGV4dHJhLWRlc2NyaXB0aW9uPlxuICogICAgIDxwPkFkZGl0aW9uYWwgaW5mbyBoZXJlPC9wPlxuICogICA8L2Rpdj5cbiAqXG4gKiAgIDwhLS0gTWFpbiBjb250ZW50IC0tPlxuICogICA8bXktY29udGVudD48L215LWNvbnRlbnQ+XG4gKlxuICogICA8ZGl2IGV4dHJhLWZvb3Rlcj5cbiAqICAgICA8cD5Gb290ZXIgY29udGVudDwvcD5cbiAqICAgPC9kaXY+XG4gKiA8L3ZhbC1wYWdlLXRlbXBsYXRlPlxuICpcbiAqIEBpbnB1dCBwcm9wcyAtIFBhZ2UgdGVtcGxhdGUgY29uZmlndXJhdGlvblxuICogQG91dHB1dCBvbkJhY2sgLSBFbWl0cyB3aGVuIGJhY2sgYnV0dG9uIGlzIGNsaWNrZWRcbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLXBhZ2UtdGVtcGxhdGUnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbXG4gICAgQ29tbW9uTW9kdWxlLFxuICAgIElvbkhlYWRlcixcbiAgICBJb25Ub29sYmFyLFxuICAgIElvblRpdGxlLFxuICAgIEV4cGFuZGFibGVUZXh0Q29tcG9uZW50LFxuICAgIElvbkdyaWQsXG4gICAgSW9uUm93LFxuICAgIElvbkNvbCxcbiAgICBCdXR0b25Db21wb25lbnQsXG4gIF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgQGlmIChwcm9wcy5wYWdlVGl0bGUpIHtcbiAgICAgIDxpb24taGVhZGVyIFtjbGFzcy5pb24tbm8tYm9yZGVyXT1cInRydWVcIj5cbiAgICAgICAgPGlvbi10b29sYmFyIHN0eWxlPVwiLS1iYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDtcIj5cbiAgICAgICAgICA8aW9uLXRpdGxlIGNsYXNzPVwicGFnZS10aXRsZVwiIHNpemU9XCJtZWRpdW1cIj57eyBwcm9wcy5wYWdlVGl0bGUgfX08L2lvbi10aXRsZT5cbiAgICAgICAgPC9pb24tdG9vbGJhcj5cbiAgICAgIDwvaW9uLWhlYWRlcj5cbiAgICB9XG4gICAgPGlvbi1ncmlkPlxuICAgICAgPGlvbi1yb3cgY2xhc3M9XCJpb24tanVzdGlmeS1jb250ZW50LWNlbnRlciBkZXNjcmlwdGlvbi1yb3dcIj5cbiAgICAgICAgPGlvbi1jb2wgc2l6ZT1cIjEyXCIgc2l6ZS1tZD1cIjEwXCIgc2l6ZS1sZz1cIjhcIj5cbiAgICAgICAgICBAaWYgKHByb3BzLnBhZ2VEZXNjcmlwdGlvbikge1xuICAgICAgICAgICAgPGRpdiBjbGFzcz1cImRlc2NyaXB0aW9uLWNvbnRhaW5lclwiPlxuICAgICAgICAgICAgICA8dmFsLWV4cGFuZGFibGUtdGV4dFxuICAgICAgICAgICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgICAgICAgICBsaW1pdDogcHJvcHMuZGVzY3JpcHRpb25MaW1pdCB8fCAxODAsXG4gICAgICAgICAgICAgICAgICBjb250ZW50OiBwcm9wcy5wYWdlRGVzY3JpcHRpb24sXG4gICAgICAgICAgICAgICAgICBjb2xvcjogcHJvcHMuZGVzY3JpcHRpb25Db2xvciB8fCAnZGFyaycsXG4gICAgICAgICAgICAgICAgfVwiXG4gICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICB9XG4gICAgICAgICAgPG5nLWNvbnRlbnQgc2VsZWN0PVwiW2V4dHJhLWRlc2NyaXB0aW9uXVwiPjwvbmctY29udGVudD5cbiAgICAgICAgPC9pb24tY29sPlxuICAgICAgPC9pb24tcm93PlxuICAgICAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuICAgICAgPG5nLWNvbnRlbnQgc2VsZWN0PVwiW2V4dHJhLWZvb3Rlcl1cIj48L25nLWNvbnRlbnQ+XG4gICAgICBAaWYgKHByb3BzLnNob3dCYWNrQnV0dG9uKSB7XG4gICAgICAgIDxpb24tcm93IGNsYXNzPVwiaW9uLWp1c3RpZnktY29udGVudC1jZW50ZXIgYmFjay1yb3dcIj5cbiAgICAgICAgICA8aW9uLWNvbCBzaXplPVwiMTJcIiBzaXplLW1kPVwiMTBcIiBzaXplLWxnPVwiOFwiPlxuICAgICAgICAgICAgPHZhbC1idXR0b25cbiAgICAgICAgICAgICAgY2xhc3M9XCJiYWNrLWJ1dHRvblwiXG4gICAgICAgICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgICAgICAgdGV4dDogcHJvcHMuYmFja0J1dHRvblRleHQgfHwgJ1ZvbHZlcicsXG4gICAgICAgICAgICAgICAgY29sb3I6ICdkYXJrJyxcbiAgICAgICAgICAgICAgICBzaXplOiAnc21hbGwnLFxuICAgICAgICAgICAgICAgIHR5cGU6ICdidXR0b24nLFxuICAgICAgICAgICAgICAgIHN0YXRlOiAnRU5BQkxFRCcsXG4gICAgICAgICAgICAgICAgZmlsbDogJ291dGxpbmUnLFxuICAgICAgICAgICAgICAgIHNoYXBlOiAncm91bmQnLFxuICAgICAgICAgICAgICAgIGljb246IHtcbiAgICAgICAgICAgICAgICAgIG5hbWU6ICdhcnJvdy1iYWNrLW91dGxpbmUnLFxuICAgICAgICAgICAgICAgICAgc2xvdDogJ3N0YXJ0J1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVwiXG4gICAgICAgICAgICAgIChvbkNsaWNrKT1cImhhbmRsZUJhY2soKVwiXG4gICAgICAgICAgICAvPlxuICAgICAgICAgIDwvaW9uLWNvbD5cbiAgICAgICAgPC9pb24tcm93PlxuICAgICAgfVxuICAgIDwvaW9uLWdyaWQ+XG4gIGAsXG4gIHN0eWxlczogYFxuICAgIC5wYWdlLXRpdGxlIHtcbiAgICAgIG1hcmdpbi1sZWZ0OiAtNHB4O1xuICAgICAgcGFkZGluZzogMDtcbiAgICAgIGZvbnQtc2l6ZTogMi41cmVtO1xuICAgICAgZm9udC13ZWlnaHQ6IDgwMDtcbiAgICAgIHRleHQtYWxpZ246IGxlZnQ7XG4gICAgfVxuXG4gICAgLmRlc2NyaXB0aW9uLXJvdyB7XG4gICAgICBtYXJnaW4tYm90dG9tOiAxNnB4O1xuICAgIH1cblxuICAgIC5kZXNjcmlwdGlvbi1jb250YWluZXIge1xuICAgICAgbWFyZ2luLXRvcDogMXJlbTtcbiAgICB9XG5cbiAgICAuYmFjay1yb3cge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMTZweDtcbiAgICB9XG5cbiAgICAuYmFjay1idXR0b24ge1xuICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICBtYXJnaW46IDFyZW0gMDtcbiAgICB9XG4gIGAsXG59KVxuZXhwb3J0IGNsYXNzIFBhZ2VUZW1wbGF0ZUNvbXBvbmVudCB7XG4gIHByaXZhdGUgbmF2ID0gaW5qZWN0KE5hdkNvbnRyb2xsZXIpO1xuXG4gIC8qKlxuICAgKiBQYWdlIHRlbXBsYXRlIGNvbmZpZ3VyYXRpb24uXG4gICAqL1xuICBASW5wdXQoKSBwcm9wczogUGFnZVRlbXBsYXRlTWV0YWRhdGEgPSB7fTtcblxuICAvKipcbiAgICogRW1pdHMgd2hlbiB0aGUgYmFjayBidXR0b24gaXMgY2xpY2tlZC5cbiAgICovXG4gIEBPdXRwdXQoKSBvbkJhY2sgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG5cbiAgLyoqXG4gICAqIEhhbmRsZXMgYmFjayBuYXZpZ2F0aW9uLlxuICAgKi9cbiAgaGFuZGxlQmFjaygpOiB2b2lkIHtcbiAgICB0aGlzLm9uQmFjay5lbWl0KCk7XG4gICAgdGhpcy5uYXYuYmFjaygpO1xuICB9XG59XG4iXX0=
188
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZS10ZW1wbGF0ZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtdGVtcGxhdGUvcGFnZS10ZW1wbGF0ZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMvQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwyREFBMkQsQ0FBQztBQUNwRyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDdEUsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLG1DQUFtQyxDQUFDOztBQUduRTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQTRCRztBQThGSCxNQUFNLE9BQU8scUJBQXFCO0lBN0ZsQztRQThGVSxRQUFHLEdBQUcsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBRXBDOztXQUVHO1FBQ00sVUFBSyxHQUF5QixFQUFFLENBQUM7UUFFMUM7O1dBRUc7UUFDTyxXQUFNLEdBQUcsSUFBSSxZQUFZLEVBQVEsQ0FBQztLQVM3QztJQVBDOztPQUVHO0lBQ0gsVUFBVTtRQUNSLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDbkIsSUFBSSxDQUFDLEdBQUcsQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNsQixDQUFDOytHQW5CVSxxQkFBcUI7bUdBQXJCLHFCQUFxQix3SUFqRnRCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdEVCxtUkFoRUMsWUFBWSwrQkFDWixjQUFjLHlFQUNkLHVCQUF1QixtRkFDdkIsT0FBTyx3RUFDUCxNQUFNLG9EQUNOLE1BQU0sa1RBQ04sZUFBZTs7NEZBbUZOLHFCQUFxQjtrQkE3RmpDLFNBQVM7K0JBQ0UsbUJBQW1CLGNBQ2pCLElBQUksV0FDUDt3QkFDUCxZQUFZO3dCQUNaLGNBQWM7d0JBQ2QsdUJBQXVCO3dCQUN2QixPQUFPO3dCQUNQLE1BQU07d0JBQ04sTUFBTTt3QkFDTixlQUFlO3FCQUNoQixZQUNTOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdEVDs4QkErQlEsS0FBSztzQkFBYixLQUFLO2dCQUtJLE1BQU07c0JBQWYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgaW5qZWN0LCBJbnB1dCwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOYXZDb250cm9sbGVyIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXInO1xuaW1wb3J0IHsgSW9uQ29sLCBJb25HcmlkLCBJb25Sb3cgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IEV4cGFuZGFibGVUZXh0Q29tcG9uZW50IH0gZnJvbSAnLi4vLi4vbW9sZWN1bGVzL2V4cGFuZGFibGUtdGV4dC9leHBhbmRhYmxlLXRleHQuY29tcG9uZW50JztcbmltcG9ydCB7IEJ1dHRvbkNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2J1dHRvbi9idXR0b24uY29tcG9uZW50JztcbmltcG9ydCB7IFRpdGxlQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvdGl0bGUvdGl0bGUuY29tcG9uZW50JztcbmltcG9ydCB7IFBhZ2VUZW1wbGF0ZU1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogdmFsLXBhZ2UtdGVtcGxhdGVcbiAqXG4gKiBBIHBhZ2UgdGVtcGxhdGUgY29tcG9uZW50IHdpdGggdGl0bGUsIGV4cGFuZGFibGUgZGVzY3JpcHRpb24sXG4gKiBjb250ZW50IHByb2plY3Rpb24sIGFuZCBvcHRpb25hbCBiYWNrIG5hdmlnYXRpb24gYnV0dG9uLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLXBhZ2UtdGVtcGxhdGVcbiAqICAgW3Byb3BzXT1cIntcbiAqICAgICBwYWdlVGl0bGU6ICdHZXR0aW5nIFN0YXJ0ZWQnLFxuICogICAgIHBhZ2VEZXNjcmlwdGlvbjogJ0xlYXJuIGhvdyB0byB1c2Ugb3VyIGNvbXBvbmVudHMuLi4nLFxuICogICAgIHNob3dCYWNrQnV0dG9uOiB0cnVlXG4gKiAgIH1cIlxuICogPlxuICogICA8ZGl2IGV4dHJhLWRlc2NyaXB0aW9uPlxuICogICAgIDxwPkFkZGl0aW9uYWwgaW5mbyBoZXJlPC9wPlxuICogICA8L2Rpdj5cbiAqXG4gKiAgIDwhLS0gTWFpbiBjb250ZW50IC0tPlxuICogICA8bXktY29udGVudD48L215LWNvbnRlbnQ+XG4gKlxuICogICA8ZGl2IGV4dHJhLWZvb3Rlcj5cbiAqICAgICA8cD5Gb290ZXIgY29udGVudDwvcD5cbiAqICAgPC9kaXY+XG4gKiA8L3ZhbC1wYWdlLXRlbXBsYXRlPlxuICpcbiAqIEBpbnB1dCBwcm9wcyAtIFBhZ2UgdGVtcGxhdGUgY29uZmlndXJhdGlvblxuICogQG91dHB1dCBvbkJhY2sgLSBFbWl0cyB3aGVuIGJhY2sgYnV0dG9uIGlzIGNsaWNrZWRcbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLXBhZ2UtdGVtcGxhdGUnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbXG4gICAgQ29tbW9uTW9kdWxlLFxuICAgIFRpdGxlQ29tcG9uZW50LFxuICAgIEV4cGFuZGFibGVUZXh0Q29tcG9uZW50LFxuICAgIElvbkdyaWQsXG4gICAgSW9uUm93LFxuICAgIElvbkNvbCxcbiAgICBCdXR0b25Db21wb25lbnQsXG4gIF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgQGlmIChwcm9wcy5wYWdlVGl0bGUpIHtcbiAgICAgIDxkaXYgY2xhc3M9XCJwYWdlLXRpdGxlLWNvbnRhaW5lclwiPlxuICAgICAgICA8dmFsLXRpdGxlXG4gICAgICAgICAgW3Byb3BzXT1cIntcbiAgICAgICAgICAgIGNvbnRlbnQ6IHByb3BzLnBhZ2VUaXRsZSxcbiAgICAgICAgICAgIHNpemU6ICdsYXJnZScsXG4gICAgICAgICAgICBjb2xvcjogJ2RhcmsnLFxuICAgICAgICAgICAgYm9sZDogdHJ1ZVxuICAgICAgICAgIH1cIlxuICAgICAgICAvPlxuICAgICAgPC9kaXY+XG4gICAgfVxuICAgIDxpb24tZ3JpZD5cbiAgICAgIDxpb24tcm93IGNsYXNzPVwiaW9uLWp1c3RpZnktY29udGVudC1jZW50ZXIgZGVzY3JpcHRpb24tcm93XCI+XG4gICAgICAgIDxpb24tY29sIHNpemU9XCIxMlwiIHNpemUtbWQ9XCIxMFwiIHNpemUtbGc9XCI4XCI+XG4gICAgICAgICAgQGlmIChwcm9wcy5wYWdlRGVzY3JpcHRpb24pIHtcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJkZXNjcmlwdGlvbi1jb250YWluZXJcIj5cbiAgICAgICAgICAgICAgPHZhbC1leHBhbmRhYmxlLXRleHRcbiAgICAgICAgICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgICAgICAgICAgbGltaXQ6IHByb3BzLmRlc2NyaXB0aW9uTGltaXQgfHwgMTgwLFxuICAgICAgICAgICAgICAgICAgY29udGVudDogcHJvcHMucGFnZURlc2NyaXB0aW9uLFxuICAgICAgICAgICAgICAgICAgY29sb3I6IHByb3BzLmRlc2NyaXB0aW9uQ29sb3IgfHwgJ2RhcmsnLFxuICAgICAgICAgICAgICAgIH1cIlxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgfVxuICAgICAgICAgIDxuZy1jb250ZW50IHNlbGVjdD1cIltleHRyYS1kZXNjcmlwdGlvbl1cIj48L25nLWNvbnRlbnQ+XG4gICAgICAgIDwvaW9uLWNvbD5cbiAgICAgIDwvaW9uLXJvdz5cbiAgICAgIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbiAgICAgIDxuZy1jb250ZW50IHNlbGVjdD1cIltleHRyYS1mb290ZXJdXCI+PC9uZy1jb250ZW50PlxuICAgICAgQGlmIChwcm9wcy5zaG93QmFja0J1dHRvbikge1xuICAgICAgICA8aW9uLXJvdyBjbGFzcz1cImlvbi1qdXN0aWZ5LWNvbnRlbnQtY2VudGVyIGJhY2stcm93XCI+XG4gICAgICAgICAgPGlvbi1jb2wgc2l6ZT1cIjEyXCIgc2l6ZS1tZD1cIjEwXCIgc2l6ZS1sZz1cIjhcIj5cbiAgICAgICAgICAgIDx2YWwtYnV0dG9uXG4gICAgICAgICAgICAgIGNsYXNzPVwiYmFjay1idXR0b25cIlxuICAgICAgICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgICAgICAgIHRleHQ6IHByb3BzLmJhY2tCdXR0b25UZXh0IHx8ICdWb2x2ZXInLFxuICAgICAgICAgICAgICAgIGNvbG9yOiAnZGFyaycsXG4gICAgICAgICAgICAgICAgc2l6ZTogJ3NtYWxsJyxcbiAgICAgICAgICAgICAgICB0eXBlOiAnYnV0dG9uJyxcbiAgICAgICAgICAgICAgICBzdGF0ZTogJ0VOQUJMRUQnLFxuICAgICAgICAgICAgICAgIGZpbGw6ICdvdXRsaW5lJyxcbiAgICAgICAgICAgICAgICBzaGFwZTogJ3JvdW5kJyxcbiAgICAgICAgICAgICAgICBpY29uOiB7XG4gICAgICAgICAgICAgICAgICBuYW1lOiAnYXJyb3ctYmFjay1vdXRsaW5lJyxcbiAgICAgICAgICAgICAgICAgIHNsb3Q6ICdzdGFydCdcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cIlxuICAgICAgICAgICAgICAob25DbGljayk9XCJoYW5kbGVCYWNrKClcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L2lvbi1jb2w+XG4gICAgICAgIDwvaW9uLXJvdz5cbiAgICAgIH1cbiAgICA8L2lvbi1ncmlkPlxuICBgLFxuICBzdHlsZXM6IGBcbiAgICAucGFnZS10aXRsZS1jb250YWluZXIge1xuICAgICAgcGFkZGluZzogMCAxNnB4O1xuICAgICAgbWFyZ2luLWJvdHRvbTogOHB4O1xuICAgIH1cblxuICAgIC5kZXNjcmlwdGlvbi1yb3cge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMTZweDtcbiAgICB9XG5cbiAgICAuZGVzY3JpcHRpb24tY29udGFpbmVyIHtcbiAgICAgIG1hcmdpbi10b3A6IDFyZW07XG4gICAgfVxuXG4gICAgLmJhY2stcm93IHtcbiAgICAgIG1hcmdpbi1ib3R0b206IDE2cHg7XG4gICAgfVxuXG4gICAgLmJhY2stYnV0dG9uIHtcbiAgICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgICAgbWFyZ2luOiAxcmVtIDA7XG4gICAgfVxuICBgLFxufSlcbmV4cG9ydCBjbGFzcyBQYWdlVGVtcGxhdGVDb21wb25lbnQge1xuICBwcml2YXRlIG5hdiA9IGluamVjdChOYXZDb250cm9sbGVyKTtcblxuICAvKipcbiAgICogUGFnZSB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uLlxuICAgKi9cbiAgQElucHV0KCkgcHJvcHM6IFBhZ2VUZW1wbGF0ZU1ldGFkYXRhID0ge307XG5cbiAgLyoqXG4gICAqIEVtaXRzIHdoZW4gdGhlIGJhY2sgYnV0dG9uIGlzIGNsaWNrZWQuXG4gICAqL1xuICBAT3V0cHV0KCkgb25CYWNrID0gbmV3IEV2ZW50RW1pdHRlcjx2b2lkPigpO1xuXG4gIC8qKlxuICAgKiBIYW5kbGVzIGJhY2sgbmF2aWdhdGlvbi5cbiAgICovXG4gIGhhbmRsZUJhY2soKTogdm9pZCB7XG4gICAgdGhpcy5vbkJhY2suZW1pdCgpO1xuICAgIHRoaXMubmF2LmJhY2soKTtcbiAgfVxufVxuIl19
@@ -69,6 +69,29 @@ export class DeviceService {
69
69
  deleteDevice(deviceId) {
70
70
  return this.http.delete(`${this.baseUrl}/${deviceId}`);
71
71
  }
72
+ /**
73
+ * Valida un token de acción SIN ejecutarlo.
74
+ * Útil para mostrar confirmación al usuario antes de ejecutar.
75
+ * Este endpoint NO requiere autenticación.
76
+ *
77
+ * @param token Token JWT de acción
78
+ * @returns Información del token si es válido
79
+ *
80
+ * @example
81
+ * ```typescript
82
+ * const token = this.route.snapshot.queryParams['token'];
83
+ * if (token) {
84
+ * const validation = await firstValueFrom(this.deviceService.validateAction(token));
85
+ * if (validation.valid) {
86
+ * // Mostrar confirmación al usuario
87
+ * console.log(`Acción: ${validation.actionType}`);
88
+ * }
89
+ * }
90
+ * ```
91
+ */
92
+ validateAction(token) {
93
+ return this.http.post(`${this.config.apiUrl}/v2/actions/validate`, { token });
94
+ }
72
95
  /**
73
96
  * Ejecuta una acción de dispositivo desde un token de email.
74
97
  * Este endpoint NO requiere autenticación.
@@ -89,7 +112,15 @@ export class DeviceService {
89
112
  * ```
90
113
  */
91
114
  executeAction(token) {
92
- return this.http.post(`${this.config.apiUrl}/v2/auth/device-action`, { token });
115
+ // Usa el endpoint unificado de acciones
116
+ return this.http.post(`${this.config.apiUrl}/v2/actions/execute`, { token }).pipe(map(response => ({
117
+ operationId: response.operationId,
118
+ success: response.success,
119
+ message: response.message,
120
+ action: response.data?.['action'] || 'refuse',
121
+ deviceId: response.data?.['deviceId'] || '',
122
+ sessionsRevoked: response.data?.['sessionsRevoked'],
123
+ })));
93
124
  }
94
125
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DeviceService, deps: [{ token: VALTECH_AUTH_CONFIG }, { token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable }); }
95
126
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DeviceService, providedIn: 'root' }); }
@@ -101,4 +132,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
101
132
  type: Inject,
102
133
  args: [VALTECH_AUTH_CONFIG]
103
134
  }] }, { type: i1.HttpClient }] });
104
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV2aWNlLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2F1dGgvZGV2aWNlLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFHbkQsT0FBTyxFQUFFLEdBQUcsRUFBYyxNQUFNLGdCQUFnQixDQUFDO0FBRWpELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLFVBQVUsQ0FBQzs7O0FBVS9DOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBeUJHO0FBRUgsTUFBTSxPQUFPLGFBQWE7SUFDeEIsWUFDdUMsTUFBeUIsRUFDdEQsSUFBZ0I7UUFEYSxXQUFNLEdBQU4sTUFBTSxDQUFtQjtRQUN0RCxTQUFJLEdBQUosSUFBSSxDQUFZO0lBQ3ZCLENBQUM7SUFFSixJQUFZLE9BQU87UUFDakIsT0FBTyxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxzQkFBc0IsQ0FBQztJQUNyRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxXQUFXO1FBQ1QsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBc0IsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FDMUQsR0FBRyxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUNsQyxDQUFDO0lBQ0osQ0FBQztJQUVEOztPQUVHO0lBQ0gsU0FBUyxDQUFDLFFBQWdCO1FBQ3hCLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQ2xCLEdBQUcsSUFBSSxDQUFDLE9BQU8sSUFBSSxRQUFRLEVBQUUsQ0FDOUIsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQUVEOzs7T0FHRztJQUNILFdBQVcsQ0FBQyxRQUFnQjtRQUMxQixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUNuQixHQUFHLElBQUksQ0FBQyxPQUFPLElBQUksUUFBUSxRQUFRLEVBQ25DLEVBQUUsQ0FDSCxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7T0FHRztJQUNILGFBQWEsQ0FBQyxRQUFnQjtRQUM1QixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUNuQixHQUFHLElBQUksQ0FBQyxPQUFPLElBQUksUUFBUSxVQUFVLEVBQ3JDLEVBQUUsQ0FDSCxDQUFDO0lBQ0osQ0FBQztJQUVEOztPQUVHO0lBQ0gsWUFBWSxDQUFDLFFBQWdCO1FBQzNCLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQXFCLEdBQUcsSUFBSSxDQUFDLE9BQU8sSUFBSSxRQUFRLEVBQUUsQ0FBQyxDQUFDO0lBQzdFLENBQUM7SUFFRDs7Ozs7Ozs7Ozs7Ozs7Ozs7O09Ba0JHO0lBQ0gsYUFBYSxDQUFDLEtBQWE7UUFDekIsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FDbkIsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sd0JBQXdCLEVBQzdDLEVBQUUsS0FBSyxFQUF5QixDQUNqQyxDQUFDO0lBQ0osQ0FBQzsrR0FqRlUsYUFBYSxrQkFFZCxtQkFBbUI7bUhBRmxCLGFBQWEsY0FEQSxNQUFNOzs0RkFDbkIsYUFBYTtrQkFEekIsVUFBVTttQkFBQyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUU7OzBCQUc3QixNQUFNOzJCQUFDLG1CQUFtQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUsIEluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSHR0cENsaWVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcbmltcG9ydCB7IE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IG1hcCwgY2F0Y2hFcnJvciB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuaW1wb3J0IHsgVkFMVEVDSF9BVVRIX0NPTkZJRyB9IGZyb20gJy4vY29uZmlnJztcbmltcG9ydCB7XG4gIFZhbHRlY2hBdXRoQ29uZmlnLFxuICBEZXZpY2VJbmZvLFxuICBMaXN0RGV2aWNlc1Jlc3BvbnNlLFxuICBEZXZpY2VBY3Rpb25SZXN1bHQsXG4gIERldmljZUFjdGlvblJlcXVlc3QsXG4gIERldmljZUFjdGlvblJlc3BvbnNlLFxufSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiBTZXJ2aWNpbyBwYXJhIGdlc3Rpw7NuIGRlIGRpc3Bvc2l0aXZvcyBkZWwgdXN1YXJpby5cbiAqIFBlcm1pdGUgbGlzdGFyLCBhcHJvYmFyLCBibG9xdWVhciB5IGVsaW1pbmFyIGRpc3Bvc2l0aXZvcyByZWdpc3RyYWRvcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0IHsgRGV2aWNlU2VydmljZSB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogQENvbXBvbmVudCh7Li4ufSlcbiAqIGV4cG9ydCBjbGFzcyBEZXZpY2VzUGFnZSB7XG4gKiAgIHByaXZhdGUgZGV2aWNlU2VydmljZSA9IGluamVjdChEZXZpY2VTZXJ2aWNlKTtcbiAqXG4gKiAgIGRldmljZXMgPSBzaWduYWw8RGV2aWNlSW5mb1tdPihbXSk7XG4gKlxuICogICBhc3luYyBuZ09uSW5pdCgpIHtcbiAqICAgICBjb25zdCBkZXZpY2VzID0gYXdhaXQgZmlyc3RWYWx1ZUZyb20odGhpcy5kZXZpY2VTZXJ2aWNlLmxpc3REZXZpY2VzKCkpO1xuICogICAgIHRoaXMuZGV2aWNlcy5zZXQoZGV2aWNlcyk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIGJsb2NrRGV2aWNlKGRldmljZUlkOiBzdHJpbmcpIHtcbiAqICAgICBhd2FpdCBmaXJzdFZhbHVlRnJvbSh0aGlzLmRldmljZVNlcnZpY2UuYmxvY2tEZXZpY2UoZGV2aWNlSWQpKTtcbiAqICAgICAvLyBSZWNhcmdhciBsaXN0YVxuICogICB9XG4gKiB9XG4gKiBgYGBcbiAqL1xuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBEZXZpY2VTZXJ2aWNlIHtcbiAgY29uc3RydWN0b3IoXG4gICAgQEluamVjdChWQUxURUNIX0FVVEhfQ09ORklHKSBwcml2YXRlIGNvbmZpZzogVmFsdGVjaEF1dGhDb25maWcsXG4gICAgcHJpdmF0ZSBodHRwOiBIdHRwQ2xpZW50XG4gICkge31cblxuICBwcml2YXRlIGdldCBiYXNlVXJsKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIGAke3RoaXMuY29uZmlnLmFwaVVybH0vdjIvdXNlcnMvbWUvZGV2aWNlc2A7XG4gIH1cblxuICAvKipcbiAgICogTGlzdGEgdG9kb3MgbG9zIGRpc3Bvc2l0aXZvcyByZWdpc3RyYWRvcyBkZWwgdXN1YXJpby5cbiAgICovXG4gIGxpc3REZXZpY2VzKCk6IE9ic2VydmFibGU8RGV2aWNlSW5mb1tdPiB7XG4gICAgcmV0dXJuIHRoaXMuaHR0cC5nZXQ8TGlzdERldmljZXNSZXNwb25zZT4odGhpcy5iYXNlVXJsKS5waXBlKFxuICAgICAgbWFwKHJlc3BvbnNlID0+IHJlc3BvbnNlLmRldmljZXMpXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPYnRpZW5lIGluZm9ybWFjacOzbiBkZSB1biBkaXNwb3NpdGl2byBlc3BlY8OtZmljby5cbiAgICovXG4gIGdldERldmljZShkZXZpY2VJZDogc3RyaW5nKTogT2JzZXJ2YWJsZTxEZXZpY2VJbmZvPiB7XG4gICAgcmV0dXJuIHRoaXMuaHR0cC5nZXQ8eyBvcGVyYXRpb25JZDogc3RyaW5nOyBkZXZpY2U6IERldmljZUluZm8gfT4oXG4gICAgICBgJHt0aGlzLmJhc2VVcmx9LyR7ZGV2aWNlSWR9YFxuICAgICkucGlwZShtYXAocmVzcG9uc2UgPT4gcmVzcG9uc2UuZGV2aWNlKSk7XG4gIH1cblxuICAvKipcbiAgICogQmxvcXVlYSB1biBkaXNwb3NpdGl2by5cbiAgICogUmV2b2NhIHRvZGFzIGxhcyBzZXNpb25lcyBhY3RpdmFzIGRlIGVzZSBkaXNwb3NpdGl2by5cbiAgICovXG4gIGJsb2NrRGV2aWNlKGRldmljZUlkOiBzdHJpbmcpOiBPYnNlcnZhYmxlPERldmljZUFjdGlvblJlc3VsdD4ge1xuICAgIHJldHVybiB0aGlzLmh0dHAucG9zdDxEZXZpY2VBY3Rpb25SZXN1bHQ+KFxuICAgICAgYCR7dGhpcy5iYXNlVXJsfS8ke2RldmljZUlkfS9ibG9ja2AsXG4gICAgICB7fVxuICAgICk7XG4gIH1cblxuICAvKipcbiAgICogQXBydWViYSB1biBkaXNwb3NpdGl2byBwZW5kaWVudGUuXG4gICAqIENhbWJpYSBlbCBlc3RhZG8gZGUgcGVuZGluZ19hcHByb3ZhbCBhIGFjdGl2ZS5cbiAgICovXG4gIGFwcHJvdmVEZXZpY2UoZGV2aWNlSWQ6IHN0cmluZyk6IE9ic2VydmFibGU8RGV2aWNlQWN0aW9uUmVzdWx0PiB7XG4gICAgcmV0dXJuIHRoaXMuaHR0cC5wb3N0PERldmljZUFjdGlvblJlc3VsdD4oXG4gICAgICBgJHt0aGlzLmJhc2VVcmx9LyR7ZGV2aWNlSWR9L2FwcHJvdmVgLFxuICAgICAge31cbiAgICApO1xuICB9XG5cbiAgLyoqXG4gICAqIEVsaW1pbmEgdW4gZGlzcG9zaXRpdm8gcmVnaXN0cmFkby5cbiAgICovXG4gIGRlbGV0ZURldmljZShkZXZpY2VJZDogc3RyaW5nKTogT2JzZXJ2YWJsZTxEZXZpY2VBY3Rpb25SZXN1bHQ+IHtcbiAgICByZXR1cm4gdGhpcy5odHRwLmRlbGV0ZTxEZXZpY2VBY3Rpb25SZXN1bHQ+KGAke3RoaXMuYmFzZVVybH0vJHtkZXZpY2VJZH1gKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFamVjdXRhIHVuYSBhY2Npw7NuIGRlIGRpc3Bvc2l0aXZvIGRlc2RlIHVuIHRva2VuIGRlIGVtYWlsLlxuICAgKiBFc3RlIGVuZHBvaW50IE5PIHJlcXVpZXJlIGF1dGVudGljYWNpw7NuLlxuICAgKiBFbCB0b2tlbiB2aWVuZSBlbiBsYSBVUkwgZGVsIGVtYWlsIGRlIGFsZXJ0YSBkZSBudWV2byBpbmljaW8gZGUgc2VzacOzbi5cbiAgICpcbiAgICogQHBhcmFtIHRva2VuIFRva2VuIEpXVCBkZSBhY2Npw7NuICgyNGgsIHVuIHNvbG8gdXNvKVxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIC8vIEVuIGxhIHDDoWdpbmEgZGUgZGlzcG9zaXRpdm9zLCBhbCBkZXRlY3RhciA/dG9rZW49eHh4IGVuIGxhIFVSTDpcbiAgICogY29uc3QgdG9rZW4gPSB0aGlzLnJvdXRlLnNuYXBzaG90LnF1ZXJ5UGFyYW1zWyd0b2tlbiddO1xuICAgKiBpZiAodG9rZW4pIHtcbiAgICogICBjb25zdCByZXN1bHQgPSBhd2FpdCBmaXJzdFZhbHVlRnJvbSh0aGlzLmRldmljZVNlcnZpY2UuZXhlY3V0ZUFjdGlvbih0b2tlbikpO1xuICAgKiAgIGlmIChyZXN1bHQuc3VjY2Vzcykge1xuICAgKiAgICAgY29uc29sZS5sb2coYERpc3Bvc2l0aXZvIGJsb3F1ZWFkbywgJHtyZXN1bHQuc2Vzc2lvbnNSZXZva2VkfSBzZXNpb25lcyBjZXJyYWRhc2ApO1xuICAgKiAgIH1cbiAgICogfVxuICAgKiBgYGBcbiAgICovXG4gIGV4ZWN1dGVBY3Rpb24odG9rZW46IHN0cmluZyk6IE9ic2VydmFibGU8RGV2aWNlQWN0aW9uUmVzcG9uc2U+IHtcbiAgICByZXR1cm4gdGhpcy5odHRwLnBvc3Q8RGV2aWNlQWN0aW9uUmVzcG9uc2U+KFxuICAgICAgYCR7dGhpcy5jb25maWcuYXBpVXJsfS92Mi9hdXRoL2RldmljZS1hY3Rpb25gLFxuICAgICAgeyB0b2tlbiB9IGFzIERldmljZUFjdGlvblJlcXVlc3RcbiAgICApO1xuICB9XG59XG4iXX0=
135
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"device.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/device.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAGnD,OAAO,EAAE,GAAG,EAAc,MAAM,gBAAgB,CAAC;AAEjD,OAAO,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;;;AAW/C;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AAEH,MAAM,OAAO,aAAa;IACxB,YACuC,MAAyB,EACtD,IAAgB;QADa,WAAM,GAAN,MAAM,CAAmB;QACtD,SAAI,GAAJ,IAAI,CAAY;IACvB,CAAC;IAEJ,IAAY,OAAO;QACjB,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,sBAAsB,CAAC;IACrD,CAAC;IAED;;OAEG;IACH,WAAW;QACT,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAsB,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,CAC1D,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,CAClC,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,QAAgB;QACxB,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAClB,GAAG,IAAI,CAAC,OAAO,IAAI,QAAQ,EAAE,CAC9B,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;IAC3C,CAAC;IAED;;;OAGG;IACH,WAAW,CAAC,QAAgB;QAC1B,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CACnB,GAAG,IAAI,CAAC,OAAO,IAAI,QAAQ,QAAQ,EACnC,EAAE,CACH,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,aAAa,CAAC,QAAgB;QAC5B,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CACnB,GAAG,IAAI,CAAC,OAAO,IAAI,QAAQ,UAAU,EACrC,EAAE,CACH,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,YAAY,CAAC,QAAgB;QAC3B,OAAO,IAAI,CAAC,IAAI,CAAC,MAAM,CAAqB,GAAG,IAAI,CAAC,OAAO,IAAI,QAAQ,EAAE,CAAC,CAAC;IAC7E,CAAC;IAED;;;;;;;;;;;;;;;;;;;OAmBG;IACH,cAAc,CAAC,KAAa;QAC1B,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CACnB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,sBAAsB,EAC3C,EAAE,KAAK,EAAE,CACV,CAAC;IACJ,CAAC;IAED;;;;;;;;;;;;;;;;;;OAkBG;IACH,aAAa,CAAC,KAAa;QACzB,wCAAwC;QACxC,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CACnB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,qBAAqB,EAC1C,EAAE,KAAK,EAAE,CACV,CAAC,IAAI,CACJ,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;YACf,WAAW,EAAE,QAAQ,CAAC,WAAW;YACjC,OAAO,EAAE,QAAQ,CAAC,OAAO;YACzB,OAAO,EAAE,QAAQ,CAAC,OAAO;YACzB,MAAM,EAAG,QAAQ,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAY,IAAI,QAAQ;YACzD,QAAQ,EAAG,QAAQ,CAAC,IAAI,EAAE,CAAC,UAAU,CAAY,IAAI,EAAE;YACvD,eAAe,EAAE,QAAQ,CAAC,IAAI,EAAE,CAAC,iBAAiB,CAAuB;SAC1E,CAAC,CAAC,CACJ,CAAC;IACJ,CAAC;+GAtHU,aAAa,kBAEd,mBAAmB;mHAFlB,aAAa,cADA,MAAM;;4FACnB,aAAa;kBADzB,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAG7B,MAAM;2BAAC,mBAAmB","sourcesContent":["import { Injectable, Inject } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { Observable } from 'rxjs';\nimport { map, catchError } from 'rxjs/operators';\n\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport {\n  ValtechAuthConfig,\n  DeviceInfo,\n  ListDevicesResponse,\n  DeviceActionResult,\n  DeviceActionRequest,\n  DeviceActionResponse,\n  ValidateActionResponse,\n} from './types';\n\n/**\n * Servicio para gestión de dispositivos del usuario.\n * Permite listar, aprobar, bloquear y eliminar dispositivos registrados.\n *\n * @example\n * ```typescript\n * import { DeviceService } from 'valtech-components';\n *\n * @Component({...})\n * export class DevicesPage {\n *   private deviceService = inject(DeviceService);\n *\n *   devices = signal<DeviceInfo[]>([]);\n *\n *   async ngOnInit() {\n *     const devices = await firstValueFrom(this.deviceService.listDevices());\n *     this.devices.set(devices);\n *   }\n *\n *   async blockDevice(deviceId: string) {\n *     await firstValueFrom(this.deviceService.blockDevice(deviceId));\n *     // Recargar lista\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class DeviceService {\n  constructor(\n    @Inject(VALTECH_AUTH_CONFIG) private config: ValtechAuthConfig,\n    private http: HttpClient\n  ) {}\n\n  private get baseUrl(): string {\n    return `${this.config.apiUrl}/v2/users/me/devices`;\n  }\n\n  /**\n   * Lista todos los dispositivos registrados del usuario.\n   */\n  listDevices(): Observable<DeviceInfo[]> {\n    return this.http.get<ListDevicesResponse>(this.baseUrl).pipe(\n      map(response => response.devices)\n    );\n  }\n\n  /**\n   * Obtiene información de un dispositivo específico.\n   */\n  getDevice(deviceId: string): Observable<DeviceInfo> {\n    return this.http.get<{ operationId: string; device: DeviceInfo }>(\n      `${this.baseUrl}/${deviceId}`\n    ).pipe(map(response => response.device));\n  }\n\n  /**\n   * Bloquea un dispositivo.\n   * Revoca todas las sesiones activas de ese dispositivo.\n   */\n  blockDevice(deviceId: string): Observable<DeviceActionResult> {\n    return this.http.post<DeviceActionResult>(\n      `${this.baseUrl}/${deviceId}/block`,\n      {}\n    );\n  }\n\n  /**\n   * Aprueba un dispositivo pendiente.\n   * Cambia el estado de pending_approval a active.\n   */\n  approveDevice(deviceId: string): Observable<DeviceActionResult> {\n    return this.http.post<DeviceActionResult>(\n      `${this.baseUrl}/${deviceId}/approve`,\n      {}\n    );\n  }\n\n  /**\n   * Elimina un dispositivo registrado.\n   */\n  deleteDevice(deviceId: string): Observable<DeviceActionResult> {\n    return this.http.delete<DeviceActionResult>(`${this.baseUrl}/${deviceId}`);\n  }\n\n  /**\n   * Valida un token de acción SIN ejecutarlo.\n   * Útil para mostrar confirmación al usuario antes de ejecutar.\n   * Este endpoint NO requiere autenticación.\n   *\n   * @param token Token JWT de acción\n   * @returns Información del token si es válido\n   *\n   * @example\n   * ```typescript\n   * const token = this.route.snapshot.queryParams['token'];\n   * if (token) {\n   *   const validation = await firstValueFrom(this.deviceService.validateAction(token));\n   *   if (validation.valid) {\n   *     // Mostrar confirmación al usuario\n   *     console.log(`Acción: ${validation.actionType}`);\n   *   }\n   * }\n   * ```\n   */\n  validateAction(token: string): Observable<ValidateActionResponse> {\n    return this.http.post<ValidateActionResponse>(\n      `${this.config.apiUrl}/v2/actions/validate`,\n      { token }\n    );\n  }\n\n  /**\n   * Ejecuta una acción de dispositivo desde un token de email.\n   * Este endpoint NO requiere autenticación.\n   * El token viene en la URL del email de alerta de nuevo inicio de sesión.\n   *\n   * @param token Token JWT de acción (24h, un solo uso)\n   *\n   * @example\n   * ```typescript\n   * // En la página de dispositivos, al detectar ?token=xxx en la URL:\n   * const token = this.route.snapshot.queryParams['token'];\n   * if (token) {\n   *   const result = await firstValueFrom(this.deviceService.executeAction(token));\n   *   if (result.success) {\n   *     console.log(`Dispositivo bloqueado, ${result.sessionsRevoked} sesiones cerradas`);\n   *   }\n   * }\n   * ```\n   */\n  executeAction(token: string): Observable<DeviceActionResponse> {\n    // Usa el endpoint unificado de acciones\n    return this.http.post<{ operationId: string; success: boolean; message: string; data?: Record<string, unknown> }>(\n      `${this.config.apiUrl}/v2/actions/execute`,\n      { token }\n    ).pipe(\n      map(response => ({\n        operationId: response.operationId,\n        success: response.success,\n        message: response.message,\n        action: (response.data?.['action'] as string) || 'refuse',\n        deviceId: (response.data?.['deviceId'] as string) || '',\n        sessionsRevoked: response.data?.['sessionsRevoked'] as number | undefined,\n      }))\n    );\n  }\n}\n"]}
@@ -26,4 +26,4 @@ export const INITIAL_MFA_STATE = {
26
26
  mfaToken: null,
27
27
  method: null,
28
28
  };
29
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/types.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAuFH;;GAEG;AACH,MAAM,CAAC,MAAM,kBAAkB,GAAc;IAC3C,eAAe,EAAE,KAAK;IACtB,SAAS,EAAE,IAAI;IACf,WAAW,EAAE,IAAI;IACjB,YAAY,EAAE,IAAI;IAClB,MAAM,EAAE,IAAI;IACZ,KAAK,EAAE,IAAI;IACX,KAAK,EAAE,EAAE;IACT,WAAW,EAAE,EAAE;IACf,YAAY,EAAE,KAAK;IACnB,SAAS,EAAE,IAAI;IACf,KAAK,EAAE,IAAI;CACZ,CAAC;AAkBF;;GAEG;AACH,MAAM,CAAC,MAAM,iBAAiB,GAAoB;IAChD,QAAQ,EAAE,KAAK;IACf,QAAQ,EAAE,IAAI;IACd,MAAM,EAAE,IAAI;CACb,CAAC","sourcesContent":["/**\n * Tipos e interfaces para el servicio de autenticación de Valtech.\n * Alineados con el backend AuthV2.\n */\n\n// =============================================================================\n// CONFIGURACIÓN\n// =============================================================================\n\n/**\n * Configuración para el servicio de autenticación.\n */\nexport interface ValtechAuthConfig {\n  /** URL base de la API (ej: 'https://api.myvaltech.com') */\n  apiUrl: string;\n  /** Prefijo para endpoints de auth (default: '/v2/auth') */\n  authPrefix?: string;\n  /** Prefijo para las claves de localStorage (default: 'valtech_auth_') */\n  storagePrefix?: string;\n  /** Tiempo antes de expiración para refrescar token en segundos (default: 60) */\n  refreshBeforeExpiry?: number;\n  /** Habilitar sincronización entre pestañas (default: true) */\n  enableTabSync?: boolean;\n  /** Ruta de redirección cuando no autenticado (default: '/login') */\n  loginRoute?: string;\n  /** Ruta de redirección cuando ya autenticado (default: '/') */\n  homeRoute?: string;\n  /** Ruta para acceso denegado (default: '/unauthorized') */\n  unauthorizedRoute?: string;\n  /** Habilitar integración con FirebaseService (default: false) */\n  enableFirebaseIntegration?: boolean;\n  /** Habilitar registro automático de device tokens para push notifications (default: false) */\n  enableDeviceRegistration?: boolean;\n}\n\n// =============================================================================\n// ESTADO DE AUTENTICACIÓN\n// =============================================================================\n\n/**\n * Estado completo de autenticación.\n */\nexport interface AuthState {\n  /** Usuario está autenticado */\n  isAuthenticated: boolean;\n  /** Estado de carga inicial */\n  isLoading: boolean;\n  /** Token de acceso actual */\n  accessToken: string | null;\n  /** Token de refresco actual */\n  refreshToken: string | null;\n  /** ID del usuario */\n  userId: string | null;\n  /** Email del usuario */\n  email: string | null;\n  /** Roles del usuario */\n  roles: string[];\n  /** Permisos del usuario (formato 'resource:action') */\n  permissions: string[];\n  /** Usuario es super admin */\n  isSuperAdmin: boolean;\n  /** Timestamp de expiración del accessToken (ms) */\n  expiresAt: number | null;\n  /** Error de autenticación (si existe) */\n  error: AuthError | null;\n}\n\n/**\n * Información del usuario autenticado.\n */\nexport interface AuthUser {\n  userId: string;\n  email: string;\n  name?: string;\n  phone?: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n  mfaEnabled?: boolean;\n  mfaMethod?: MFAMethod;\n}\n\n/**\n * Error de autenticación.\n */\nexport interface AuthError {\n  code: string;\n  message: string;\n}\n\n/**\n * Estado inicial de autenticación.\n */\nexport const INITIAL_AUTH_STATE: AuthState = {\n  isAuthenticated: false,\n  isLoading: true,\n  accessToken: null,\n  refreshToken: null,\n  userId: null,\n  email: null,\n  roles: [],\n  permissions: [],\n  isSuperAdmin: false,\n  expiresAt: null,\n  error: null,\n};\n\n// =============================================================================\n// MFA\n// =============================================================================\n\n/** Métodos de MFA soportados */\nexport type MFAMethod = 'EMAIL' | 'SMS' | 'TOTP';\n\n/**\n * Estado de MFA pendiente.\n */\nexport interface MFAPendingState {\n  required: boolean;\n  mfaToken: string | null;\n  method: MFAMethod | null;\n}\n\n/**\n * Estado inicial de MFA.\n */\nexport const INITIAL_MFA_STATE: MFAPendingState = {\n  required: false,\n  mfaToken: null,\n  method: null,\n};\n\n/**\n * Resultado de setup de MFA.\n */\nexport interface MFASetupResult {\n  codeSent: boolean;\n  message: string;\n}\n\n/**\n * Estado de MFA del usuario.\n */\nexport interface MFAStatus {\n  enabled: boolean;\n  method: MFAMethod | null;\n}\n\n// =============================================================================\n// TOTP MFA (Google Authenticator)\n// =============================================================================\n\n/**\n * Response de setup de TOTP.\n */\nexport interface TOTPSetupResponse {\n  operationId: string;\n  /** Secreto TOTP en base32 (para entrada manual) */\n  secret: string;\n  /** URL para generar QR code (otpauth://...) */\n  qrCodeUrl: string;\n  /** Códigos de respaldo de un solo uso */\n  backupCodes: string[];\n  message: string;\n}\n\n/**\n * Request para verificar setup de TOTP.\n */\nexport interface TOTPVerifySetupRequest {\n  /** Código TOTP de 6 dígitos */\n  code: string;\n}\n\n/**\n * Response de verificación de setup TOTP.\n */\nexport interface TOTPVerifySetupResponse {\n  operationId: string;\n  enabled: boolean;\n  message: string;\n}\n\n/**\n * Request para deshabilitar TOTP.\n */\nexport interface TOTPDisableRequest {\n  password: string;\n}\n\n/**\n * Response de deshabilitar TOTP.\n */\nexport interface TOTPDisableResponse {\n  operationId: string;\n  disabled: boolean;\n  message: string;\n}\n\n/**\n * Response de regenerar códigos de respaldo.\n */\nexport interface RegenerateBackupCodesResponse {\n  operationId: string;\n  backupCodes: string[];\n  message: string;\n}\n\n/**\n * Response de cantidad de códigos de respaldo.\n */\nexport interface BackupCodesCountResponse {\n  operationId: string;\n  count: number;\n}\n\n// =============================================================================\n// REQUESTS/RESPONSES (alineados con backend AuthV2)\n// =============================================================================\n\n/**\n * Request para signup (registro).\n */\nexport interface SignupRequest {\n  email: string;\n  password: string;\n  name: string;\n  phone?: string;\n}\n\n/**\n * Response de signup (registro).\n */\nexport interface SignupResponse {\n  operationId: string;\n  userId: string;\n  message: string;\n}\n\n/**\n * Request para verificar email.\n */\nexport interface VerifyEmailRequest {\n  email: string;\n  code: string;\n}\n\n/**\n * Response de verificación de email.\n * Si es exitoso, incluye tokens para auto-login.\n */\nexport interface VerifyEmailResponse {\n  operationId: string;\n  verified: boolean;\n  accessToken?: string;\n  refreshToken?: string;\n  firebaseToken?: string;\n  expiresIn?: number;\n  tokenType?: string;\n}\n\n/**\n * Request para reenviar código de verificación.\n */\nexport interface ResendCodeRequest {\n  email: string;\n  type: 'EMAIL_VERIFY' | 'PASSWORD_RESET';\n}\n\n/**\n * Response de reenvío de código.\n */\nexport interface ResendCodeResponse {\n  operationId: string;\n  sent: boolean;\n}\n\n/**\n * Request para signin.\n */\nexport interface SigninRequest {\n  email: string;\n  password: string;\n  /** Plataforma del dispositivo (para detección de nuevo dispositivo) */\n  platform?: DevicePlatform;\n  /** Navegador (solo web) */\n  browser?: string;\n  /** Sistema operativo */\n  os?: string;\n  /** Nombre amigable del dispositivo */\n  deviceName?: string;\n}\n\n/**\n * Response de signin.\n */\nexport interface SigninResponse {\n  operationId: string;\n  accessToken?: string;\n  refreshToken?: string;\n  firebaseToken?: string;\n  expiresIn?: number;\n  tokenType?: string;\n  mfaRequired?: boolean;\n  mfaToken?: string;\n  mfaMethod?: MFAMethod;\n  roles?: string[];\n  permissions?: string[];\n  /** Indica si el login fue desde un dispositivo nuevo/no reconocido */\n  isNewDevice?: boolean;\n  /** ID del dispositivo usado para el login */\n  deviceId?: string;\n}\n\n/**\n * Request para verificar MFA.\n */\nexport interface MFAVerifyRequest {\n  mfaToken: string;\n  code: string;\n}\n\n/**\n * Response de verificar MFA.\n */\nexport interface MFAVerifyResponse {\n  operationId: string;\n  accessToken: string;\n  refreshToken: string;\n  firebaseToken?: string;\n  expiresIn: number;\n  tokenType: string;\n  roles?: string[];\n  permissions?: string[];\n}\n\n/**\n * Request para refrescar token.\n */\nexport interface RefreshRequest {\n  refreshToken: string;\n}\n\n/**\n * Response de refrescar token.\n * Implementa token rotation: cada refresh genera un nuevo refresh token.\n */\nexport interface RefreshResponse {\n  operationId: string;\n  accessToken: string;\n  refreshToken: string; // Token rotation: nuevo refresh token\n  expiresIn: number;\n  firebaseToken?: string;\n  roles?: string[];\n  permissions?: string[];\n}\n\n/**\n * Request para logout.\n */\nexport interface LogoutRequest {\n  refreshToken: string;\n}\n\n/**\n * Response de logout.\n */\nexport interface LogoutResponse {\n  operationId: string;\n  success: boolean;\n}\n\n/**\n * Response de obtener permisos.\n */\nexport interface GetPermissionsResponse {\n  operationId: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n}\n\n/**\n * Request para setup de MFA.\n */\nexport interface MFASetupRequest {\n  method: MFAMethod;\n  phone?: string;\n}\n\n/**\n * Response de setup de MFA.\n */\nexport interface MFASetupResponse {\n  operationId: string;\n  codeSent: boolean;\n  message: string;\n}\n\n/**\n * Request para confirmar MFA.\n */\nexport interface MFAConfirmRequest {\n  code: string;\n}\n\n/**\n * Response de confirmar MFA.\n */\nexport interface MFAConfirmResponse {\n  operationId: string;\n  mfaEnabled: boolean;\n  method: MFAMethod;\n}\n\n/**\n * Request para deshabilitar MFA.\n */\nexport interface MFADisableRequest {\n  password: string;\n}\n\n/**\n * Response de deshabilitar MFA.\n */\nexport interface MFADisableResponse {\n  operationId: string;\n  mfaDisabled: boolean;\n}\n\n/**\n * Request para iniciar recuperación de contraseña.\n */\nexport interface ForgotPasswordRequest {\n  email: string;\n}\n\n/**\n * Response de forgot password (siempre igual por seguridad).\n */\nexport interface ForgotPasswordResponse {\n  operationId: string;\n  message: string;\n}\n\n/**\n * Request para resetear contraseña con código.\n */\nexport interface ResetPasswordRequest {\n  email: string;\n  code: string;\n  newPassword: string;\n}\n\n/**\n * Response de reset password.\n */\nexport interface ResetPasswordResponse {\n  operationId: string;\n  success: boolean;\n}\n\n/**\n * Response de obtener perfil del usuario.\n */\nexport interface GetProfileResponse {\n  operationId: string;\n  userId: string;\n  email: string;\n  name: string;\n  phone?: string;\n  emailVerified: boolean;\n  phoneVerified: boolean;\n  mfaEnabled: boolean;\n  mfaMethod?: MFAMethod;\n  createdAt: string;\n  updatedAt: string;\n}\n\n/**\n * Request para actualizar perfil.\n */\nexport interface UpdateProfileRequest {\n  name?: string;\n  phone?: string;\n}\n\n/**\n * Response de actualizar perfil.\n */\nexport interface UpdateProfileResponse {\n  operationId: string;\n  updated: boolean;\n}\n\n// =============================================================================\n// EVENTOS DE SINCRONIZACIÓN\n// =============================================================================\n\n// =============================================================================\n// SWITCH ORGANIZATION\n// =============================================================================\n\n/**\n * Request para cambiar de organización activa.\n */\nexport interface SwitchOrgRequest {\n  organizationId: string;\n}\n\n/**\n * Response de cambio de organización.\n */\nexport interface SwitchOrgResponse {\n  operationId: string;\n  firebaseToken: string;\n  activeOrg: string;\n}\n\n// =============================================================================\n// EVENTOS DE SINCRONIZACIÓN\n// =============================================================================\n\n/** Tipos de eventos de sincronización entre pestañas */\nexport type AuthSyncEventType =\n  | 'LOGIN'\n  | 'LOGOUT'\n  | 'TOKEN_REFRESH'\n  | 'PERMISSIONS_UPDATE'\n  | 'ORG_SWITCH';\n\n/**\n * Evento de sincronización entre pestañas.\n */\nexport interface AuthSyncEvent {\n  type: AuthSyncEventType;\n  timestamp: number;\n  payload?: {\n    accessToken?: string;\n    refreshToken?: string; // Token rotation support\n    expiresAt?: number;\n    activeOrg?: string;\n  };\n}\n\n// =============================================================================\n// JWT CLAIMS\n// =============================================================================\n\n/**\n * Claims del JWT de acceso.\n */\nexport interface JWTClaims {\n  /** User ID */\n  uid: string;\n  /** Email */\n  email: string;\n  /** Session ID */\n  sid?: string;\n  /** Issued at */\n  iat: number;\n  /** Expiration */\n  exp: number;\n}\n\n// =============================================================================\n// STORAGE\n// =============================================================================\n\n/**\n * Datos persistidos en storage.\n */\nexport interface StoredAuthState {\n  accessToken: string;\n  refreshToken: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n  expiresAt?: number;\n}\n\n// =============================================================================\n// DEVICE REGISTRATION (Push Notifications)\n// =============================================================================\n\n/**\n * Plataformas soportadas para push notifications.\n */\nexport type DevicePlatform = 'web' | 'ios' | 'android';\n\n/**\n * Request para registrar un dispositivo.\n */\nexport interface RegisterDeviceRequest {\n  /** FCM token del dispositivo */\n  token: string;\n  /** Plataforma del dispositivo */\n  platform: DevicePlatform;\n  /** Navegador (solo web) */\n  browser?: string;\n  /** Sistema operativo */\n  os?: string;\n  /** Versión de la aplicación */\n  appVersion?: string;\n  /** Nombre amigable del dispositivo */\n  name?: string;\n  /** Metadata adicional */\n  metadata?: Record<string, string>;\n}\n\n/**\n * Response de registro de dispositivo.\n */\nexport interface RegisterDeviceResponse {\n  operationId: string;\n  device: {\n    deviceId: string;\n    token: string;\n    platform: DevicePlatform;\n    browser?: string;\n    os?: string;\n    lastActive: string;\n    createdAt: string;\n  };\n  /** true si es un nuevo dispositivo, false si se actualizó uno existente */\n  isNew: boolean;\n}\n\n/**\n * Resultado de habilitar notificaciones.\n */\nexport interface EnableNotificationsResult {\n  /** Si se otorgaron los permisos */\n  granted: boolean;\n  /** FCM token (solo si granted=true) */\n  token?: string;\n  /** Si el dispositivo fue registrado en el backend */\n  registered?: boolean;\n}\n\n/**\n * Estado de permisos de notificación.\n */\nexport type NotificationPermissionState = 'granted' | 'denied' | 'default' | 'unsupported';\n\n/**\n * Resultado del registro manual de dispositivo.\n */\nexport interface RegisterDeviceResult {\n  /** Si el dispositivo fue registrado exitosamente */\n  registered: boolean;\n  /** ID del dispositivo (solo si registered=true) */\n  deviceId?: string;\n  /** FCM token usado para el registro */\n  token?: string;\n  /** Mensaje de error (solo si registered=false) */\n  error?: string;\n}\n\n// =============================================================================\n// DEVICE MANAGEMENT (Detección de dispositivos nuevos)\n// =============================================================================\n\n/**\n * Estados posibles de un dispositivo.\n */\nexport type DeviceStatus = 'active' | 'pending_approval' | 'blocked';\n\n/**\n * Información completa de un dispositivo registrado.\n */\nexport interface DeviceInfo {\n  /** ID único del dispositivo */\n  deviceId: string;\n  /** Plataforma del dispositivo */\n  platform: DevicePlatform;\n  /** Navegador (solo web) */\n  browser?: string;\n  /** Sistema operativo */\n  os?: string;\n  /** Nombre amigable del dispositivo */\n  name?: string;\n  /** Estado actual del dispositivo */\n  status: DeviceStatus;\n  /** Dirección IP del último acceso */\n  ipAddress?: string;\n  /** Ubicación geográfica aproximada */\n  location?: string;\n  /** Última actividad */\n  lastActive: string;\n  /** Fecha de creación/registro */\n  createdAt: string;\n  /** Fecha de aprobación (si aplica) */\n  approvedAt?: string;\n  /** Fecha de bloqueo (si aplica) */\n  blockedAt?: string;\n}\n\n/**\n * Response de listar dispositivos del usuario.\n */\nexport interface ListDevicesResponse {\n  operationId: string;\n  devices: DeviceInfo[];\n}\n\n/**\n * Response de acción sobre dispositivo (block/approve/delete).\n */\nexport interface DeviceActionResult {\n  operationId: string;\n  success: boolean;\n  message: string;\n}\n\n// =============================================================================\n// SESSION MANAGEMENT (Gestión de sesiones activas)\n// =============================================================================\n\n/**\n * Información de una sesión activa.\n */\nexport interface SessionInfo {\n  /** ID único de la sesión */\n  sessionId: string;\n  /** ID del dispositivo asociado (si existe) */\n  deviceId?: string;\n  /** Información del dispositivo (User-Agent) */\n  deviceInfo?: string;\n  /** Dirección IP de la sesión */\n  ip?: string;\n  /** Ubicación geográfica aproximada */\n  location?: string;\n  /** Fecha de creación de la sesión */\n  createdAt: string;\n  /** Fecha de expiración de la sesión */\n  expiresAt: string;\n  /** Indica si es la sesión actual del usuario */\n  isCurrent: boolean;\n}\n\n/**\n * Response de listar sesiones activas.\n */\nexport interface ListSessionsResponse {\n  operationId: string;\n  sessions: SessionInfo[];\n}\n\n/**\n * Response de revocar sesión(es).\n */\nexport interface RevokeSessionsResponse {\n  operationId: string;\n  success: boolean;\n  /** Número de sesiones revocadas */\n  sessionsRevoked: number;\n}\n\n// =============================================================================\n// DEVICE ACTION (Acciones desde email de alerta)\n// =============================================================================\n\n/**\n * Request para ejecutar acción de dispositivo desde email.\n * El token viene en la URL del email de alerta.\n */\nexport interface DeviceActionRequest {\n  /** Token JWT de acción (24h, un solo uso) */\n  token: string;\n}\n\n/**\n * Response de ejecutar acción de dispositivo.\n */\nexport interface DeviceActionResponse {\n  operationId: string;\n  /** Si la acción fue ejecutada exitosamente */\n  success: boolean;\n  /** Acción ejecutada (refuse, approve, block) */\n  action: string;\n  /** ID del dispositivo afectado */\n  deviceId: string;\n  /** Número de sesiones revocadas (si action=refuse/block) */\n  sessionsRevoked?: number;\n  /** Mensaje descriptivo */\n  message: string;\n}\n"]}
29
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/types.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAuFH;;GAEG;AACH,MAAM,CAAC,MAAM,kBAAkB,GAAc;IAC3C,eAAe,EAAE,KAAK;IACtB,SAAS,EAAE,IAAI;IACf,WAAW,EAAE,IAAI;IACjB,YAAY,EAAE,IAAI;IAClB,MAAM,EAAE,IAAI;IACZ,KAAK,EAAE,IAAI;IACX,KAAK,EAAE,EAAE;IACT,WAAW,EAAE,EAAE;IACf,YAAY,EAAE,KAAK;IACnB,SAAS,EAAE,IAAI;IACf,KAAK,EAAE,IAAI;CACZ,CAAC;AAkBF;;GAEG;AACH,MAAM,CAAC,MAAM,iBAAiB,GAAoB;IAChD,QAAQ,EAAE,KAAK;IACf,QAAQ,EAAE,IAAI;IACd,MAAM,EAAE,IAAI;CACb,CAAC","sourcesContent":["/**\n * Tipos e interfaces para el servicio de autenticación de Valtech.\n * Alineados con el backend AuthV2.\n */\n\n// =============================================================================\n// CONFIGURACIÓN\n// =============================================================================\n\n/**\n * Configuración para el servicio de autenticación.\n */\nexport interface ValtechAuthConfig {\n  /** URL base de la API (ej: 'https://api.myvaltech.com') */\n  apiUrl: string;\n  /** Prefijo para endpoints de auth (default: '/v2/auth') */\n  authPrefix?: string;\n  /** Prefijo para las claves de localStorage (default: 'valtech_auth_') */\n  storagePrefix?: string;\n  /** Tiempo antes de expiración para refrescar token en segundos (default: 60) */\n  refreshBeforeExpiry?: number;\n  /** Habilitar sincronización entre pestañas (default: true) */\n  enableTabSync?: boolean;\n  /** Ruta de redirección cuando no autenticado (default: '/login') */\n  loginRoute?: string;\n  /** Ruta de redirección cuando ya autenticado (default: '/') */\n  homeRoute?: string;\n  /** Ruta para acceso denegado (default: '/unauthorized') */\n  unauthorizedRoute?: string;\n  /** Habilitar integración con FirebaseService (default: false) */\n  enableFirebaseIntegration?: boolean;\n  /** Habilitar registro automático de device tokens para push notifications (default: false) */\n  enableDeviceRegistration?: boolean;\n}\n\n// =============================================================================\n// ESTADO DE AUTENTICACIÓN\n// =============================================================================\n\n/**\n * Estado completo de autenticación.\n */\nexport interface AuthState {\n  /** Usuario está autenticado */\n  isAuthenticated: boolean;\n  /** Estado de carga inicial */\n  isLoading: boolean;\n  /** Token de acceso actual */\n  accessToken: string | null;\n  /** Token de refresco actual */\n  refreshToken: string | null;\n  /** ID del usuario */\n  userId: string | null;\n  /** Email del usuario */\n  email: string | null;\n  /** Roles del usuario */\n  roles: string[];\n  /** Permisos del usuario (formato 'resource:action') */\n  permissions: string[];\n  /** Usuario es super admin */\n  isSuperAdmin: boolean;\n  /** Timestamp de expiración del accessToken (ms) */\n  expiresAt: number | null;\n  /** Error de autenticación (si existe) */\n  error: AuthError | null;\n}\n\n/**\n * Información del usuario autenticado.\n */\nexport interface AuthUser {\n  userId: string;\n  email: string;\n  name?: string;\n  phone?: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n  mfaEnabled?: boolean;\n  mfaMethod?: MFAMethod;\n}\n\n/**\n * Error de autenticación.\n */\nexport interface AuthError {\n  code: string;\n  message: string;\n}\n\n/**\n * Estado inicial de autenticación.\n */\nexport const INITIAL_AUTH_STATE: AuthState = {\n  isAuthenticated: false,\n  isLoading: true,\n  accessToken: null,\n  refreshToken: null,\n  userId: null,\n  email: null,\n  roles: [],\n  permissions: [],\n  isSuperAdmin: false,\n  expiresAt: null,\n  error: null,\n};\n\n// =============================================================================\n// MFA\n// =============================================================================\n\n/** Métodos de MFA soportados */\nexport type MFAMethod = 'EMAIL' | 'SMS' | 'TOTP';\n\n/**\n * Estado de MFA pendiente.\n */\nexport interface MFAPendingState {\n  required: boolean;\n  mfaToken: string | null;\n  method: MFAMethod | null;\n}\n\n/**\n * Estado inicial de MFA.\n */\nexport const INITIAL_MFA_STATE: MFAPendingState = {\n  required: false,\n  mfaToken: null,\n  method: null,\n};\n\n/**\n * Resultado de setup de MFA.\n */\nexport interface MFASetupResult {\n  codeSent: boolean;\n  message: string;\n}\n\n/**\n * Estado de MFA del usuario.\n */\nexport interface MFAStatus {\n  enabled: boolean;\n  method: MFAMethod | null;\n}\n\n// =============================================================================\n// TOTP MFA (Google Authenticator)\n// =============================================================================\n\n/**\n * Response de setup de TOTP.\n */\nexport interface TOTPSetupResponse {\n  operationId: string;\n  /** Secreto TOTP en base32 (para entrada manual) */\n  secret: string;\n  /** URL para generar QR code (otpauth://...) */\n  qrCodeUrl: string;\n  /** Códigos de respaldo de un solo uso */\n  backupCodes: string[];\n  message: string;\n}\n\n/**\n * Request para verificar setup de TOTP.\n */\nexport interface TOTPVerifySetupRequest {\n  /** Código TOTP de 6 dígitos */\n  code: string;\n}\n\n/**\n * Response de verificación de setup TOTP.\n */\nexport interface TOTPVerifySetupResponse {\n  operationId: string;\n  enabled: boolean;\n  message: string;\n}\n\n/**\n * Request para deshabilitar TOTP.\n */\nexport interface TOTPDisableRequest {\n  password: string;\n}\n\n/**\n * Response de deshabilitar TOTP.\n */\nexport interface TOTPDisableResponse {\n  operationId: string;\n  disabled: boolean;\n  message: string;\n}\n\n/**\n * Response de regenerar códigos de respaldo.\n */\nexport interface RegenerateBackupCodesResponse {\n  operationId: string;\n  backupCodes: string[];\n  message: string;\n}\n\n/**\n * Response de cantidad de códigos de respaldo.\n */\nexport interface BackupCodesCountResponse {\n  operationId: string;\n  count: number;\n}\n\n// =============================================================================\n// REQUESTS/RESPONSES (alineados con backend AuthV2)\n// =============================================================================\n\n/**\n * Request para signup (registro).\n */\nexport interface SignupRequest {\n  email: string;\n  password: string;\n  name: string;\n  phone?: string;\n}\n\n/**\n * Response de signup (registro).\n */\nexport interface SignupResponse {\n  operationId: string;\n  userId: string;\n  message: string;\n}\n\n/**\n * Request para verificar email.\n */\nexport interface VerifyEmailRequest {\n  email: string;\n  code: string;\n}\n\n/**\n * Response de verificación de email.\n * Si es exitoso, incluye tokens para auto-login.\n */\nexport interface VerifyEmailResponse {\n  operationId: string;\n  verified: boolean;\n  accessToken?: string;\n  refreshToken?: string;\n  firebaseToken?: string;\n  expiresIn?: number;\n  tokenType?: string;\n}\n\n/**\n * Request para reenviar código de verificación.\n */\nexport interface ResendCodeRequest {\n  email: string;\n  type: 'EMAIL_VERIFY' | 'PASSWORD_RESET';\n}\n\n/**\n * Response de reenvío de código.\n */\nexport interface ResendCodeResponse {\n  operationId: string;\n  sent: boolean;\n}\n\n/**\n * Request para signin.\n */\nexport interface SigninRequest {\n  email: string;\n  password: string;\n  /** Plataforma del dispositivo (para detección de nuevo dispositivo) */\n  platform?: DevicePlatform;\n  /** Navegador (solo web) */\n  browser?: string;\n  /** Sistema operativo */\n  os?: string;\n  /** Nombre amigable del dispositivo */\n  deviceName?: string;\n}\n\n/**\n * Response de signin.\n */\nexport interface SigninResponse {\n  operationId: string;\n  accessToken?: string;\n  refreshToken?: string;\n  firebaseToken?: string;\n  expiresIn?: number;\n  tokenType?: string;\n  mfaRequired?: boolean;\n  mfaToken?: string;\n  mfaMethod?: MFAMethod;\n  roles?: string[];\n  permissions?: string[];\n  /** Indica si el login fue desde un dispositivo nuevo/no reconocido */\n  isNewDevice?: boolean;\n  /** ID del dispositivo usado para el login */\n  deviceId?: string;\n}\n\n/**\n * Request para verificar MFA.\n */\nexport interface MFAVerifyRequest {\n  mfaToken: string;\n  code: string;\n}\n\n/**\n * Response de verificar MFA.\n */\nexport interface MFAVerifyResponse {\n  operationId: string;\n  accessToken: string;\n  refreshToken: string;\n  firebaseToken?: string;\n  expiresIn: number;\n  tokenType: string;\n  roles?: string[];\n  permissions?: string[];\n}\n\n/**\n * Request para refrescar token.\n */\nexport interface RefreshRequest {\n  refreshToken: string;\n}\n\n/**\n * Response de refrescar token.\n * Implementa token rotation: cada refresh genera un nuevo refresh token.\n */\nexport interface RefreshResponse {\n  operationId: string;\n  accessToken: string;\n  refreshToken: string; // Token rotation: nuevo refresh token\n  expiresIn: number;\n  firebaseToken?: string;\n  roles?: string[];\n  permissions?: string[];\n}\n\n/**\n * Request para logout.\n */\nexport interface LogoutRequest {\n  refreshToken: string;\n}\n\n/**\n * Response de logout.\n */\nexport interface LogoutResponse {\n  operationId: string;\n  success: boolean;\n}\n\n/**\n * Response de obtener permisos.\n */\nexport interface GetPermissionsResponse {\n  operationId: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n}\n\n/**\n * Request para setup de MFA.\n */\nexport interface MFASetupRequest {\n  method: MFAMethod;\n  phone?: string;\n}\n\n/**\n * Response de setup de MFA.\n */\nexport interface MFASetupResponse {\n  operationId: string;\n  codeSent: boolean;\n  message: string;\n}\n\n/**\n * Request para confirmar MFA.\n */\nexport interface MFAConfirmRequest {\n  code: string;\n}\n\n/**\n * Response de confirmar MFA.\n */\nexport interface MFAConfirmResponse {\n  operationId: string;\n  mfaEnabled: boolean;\n  method: MFAMethod;\n}\n\n/**\n * Request para deshabilitar MFA.\n */\nexport interface MFADisableRequest {\n  password: string;\n}\n\n/**\n * Response de deshabilitar MFA.\n */\nexport interface MFADisableResponse {\n  operationId: string;\n  mfaDisabled: boolean;\n}\n\n/**\n * Request para iniciar recuperación de contraseña.\n */\nexport interface ForgotPasswordRequest {\n  email: string;\n}\n\n/**\n * Response de forgot password (siempre igual por seguridad).\n */\nexport interface ForgotPasswordResponse {\n  operationId: string;\n  message: string;\n}\n\n/**\n * Request para resetear contraseña con código.\n */\nexport interface ResetPasswordRequest {\n  email: string;\n  code: string;\n  newPassword: string;\n}\n\n/**\n * Response de reset password.\n */\nexport interface ResetPasswordResponse {\n  operationId: string;\n  success: boolean;\n}\n\n/**\n * Response de obtener perfil del usuario.\n */\nexport interface GetProfileResponse {\n  operationId: string;\n  userId: string;\n  email: string;\n  name: string;\n  phone?: string;\n  emailVerified: boolean;\n  phoneVerified: boolean;\n  mfaEnabled: boolean;\n  mfaMethod?: MFAMethod;\n  createdAt: string;\n  updatedAt: string;\n}\n\n/**\n * Request para actualizar perfil.\n */\nexport interface UpdateProfileRequest {\n  name?: string;\n  phone?: string;\n}\n\n/**\n * Response de actualizar perfil.\n */\nexport interface UpdateProfileResponse {\n  operationId: string;\n  updated: boolean;\n}\n\n// =============================================================================\n// EVENTOS DE SINCRONIZACIÓN\n// =============================================================================\n\n// =============================================================================\n// SWITCH ORGANIZATION\n// =============================================================================\n\n/**\n * Request para cambiar de organización activa.\n */\nexport interface SwitchOrgRequest {\n  organizationId: string;\n}\n\n/**\n * Response de cambio de organización.\n */\nexport interface SwitchOrgResponse {\n  operationId: string;\n  firebaseToken: string;\n  activeOrg: string;\n}\n\n// =============================================================================\n// EVENTOS DE SINCRONIZACIÓN\n// =============================================================================\n\n/** Tipos de eventos de sincronización entre pestañas */\nexport type AuthSyncEventType =\n  | 'LOGIN'\n  | 'LOGOUT'\n  | 'TOKEN_REFRESH'\n  | 'PERMISSIONS_UPDATE'\n  | 'ORG_SWITCH';\n\n/**\n * Evento de sincronización entre pestañas.\n */\nexport interface AuthSyncEvent {\n  type: AuthSyncEventType;\n  timestamp: number;\n  payload?: {\n    accessToken?: string;\n    refreshToken?: string; // Token rotation support\n    expiresAt?: number;\n    activeOrg?: string;\n  };\n}\n\n// =============================================================================\n// JWT CLAIMS\n// =============================================================================\n\n/**\n * Claims del JWT de acceso.\n */\nexport interface JWTClaims {\n  /** User ID */\n  uid: string;\n  /** Email */\n  email: string;\n  /** Session ID */\n  sid?: string;\n  /** Issued at */\n  iat: number;\n  /** Expiration */\n  exp: number;\n}\n\n// =============================================================================\n// STORAGE\n// =============================================================================\n\n/**\n * Datos persistidos en storage.\n */\nexport interface StoredAuthState {\n  accessToken: string;\n  refreshToken: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n  expiresAt?: number;\n}\n\n// =============================================================================\n// DEVICE REGISTRATION (Push Notifications)\n// =============================================================================\n\n/**\n * Plataformas soportadas para push notifications.\n */\nexport type DevicePlatform = 'web' | 'ios' | 'android';\n\n/**\n * Request para registrar un dispositivo.\n */\nexport interface RegisterDeviceRequest {\n  /** FCM token del dispositivo */\n  token: string;\n  /** Plataforma del dispositivo */\n  platform: DevicePlatform;\n  /** Navegador (solo web) */\n  browser?: string;\n  /** Sistema operativo */\n  os?: string;\n  /** Versión de la aplicación */\n  appVersion?: string;\n  /** Nombre amigable del dispositivo */\n  name?: string;\n  /** Metadata adicional */\n  metadata?: Record<string, string>;\n}\n\n/**\n * Response de registro de dispositivo.\n */\nexport interface RegisterDeviceResponse {\n  operationId: string;\n  device: {\n    deviceId: string;\n    token: string;\n    platform: DevicePlatform;\n    browser?: string;\n    os?: string;\n    lastActive: string;\n    createdAt: string;\n  };\n  /** true si es un nuevo dispositivo, false si se actualizó uno existente */\n  isNew: boolean;\n}\n\n/**\n * Resultado de habilitar notificaciones.\n */\nexport interface EnableNotificationsResult {\n  /** Si se otorgaron los permisos */\n  granted: boolean;\n  /** FCM token (solo si granted=true) */\n  token?: string;\n  /** Si el dispositivo fue registrado en el backend */\n  registered?: boolean;\n}\n\n/**\n * Estado de permisos de notificación.\n */\nexport type NotificationPermissionState = 'granted' | 'denied' | 'default' | 'unsupported';\n\n/**\n * Resultado del registro manual de dispositivo.\n */\nexport interface RegisterDeviceResult {\n  /** Si el dispositivo fue registrado exitosamente */\n  registered: boolean;\n  /** ID del dispositivo (solo si registered=true) */\n  deviceId?: string;\n  /** FCM token usado para el registro */\n  token?: string;\n  /** Mensaje de error (solo si registered=false) */\n  error?: string;\n}\n\n// =============================================================================\n// DEVICE MANAGEMENT (Detección de dispositivos nuevos)\n// =============================================================================\n\n/**\n * Estados posibles de un dispositivo.\n */\nexport type DeviceStatus = 'active' | 'pending_approval' | 'blocked';\n\n/**\n * Información completa de un dispositivo registrado.\n */\nexport interface DeviceInfo {\n  /** ID único del dispositivo */\n  deviceId: string;\n  /** Plataforma del dispositivo */\n  platform: DevicePlatform;\n  /** Navegador (solo web) */\n  browser?: string;\n  /** Sistema operativo */\n  os?: string;\n  /** Nombre amigable del dispositivo */\n  name?: string;\n  /** Estado actual del dispositivo */\n  status: DeviceStatus;\n  /** Dirección IP del último acceso */\n  ipAddress?: string;\n  /** Ubicación geográfica aproximada */\n  location?: string;\n  /** Última actividad */\n  lastActive: string;\n  /** Fecha de creación/registro */\n  createdAt: string;\n  /** Fecha de aprobación (si aplica) */\n  approvedAt?: string;\n  /** Fecha de bloqueo (si aplica) */\n  blockedAt?: string;\n}\n\n/**\n * Response de listar dispositivos del usuario.\n */\nexport interface ListDevicesResponse {\n  operationId: string;\n  devices: DeviceInfo[];\n}\n\n/**\n * Response de acción sobre dispositivo (block/approve/delete).\n */\nexport interface DeviceActionResult {\n  operationId: string;\n  success: boolean;\n  message: string;\n}\n\n// =============================================================================\n// SESSION MANAGEMENT (Gestión de sesiones activas)\n// =============================================================================\n\n/**\n * Información de una sesión activa.\n */\nexport interface SessionInfo {\n  /** ID único de la sesión */\n  sessionId: string;\n  /** ID del dispositivo asociado (si existe) */\n  deviceId?: string;\n  /** Información del dispositivo (User-Agent) */\n  deviceInfo?: string;\n  /** Dirección IP de la sesión */\n  ip?: string;\n  /** Ubicación geográfica aproximada */\n  location?: string;\n  /** Fecha de creación de la sesión */\n  createdAt: string;\n  /** Fecha de expiración de la sesión */\n  expiresAt: string;\n  /** Indica si es la sesión actual del usuario */\n  isCurrent: boolean;\n}\n\n/**\n * Response de listar sesiones activas.\n */\nexport interface ListSessionsResponse {\n  operationId: string;\n  sessions: SessionInfo[];\n}\n\n/**\n * Response de revocar sesión(es).\n */\nexport interface RevokeSessionsResponse {\n  operationId: string;\n  success: boolean;\n  /** Número de sesiones revocadas */\n  sessionsRevoked: number;\n}\n\n// =============================================================================\n// DEVICE ACTION (Acciones desde email de alerta)\n// =============================================================================\n\n/**\n * Request para ejecutar acción de dispositivo desde email.\n * El token viene en la URL del email de alerta.\n */\nexport interface DeviceActionRequest {\n  /** Token JWT de acción (24h, un solo uso) */\n  token: string;\n}\n\n/**\n * Response de ejecutar acción de dispositivo.\n */\nexport interface DeviceActionResponse {\n  operationId: string;\n  /** Si la acción fue ejecutada exitosamente */\n  success: boolean;\n  /** Acción ejecutada (refuse, approve, block) */\n  action: string;\n  /** ID del dispositivo afectado */\n  deviceId: string;\n  /** Número de sesiones revocadas (si action=refuse/block) */\n  sessionsRevoked?: number;\n  /** Mensaje descriptivo */\n  message: string;\n}\n\n/**\n * Response de validar token de acción (sin ejecutar).\n * Usado para mostrar confirmación al usuario antes de ejecutar.\n */\nexport interface ValidateActionResponse {\n  operationId: string;\n  /** Si el token es válido */\n  valid: boolean;\n  /** Tipo de acción (device:refuse, device:approve, etc.) */\n  actionType?: string;\n  /** Tipo de recurso objetivo (device, document, etc.) */\n  targetType?: string;\n  /** ID del recurso objetivo */\n  targetId?: string;\n  /** Mensaje de error (si valid=false) */\n  message?: string;\n}\n"]}
@@ -20105,11 +20105,16 @@ class PageTemplateComponent {
20105
20105
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageTemplateComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
20106
20106
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageTemplateComponent, isStandalone: true, selector: "val-page-template", inputs: { props: "props" }, outputs: { onBack: "onBack" }, ngImport: i0, template: `
20107
20107
  @if (props.pageTitle) {
20108
- <ion-header [class.ion-no-border]="true">
20109
- <ion-toolbar style="--background: transparent;">
20110
- <ion-title class="page-title" size="medium">{{ props.pageTitle }}</ion-title>
20111
- </ion-toolbar>
20112
- </ion-header>
20108
+ <div class="page-title-container">
20109
+ <val-title
20110
+ [props]="{
20111
+ content: props.pageTitle,
20112
+ size: 'large',
20113
+ color: 'dark',
20114
+ bold: true
20115
+ }"
20116
+ />
20117
+ </div>
20113
20118
  }
20114
20119
  <ion-grid>
20115
20120
  <ion-row class="ion-justify-content-center description-row">
@@ -20154,15 +20159,13 @@ class PageTemplateComponent {
20154
20159
  </ion-row>
20155
20160
  }
20156
20161
  </ion-grid>
20157
- `, isInline: true, styles: [".page-title{margin-left:-4px;padding:0;font-size:2.5rem;font-weight:800;text-align:left}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: ExpandableTextComponent, selector: "val-expandable-text", inputs: ["props"] }, { kind: "component", type: IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { kind: "component", type: IonRow, selector: "ion-row" }, { kind: "component", type: IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: ButtonComponent, selector: "val-button", inputs: ["props"], outputs: ["onClick"] }] }); }
20162
+ `, isInline: true, styles: [".page-title-container{padding:0 16px;margin-bottom:8px}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: TitleComponent, selector: "val-title", inputs: ["props"] }, { kind: "component", type: ExpandableTextComponent, selector: "val-expandable-text", inputs: ["props"] }, { kind: "component", type: IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { kind: "component", type: IonRow, selector: "ion-row" }, { kind: "component", type: IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: ButtonComponent, selector: "val-button", inputs: ["props"], outputs: ["onClick"] }] }); }
20158
20163
  }
20159
20164
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageTemplateComponent, decorators: [{
20160
20165
  type: Component,
20161
20166
  args: [{ selector: 'val-page-template', standalone: true, imports: [
20162
20167
  CommonModule,
20163
- IonHeader,
20164
- IonToolbar,
20165
- IonTitle,
20168
+ TitleComponent,
20166
20169
  ExpandableTextComponent,
20167
20170
  IonGrid,
20168
20171
  IonRow,
@@ -20170,11 +20173,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
20170
20173
  ButtonComponent,
20171
20174
  ], template: `
20172
20175
  @if (props.pageTitle) {
20173
- <ion-header [class.ion-no-border]="true">
20174
- <ion-toolbar style="--background: transparent;">
20175
- <ion-title class="page-title" size="medium">{{ props.pageTitle }}</ion-title>
20176
- </ion-toolbar>
20177
- </ion-header>
20176
+ <div class="page-title-container">
20177
+ <val-title
20178
+ [props]="{
20179
+ content: props.pageTitle,
20180
+ size: 'large',
20181
+ color: 'dark',
20182
+ bold: true
20183
+ }"
20184
+ />
20185
+ </div>
20178
20186
  }
20179
20187
  <ion-grid>
20180
20188
  <ion-row class="ion-justify-content-center description-row">
@@ -20219,7 +20227,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
20219
20227
  </ion-row>
20220
20228
  }
20221
20229
  </ion-grid>
20222
- `, styles: [".page-title{margin-left:-4px;padding:0;font-size:2.5rem;font-weight:800;text-align:left}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"] }]
20230
+ `, styles: [".page-title-container{padding:0 16px;margin-bottom:8px}.description-row{margin-bottom:16px}.description-container{margin-top:1rem}.back-row{margin-bottom:16px}.back-button{display:block;margin:1rem 0}\n"] }]
20223
20231
  }], propDecorators: { props: [{
20224
20232
  type: Input
20225
20233
  }], onBack: [{
@@ -26264,6 +26272,29 @@ class DeviceService {
26264
26272
  deleteDevice(deviceId) {
26265
26273
  return this.http.delete(`${this.baseUrl}/${deviceId}`);
26266
26274
  }
26275
+ /**
26276
+ * Valida un token de acción SIN ejecutarlo.
26277
+ * Útil para mostrar confirmación al usuario antes de ejecutar.
26278
+ * Este endpoint NO requiere autenticación.
26279
+ *
26280
+ * @param token Token JWT de acción
26281
+ * @returns Información del token si es válido
26282
+ *
26283
+ * @example
26284
+ * ```typescript
26285
+ * const token = this.route.snapshot.queryParams['token'];
26286
+ * if (token) {
26287
+ * const validation = await firstValueFrom(this.deviceService.validateAction(token));
26288
+ * if (validation.valid) {
26289
+ * // Mostrar confirmación al usuario
26290
+ * console.log(`Acción: ${validation.actionType}`);
26291
+ * }
26292
+ * }
26293
+ * ```
26294
+ */
26295
+ validateAction(token) {
26296
+ return this.http.post(`${this.config.apiUrl}/v2/actions/validate`, { token });
26297
+ }
26267
26298
  /**
26268
26299
  * Ejecuta una acción de dispositivo desde un token de email.
26269
26300
  * Este endpoint NO requiere autenticación.
@@ -26284,7 +26315,15 @@ class DeviceService {
26284
26315
  * ```
26285
26316
  */
26286
26317
  executeAction(token) {
26287
- return this.http.post(`${this.config.apiUrl}/v2/auth/device-action`, { token });
26318
+ // Usa el endpoint unificado de acciones
26319
+ return this.http.post(`${this.config.apiUrl}/v2/actions/execute`, { token }).pipe(map$1(response => ({
26320
+ operationId: response.operationId,
26321
+ success: response.success,
26322
+ message: response.message,
26323
+ action: response.data?.['action'] || 'refuse',
26324
+ deviceId: response.data?.['deviceId'] || '',
26325
+ sessionsRevoked: response.data?.['sessionsRevoked'],
26326
+ })));
26288
26327
  }
26289
26328
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DeviceService, deps: [{ token: VALTECH_AUTH_CONFIG }, { token: i1$8.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable }); }
26290
26329
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DeviceService, providedIn: 'root' }); }