valtech-components 2.0.524 → 2.0.525

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,5 +1,5 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, Input } from '@angular/core';
2
+ import { Component, computed, input } from '@angular/core';
3
3
  import { IonCol, IonGrid, IonRow } from '@ionic/angular/standalone';
4
4
  import { FooterLinksComponent } from '../../molecules/footer-links/footer-links.component';
5
5
  import { RightsFooterComponent } from '../../atoms/rights-footer/rights-footer.component';
@@ -30,22 +30,28 @@ export class CompanyFooterComponent {
30
30
  constructor() {
31
31
  /**
32
32
  * Footer configuration.
33
+ * Signal-based input for full reactivity with computed().
33
34
  */
34
- this.props = {
35
+ this.props = input({
35
36
  links: {
36
37
  leftLinks: { title: '', size: 'medium', links: [] },
37
38
  rightLinks: { title: '', size: 'medium', links: [] },
38
39
  },
39
- };
40
+ });
41
+ // Computed helpers for template bindings
42
+ this.propsWithPadding = computed(() => this.props()?.withPadding);
43
+ this.propsWithBackground = computed(() => this.props()?.withBackground);
44
+ this.propsLinks = computed(() => this.props()?.links);
45
+ this.propsRights = computed(() => this.props()?.rights || {});
40
46
  }
41
47
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CompanyFooterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
42
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CompanyFooterComponent, isStandalone: true, selector: "val-company-footer", inputs: { props: "props" }, ngImport: i0, template: `
43
- <footer [class.withPadding]="props.withPadding" [class.withBackground]="props.withBackground">
48
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: CompanyFooterComponent, isStandalone: true, selector: "val-company-footer", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: `
49
+ <footer [class.withPadding]="propsWithPadding()" [class.withBackground]="propsWithBackground()">
44
50
  <ion-grid>
45
- <val-footer-links [props]="props.links" />
51
+ <val-footer-links [props]="propsLinks()" />
46
52
  <ion-row>
47
53
  <ion-col size="12">
48
- <val-rights-footer [props]="props.rights || {}" />
54
+ <val-rights-footer [props]="propsRights()" />
49
55
  </ion-col>
50
56
  </ion-row>
51
57
  </ion-grid>
@@ -55,18 +61,16 @@ export class CompanyFooterComponent {
55
61
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CompanyFooterComponent, decorators: [{
56
62
  type: Component,
57
63
  args: [{ selector: 'val-company-footer', standalone: true, imports: [CommonModule, IonGrid, IonRow, IonCol, RightsFooterComponent, FooterLinksComponent], template: `
58
- <footer [class.withPadding]="props.withPadding" [class.withBackground]="props.withBackground">
64
+ <footer [class.withPadding]="propsWithPadding()" [class.withBackground]="propsWithBackground()">
59
65
  <ion-grid>
60
- <val-footer-links [props]="props.links" />
66
+ <val-footer-links [props]="propsLinks()" />
61
67
  <ion-row>
62
68
  <ion-col size="12">
63
- <val-rights-footer [props]="props.rights || {}" />
69
+ <val-rights-footer [props]="propsRights()" />
64
70
  </ion-col>
65
71
  </ion-row>
66
72
  </ion-grid>
67
73
  </footer>
68
74
  `, styles: [".withPadding{padding:16px}.withBackground{background-color:var(--ion-background-color)}\n"] }]
69
- }], propDecorators: { props: [{
70
- type: Input
71
- }] } });
72
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tcGFueS1mb290ZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL29yZ2FuaXNtcy9jb21wYW55LWZvb3Rlci9jb21wYW55LWZvb3Rlci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2pELE9BQU8sRUFBRSxNQUFNLEVBQUUsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ3BFLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHFEQUFxRCxDQUFDO0FBQzNGLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLG1EQUFtRCxDQUFDOztBQUcxRjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBcUJHO0FBMkJILE1BQU0sT0FBTyxzQkFBc0I7SUExQm5DO1FBMkJFOztXQUVHO1FBQ00sVUFBSyxHQUEwQjtZQUN0QyxLQUFLLEVBQUU7Z0JBQ0wsU0FBUyxFQUFFLEVBQUUsS0FBSyxFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUU7Z0JBQ25ELFVBQVUsRUFBRSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUUsSUFBSSxFQUFFLFFBQVEsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFO2FBQ3JEO1NBQ0YsQ0FBQztLQUNIOytHQVZZLHNCQUFzQjttR0FBdEIsc0JBQXNCLDBHQXRCdkI7Ozs7Ozs7Ozs7O0dBV1Qsa0tBWlMsWUFBWSwrQkFBRSxPQUFPLHdFQUFFLE1BQU0sb0RBQUUsTUFBTSxrVEFBRSxxQkFBcUIsaUZBQUUsb0JBQW9COzs0RkF1QmpGLHNCQUFzQjtrQkExQmxDLFNBQVM7K0JBQ0Usb0JBQW9CLGNBQ2xCLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxxQkFBcUIsRUFBRSxvQkFBb0IsQ0FBQyxZQUNuRjs7Ozs7Ozs7Ozs7R0FXVDs4QkFlUSxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSW9uQ29sLCBJb25HcmlkLCBJb25Sb3cgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IEZvb3RlckxpbmtzQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vbW9sZWN1bGVzL2Zvb3Rlci1saW5rcy9mb290ZXItbGlua3MuY29tcG9uZW50JztcbmltcG9ydCB7IFJpZ2h0c0Zvb3RlckNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL3JpZ2h0cy1mb290ZXIvcmlnaHRzLWZvb3Rlci5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ29tcGFueUZvb3Rlck1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogdmFsLWNvbXBhbnktZm9vdGVyXG4gKlxuICogQSBjb21wbGV0ZSBmb290ZXIgY29tcG9uZW50IGNvbWJpbmluZyBmb290ZXIgbGlua3MgKHdpdGggbG9nbyBhbmQgc29jaWFsIGljb25zKVxuICogYW5kIGEgcmlnaHRzL2NvcHlyaWdodCBmb290ZXIuIENvbmZpZ3VyYWJsZSBwYWRkaW5nIGFuZCBiYWNrZ3JvdW5kIG9wdGlvbnMuXG4gKlxuICogQGV4YW1wbGVcbiAqIDx2YWwtY29tcGFueS1mb290ZXJcbiAqICAgW3Byb3BzXT1cIntcbiAqICAgICBsaW5rczoge1xuICogICAgICAgbGVmdExpbmtzOiB7IHRpdGxlOiAnQ29tcGFueScsIHNpemU6ICdtZWRpdW0nLCBsaW5rczogWy4uLl0gfSxcbiAqICAgICAgIHJpZ2h0TGlua3M6IHsgdGl0bGU6ICdTdXBwb3J0Jywgc2l6ZTogJ21lZGl1bScsIGxpbmtzOiBbLi4uXSB9LFxuICogICAgICAgbG9nbzogJy9hc3NldHMvbG9nby5wbmcnLFxuICogICAgICAgc29jaWFsTGlua3M6IFsuLi5dXG4gKiAgICAgfSxcbiAqICAgICByaWdodHM6IHsgY29tcGFueU5hbWU6ICdBY21lIEluYycgfSxcbiAqICAgICB3aXRoUGFkZGluZzogdHJ1ZVxuICogICB9XCJcbiAqID48L3ZhbC1jb21wYW55LWZvb3Rlcj5cbiAqXG4gKiBAaW5wdXQgcHJvcHMgLSBGb290ZXIgY29uZmlndXJhdGlvblxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtY29tcGFueS1mb290ZXInLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBJb25HcmlkLCBJb25Sb3csIElvbkNvbCwgUmlnaHRzRm9vdGVyQ29tcG9uZW50LCBGb290ZXJMaW5rc0NvbXBvbmVudF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgPGZvb3RlciBbY2xhc3Mud2l0aFBhZGRpbmddPVwicHJvcHMud2l0aFBhZGRpbmdcIiBbY2xhc3Mud2l0aEJhY2tncm91bmRdPVwicHJvcHMud2l0aEJhY2tncm91bmRcIj5cbiAgICAgIDxpb24tZ3JpZD5cbiAgICAgICAgPHZhbC1mb290ZXItbGlua3MgW3Byb3BzXT1cInByb3BzLmxpbmtzXCIgLz5cbiAgICAgICAgPGlvbi1yb3c+XG4gICAgICAgICAgPGlvbi1jb2wgc2l6ZT1cIjEyXCI+XG4gICAgICAgICAgICA8dmFsLXJpZ2h0cy1mb290ZXIgW3Byb3BzXT1cInByb3BzLnJpZ2h0cyB8fCB7fVwiIC8+XG4gICAgICAgICAgPC9pb24tY29sPlxuICAgICAgICA8L2lvbi1yb3c+XG4gICAgICA8L2lvbi1ncmlkPlxuICAgIDwvZm9vdGVyPlxuICBgLFxuICBzdHlsZXM6IGBcbiAgICAud2l0aFBhZGRpbmcge1xuICAgICAgcGFkZGluZzogMTZweDtcbiAgICB9XG5cbiAgICAud2l0aEJhY2tncm91bmQge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0taW9uLWJhY2tncm91bmQtY29sb3IpO1xuICAgIH1cbiAgYCxcbn0pXG5leHBvcnQgY2xhc3MgQ29tcGFueUZvb3RlckNvbXBvbmVudCB7XG4gIC8qKlxuICAgKiBGb290ZXIgY29uZmlndXJhdGlvbi5cbiAgICovXG4gIEBJbnB1dCgpIHByb3BzOiBDb21wYW55Rm9vdGVyTWV0YWRhdGEgPSB7XG4gICAgbGlua3M6IHtcbiAgICAgIGxlZnRMaW5rczogeyB0aXRsZTogJycsIHNpemU6ICdtZWRpdW0nLCBsaW5rczogW10gfSxcbiAgICAgIHJpZ2h0TGlua3M6IHsgdGl0bGU6ICcnLCBzaXplOiAnbWVkaXVtJywgbGlua3M6IFtdIH0sXG4gICAgfSxcbiAgfTtcbn1cbiJdfQ==
75
+ }] });
76
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tcGFueS1mb290ZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL29yZ2FuaXNtcy9jb21wYW55LWZvb3Rlci9jb21wYW55LWZvb3Rlci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsUUFBUSxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzRCxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxxREFBcUQsQ0FBQztBQUMzRixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxtREFBbUQsQ0FBQzs7QUFHMUY7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXFCRztBQTJCSCxNQUFNLE9BQU8sc0JBQXNCO0lBMUJuQztRQTJCRTs7O1dBR0c7UUFDTSxVQUFLLEdBQUcsS0FBSyxDQUF3QjtZQUM1QyxLQUFLLEVBQUU7Z0JBQ0wsU0FBUyxFQUFFLEVBQUUsS0FBSyxFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUU7Z0JBQ25ELFVBQVUsRUFBRSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUUsSUFBSSxFQUFFLFFBQVEsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFO2FBQ3JEO1NBQ0YsQ0FBQyxDQUFDO1FBRUgseUNBQXlDO1FBQ3pDLHFCQUFnQixHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLEVBQUUsV0FBVyxDQUFDLENBQUM7UUFDN0Qsd0JBQW1CLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztRQUNuRSxlQUFVLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztRQUNqRCxnQkFBVyxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLEVBQUUsTUFBTSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0tBQzFEOytHQWpCWSxzQkFBc0I7bUdBQXRCLHNCQUFzQixrTkF0QnZCOzs7Ozs7Ozs7OztHQVdULGtLQVpTLFlBQVksK0JBQUUsT0FBTyx3RUFBRSxNQUFNLG9EQUFFLE1BQU0sa1RBQUUscUJBQXFCLGlGQUFFLG9CQUFvQjs7NEZBdUJqRixzQkFBc0I7a0JBMUJsQyxTQUFTOytCQUNFLG9CQUFvQixjQUNsQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUscUJBQXFCLEVBQUUsb0JBQW9CLENBQUMsWUFDbkY7Ozs7Ozs7Ozs7O0dBV1QiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBjb21wdXRlZCwgaW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IElvbkNvbCwgSW9uR3JpZCwgSW9uUm93IH0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5pbXBvcnQgeyBGb290ZXJMaW5rc0NvbXBvbmVudCB9IGZyb20gJy4uLy4uL21vbGVjdWxlcy9mb290ZXItbGlua3MvZm9vdGVyLWxpbmtzLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBSaWdodHNGb290ZXJDb21wb25lbnQgfSBmcm9tICcuLi8uLi9hdG9tcy9yaWdodHMtZm9vdGVyL3JpZ2h0cy1mb290ZXIuY29tcG9uZW50JztcbmltcG9ydCB7IENvbXBhbnlGb290ZXJNZXRhZGF0YSB9IGZyb20gJy4vdHlwZXMnO1xuXG4vKipcbiAqIHZhbC1jb21wYW55LWZvb3RlclxuICpcbiAqIEEgY29tcGxldGUgZm9vdGVyIGNvbXBvbmVudCBjb21iaW5pbmcgZm9vdGVyIGxpbmtzICh3aXRoIGxvZ28gYW5kIHNvY2lhbCBpY29ucylcbiAqIGFuZCBhIHJpZ2h0cy9jb3B5cmlnaHQgZm9vdGVyLiBDb25maWd1cmFibGUgcGFkZGluZyBhbmQgYmFja2dyb3VuZCBvcHRpb25zLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLWNvbXBhbnktZm9vdGVyXG4gKiAgIFtwcm9wc109XCJ7XG4gKiAgICAgbGlua3M6IHtcbiAqICAgICAgIGxlZnRMaW5rczogeyB0aXRsZTogJ0NvbXBhbnknLCBzaXplOiAnbWVkaXVtJywgbGlua3M6IFsuLi5dIH0sXG4gKiAgICAgICByaWdodExpbmtzOiB7IHRpdGxlOiAnU3VwcG9ydCcsIHNpemU6ICdtZWRpdW0nLCBsaW5rczogWy4uLl0gfSxcbiAqICAgICAgIGxvZ286ICcvYXNzZXRzL2xvZ28ucG5nJyxcbiAqICAgICAgIHNvY2lhbExpbmtzOiBbLi4uXVxuICogICAgIH0sXG4gKiAgICAgcmlnaHRzOiB7IGNvbXBhbnlOYW1lOiAnQWNtZSBJbmMnIH0sXG4gKiAgICAgd2l0aFBhZGRpbmc6IHRydWVcbiAqICAgfVwiXG4gKiA+PC92YWwtY29tcGFueS1mb290ZXI+XG4gKlxuICogQGlucHV0IHByb3BzIC0gRm9vdGVyIGNvbmZpZ3VyYXRpb25cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLWNvbXBhbnktZm9vdGVyJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgSW9uR3JpZCwgSW9uUm93LCBJb25Db2wsIFJpZ2h0c0Zvb3RlckNvbXBvbmVudCwgRm9vdGVyTGlua3NDb21wb25lbnRdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxmb290ZXIgW2NsYXNzLndpdGhQYWRkaW5nXT1cInByb3BzV2l0aFBhZGRpbmcoKVwiIFtjbGFzcy53aXRoQmFja2dyb3VuZF09XCJwcm9wc1dpdGhCYWNrZ3JvdW5kKClcIj5cbiAgICAgIDxpb24tZ3JpZD5cbiAgICAgICAgPHZhbC1mb290ZXItbGlua3MgW3Byb3BzXT1cInByb3BzTGlua3MoKVwiIC8+XG4gICAgICAgIDxpb24tcm93PlxuICAgICAgICAgIDxpb24tY29sIHNpemU9XCIxMlwiPlxuICAgICAgICAgICAgPHZhbC1yaWdodHMtZm9vdGVyIFtwcm9wc109XCJwcm9wc1JpZ2h0cygpXCIgLz5cbiAgICAgICAgICA8L2lvbi1jb2w+XG4gICAgICAgIDwvaW9uLXJvdz5cbiAgICAgIDwvaW9uLWdyaWQ+XG4gICAgPC9mb290ZXI+XG4gIGAsXG4gIHN0eWxlczogYFxuICAgIC53aXRoUGFkZGluZyB7XG4gICAgICBwYWRkaW5nOiAxNnB4O1xuICAgIH1cblxuICAgIC53aXRoQmFja2dyb3VuZCB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1pb24tYmFja2dyb3VuZC1jb2xvcik7XG4gICAgfVxuICBgLFxufSlcbmV4cG9ydCBjbGFzcyBDb21wYW55Rm9vdGVyQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIEZvb3RlciBjb25maWd1cmF0aW9uLlxuICAgKiBTaWduYWwtYmFzZWQgaW5wdXQgZm9yIGZ1bGwgcmVhY3Rpdml0eSB3aXRoIGNvbXB1dGVkKCkuXG4gICAqL1xuICByZWFkb25seSBwcm9wcyA9IGlucHV0PENvbXBhbnlGb290ZXJNZXRhZGF0YT4oe1xuICAgIGxpbmtzOiB7XG4gICAgICBsZWZ0TGlua3M6IHsgdGl0bGU6ICcnLCBzaXplOiAnbWVkaXVtJywgbGlua3M6IFtdIH0sXG4gICAgICByaWdodExpbmtzOiB7IHRpdGxlOiAnJywgc2l6ZTogJ21lZGl1bScsIGxpbmtzOiBbXSB9LFxuICAgIH0sXG4gIH0pO1xuXG4gIC8vIENvbXB1dGVkIGhlbHBlcnMgZm9yIHRlbXBsYXRlIGJpbmRpbmdzXG4gIHByb3BzV2l0aFBhZGRpbmcgPSBjb21wdXRlZCgoKSA9PiB0aGlzLnByb3BzKCk/LndpdGhQYWRkaW5nKTtcbiAgcHJvcHNXaXRoQmFja2dyb3VuZCA9IGNvbXB1dGVkKCgpID0+IHRoaXMucHJvcHMoKT8ud2l0aEJhY2tncm91bmQpO1xuICBwcm9wc0xpbmtzID0gY29tcHV0ZWQoKCkgPT4gdGhpcy5wcm9wcygpPy5saW5rcyk7XG4gIHByb3BzUmlnaHRzID0gY29tcHV0ZWQoKCkgPT4gdGhpcy5wcm9wcygpPy5yaWdodHMgfHwge30pO1xufVxuIl19
@@ -1,5 +1,5 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
2
+ import { Component, computed, EventEmitter, inject, input, Output } from '@angular/core';
3
3
  import { NavigationEnd, Router, RouterOutlet } from '@angular/router';
4
4
  import { IonContent } from '@ionic/angular/standalone';
5
5
  import { filter } from 'rxjs';
@@ -38,10 +38,11 @@ export class PageWrapperComponent {
38
38
  this.router = inject(Router);
39
39
  /**
40
40
  * Page wrapper configuration.
41
+ * Signal-based input for full reactivity with computed().
41
42
  */
42
- this.props = {
43
+ this.props = input({
43
44
  scrollToTopOnNavigate: true,
44
- };
45
+ });
45
46
  /**
46
47
  * Emits when a header action is clicked.
47
48
  */
@@ -79,15 +80,23 @@ export class PageWrapperComponent {
79
80
  ],
80
81
  },
81
82
  };
82
- }
83
- /**
84
- * Gets header props, using cached default if not provided.
85
- */
86
- get headerProps() {
87
- return this.props.header || this.defaultHeader;
83
+ // Computed helpers for template bindings
84
+ this.headerProps = computed(() => this.props()?.header || this.defaultHeader);
85
+ this.contentId = computed(() => this.props()?.contentId || 'page-wrapper');
86
+ this.propsFooter = computed(() => this.props()?.footer);
87
+ this.background = computed(() => {
88
+ if (this.theme.IsDark) {
89
+ return 'var(--ion-background-color)';
90
+ }
91
+ const bg = this.props()?.background;
92
+ if (!bg) {
93
+ return 'var(--ion-background-color)';
94
+ }
95
+ return resolveColor(bg);
96
+ });
88
97
  }
89
98
  ngOnInit() {
90
- if (this.props.scrollToTopOnNavigate !== false) {
99
+ if (this.props()?.scrollToTopOnNavigate !== false) {
91
100
  this.routerSubscription = this.router.events
92
101
  .pipe(filter((event) => event instanceof NavigationEnd))
93
102
  .subscribe(() => {
@@ -106,25 +115,12 @@ export class PageWrapperComponent {
106
115
  ionViewWillEnter() {
107
116
  this.scrollToTop();
108
117
  }
109
- /**
110
- * Gets the background color based on theme.
111
- */
112
- getBackground() {
113
- if (this.theme.IsDark) {
114
- return 'var(--ion-background-color)';
115
- }
116
- const bg = this.props.background;
117
- if (!bg) {
118
- return 'var(--ion-background-color)';
119
- }
120
- return resolveColor(bg);
121
- }
122
118
  /**
123
119
  * Scrolls the content to the top.
124
120
  */
125
121
  scrollToTop() {
126
- const contentId = this.props.contentId || 'page-wrapper';
127
- const ionContent = document.querySelector(`#${contentId}`);
122
+ const id = this.props()?.contentId || 'page-wrapper';
123
+ const ionContent = document.querySelector(`#${id}`);
128
124
  if (ionContent && ionContent.scrollToTop) {
129
125
  ionContent.scrollToTop(300);
130
126
  }
@@ -137,30 +133,31 @@ export class PageWrapperComponent {
137
133
  */
138
134
  onHeaderClickHandler(token) {
139
135
  this.onHeaderClick.emit(token);
140
- if (token === 'header-logo' && this.props.homeRoute) {
141
- this.nav.navigateByUrl(this.props.homeRoute);
136
+ const homeRoute = this.props()?.homeRoute;
137
+ if (token === 'header-logo' && homeRoute) {
138
+ this.nav.navigateByUrl(homeRoute);
142
139
  }
143
140
  }
144
141
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageWrapperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
145
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageWrapperComponent, isStandalone: true, selector: "val-page-wrapper", inputs: { props: "props" }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
142
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageWrapperComponent, isStandalone: true, selector: "val-page-wrapper", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
146
143
  <div class="ion-page">
147
144
  <val-header
148
- [props]="headerProps"
145
+ [props]="headerProps()"
149
146
  (onClick)="onHeaderClickHandler($event)"
150
147
  />
151
148
  <ion-content
152
- [id]="props.contentId || 'page-wrapper'"
149
+ [id]="contentId()"
153
150
  class="ion-padding"
154
151
  [fullscreen]="true"
155
152
  [ngStyle]="{
156
- '--background': getBackground()
153
+ '--background': background()
157
154
  }"
158
155
  >
159
156
  <main>
160
157
  <router-outlet></router-outlet>
161
158
  </main>
162
- @if (props.footer) {
163
- <val-company-footer [props]="props.footer" />
159
+ @if (propsFooter()) {
160
+ <val-company-footer [props]="propsFooter()" />
164
161
  }
165
162
  </ion-content>
166
163
  </div>
@@ -171,29 +168,27 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
171
168
  args: [{ selector: 'val-page-wrapper', standalone: true, imports: [CommonModule, HeaderComponent, CompanyFooterComponent, RouterOutlet, IonContent], template: `
172
169
  <div class="ion-page">
173
170
  <val-header
174
- [props]="headerProps"
171
+ [props]="headerProps()"
175
172
  (onClick)="onHeaderClickHandler($event)"
176
173
  />
177
174
  <ion-content
178
- [id]="props.contentId || 'page-wrapper'"
175
+ [id]="contentId()"
179
176
  class="ion-padding"
180
177
  [fullscreen]="true"
181
178
  [ngStyle]="{
182
- '--background': getBackground()
179
+ '--background': background()
183
180
  }"
184
181
  >
185
182
  <main>
186
183
  <router-outlet></router-outlet>
187
184
  </main>
188
- @if (props.footer) {
189
- <val-company-footer [props]="props.footer" />
185
+ @if (propsFooter()) {
186
+ <val-company-footer [props]="propsFooter()" />
190
187
  }
191
188
  </ion-content>
192
189
  </div>
193
190
  `, styles: ["main{min-height:60vh}\n"] }]
194
- }], propDecorators: { props: [{
195
- type: Input
196
- }], onHeaderClick: [{
191
+ }], propDecorators: { onHeaderClick: [{
197
192
  type: Output
198
193
  }] } });
199
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-wrapper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-wrapper/page-wrapper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { NavigationEnd, Router, RouterOutlet } from '@angular/router';\nimport { IonContent, ViewWillEnter, ViewWillLeave } from '@ionic/angular/standalone';\nimport { filter, Subscription } from 'rxjs';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { CompanyFooterComponent } from '../../organisms/company-footer/company-footer.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageWrapperMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-wrapper\n *\n * A complete page wrapper template with header, router outlet, footer,\n * and automatic scroll-to-top on navigation.\n *\n * @example\n * <val-page-wrapper\n *   [props]=\"{\n *     homeRoute: '/',\n *     scrollToTopOnNavigate: true,\n *     footer: {\n *       links: { ... },\n *       rights: { companyName: 'Acme' }\n *     }\n *   }\"\n * ></val-page-wrapper>\n *\n * @input props - Page wrapper configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-wrapper',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, CompanyFooterComponent, RouterOutlet, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [id]=\"props.contentId || 'page-wrapper'\"\n        class=\"ion-padding\"\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <router-outlet></router-outlet>\n        </main>\n        @if (props.footer) {\n          <val-company-footer [props]=\"props.footer\" />\n        }\n      </ion-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageWrapperComponent implements ViewWillEnter, ViewWillLeave, OnInit, OnDestroy {\n  private theme = inject(ThemeService);\n  private nav = inject(NavigationService);\n  private router = inject(Router);\n  private routerSubscription?: Subscription;\n\n  /**\n   * Page wrapper configuration.\n   */\n  @Input() props: PageWrapperMetadata = {\n    scrollToTopOnNavigate: true,\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  ngOnInit(): void {\n    if (this.props.scrollToTopOnNavigate !== false) {\n      this.routerSubscription = this.router.events\n        .pipe(filter((event) => event instanceof NavigationEnd))\n        .subscribe(() => {\n          this.scrollToTop();\n        });\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.routerSubscription) {\n      this.routerSubscription.unsubscribe();\n    }\n  }\n\n  ionViewWillLeave(): void {\n    // Optional: can add logic on view leave\n  }\n\n  ionViewWillEnter(): void {\n    this.scrollToTop();\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   * Scrolls the content to the top.\n   */\n  private scrollToTop(): void {\n    const contentId = this.props.contentId || 'page-wrapper';\n    const ionContent = document.querySelector(`#${contentId}`) as any;\n    if (ionContent && ionContent.scrollToTop) {\n      ionContent.scrollToTop(300);\n    } else {\n      window.scrollTo({ top: 0, behavior: 'smooth' });\n    }\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
194
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-wrapper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-wrapper/page-wrapper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, computed, EventEmitter, inject, input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { NavigationEnd, Router, RouterOutlet } from '@angular/router';\nimport { IonContent, ViewWillEnter, ViewWillLeave } from '@ionic/angular/standalone';\nimport { filter, Subscription } from 'rxjs';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { CompanyFooterComponent } from '../../organisms/company-footer/company-footer.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageWrapperMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-wrapper\n *\n * A complete page wrapper template with header, router outlet, footer,\n * and automatic scroll-to-top on navigation.\n *\n * @example\n * <val-page-wrapper\n *   [props]=\"{\n *     homeRoute: '/',\n *     scrollToTopOnNavigate: true,\n *     footer: {\n *       links: { ... },\n *       rights: { companyName: 'Acme' }\n *     }\n *   }\"\n * ></val-page-wrapper>\n *\n * @input props - Page wrapper configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-wrapper',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, CompanyFooterComponent, RouterOutlet, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps()\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [id]=\"contentId()\"\n        class=\"ion-padding\"\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': background()\n        }\"\n      >\n        <main>\n          <router-outlet></router-outlet>\n        </main>\n        @if (propsFooter()) {\n          <val-company-footer [props]=\"propsFooter()\" />\n        }\n      </ion-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageWrapperComponent implements ViewWillEnter, ViewWillLeave, OnInit, OnDestroy {\n  private theme = inject(ThemeService);\n  private nav = inject(NavigationService);\n  private router = inject(Router);\n  private routerSubscription?: Subscription;\n\n  /**\n   * Page wrapper configuration.\n   * Signal-based input for full reactivity with computed().\n   */\n  readonly props = input<PageWrapperMetadata>({\n    scrollToTopOnNavigate: true,\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  // Computed helpers for template bindings\n  headerProps = computed(() => this.props()?.header || this.defaultHeader);\n  contentId = computed(() => this.props()?.contentId || 'page-wrapper');\n  propsFooter = computed(() => this.props()?.footer);\n  background = computed(() => {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n    const bg = this.props()?.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n    return resolveColor(bg);\n  });\n\n  ngOnInit(): void {\n    if (this.props()?.scrollToTopOnNavigate !== false) {\n      this.routerSubscription = this.router.events\n        .pipe(filter((event) => event instanceof NavigationEnd))\n        .subscribe(() => {\n          this.scrollToTop();\n        });\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.routerSubscription) {\n      this.routerSubscription.unsubscribe();\n    }\n  }\n\n  ionViewWillLeave(): void {\n    // Optional: can add logic on view leave\n  }\n\n  ionViewWillEnter(): void {\n    this.scrollToTop();\n  }\n\n  /**\n   * Scrolls the content to the top.\n   */\n  private scrollToTop(): void {\n    const id = this.props()?.contentId || 'page-wrapper';\n    const ionContent = document.querySelector(`#${id}`) as any;\n    if (ionContent && ionContent.scrollToTop) {\n      ionContent.scrollToTop(300);\n    } else {\n      window.scrollTo({ top: 0, behavior: 'smooth' });\n    }\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    const homeRoute = this.props()?.homeRoute;\n    if (token === 'header-logo' && homeRoute) {\n      this.nav.navigateByUrl(homeRoute);\n    }\n  }\n}\n"]}
@@ -20742,22 +20742,28 @@ class CompanyFooterComponent {
20742
20742
  constructor() {
20743
20743
  /**
20744
20744
  * Footer configuration.
20745
+ * Signal-based input for full reactivity with computed().
20745
20746
  */
20746
- this.props = {
20747
+ this.props = input({
20747
20748
  links: {
20748
20749
  leftLinks: { title: '', size: 'medium', links: [] },
20749
20750
  rightLinks: { title: '', size: 'medium', links: [] },
20750
20751
  },
20751
- };
20752
+ });
20753
+ // Computed helpers for template bindings
20754
+ this.propsWithPadding = computed(() => this.props()?.withPadding);
20755
+ this.propsWithBackground = computed(() => this.props()?.withBackground);
20756
+ this.propsLinks = computed(() => this.props()?.links);
20757
+ this.propsRights = computed(() => this.props()?.rights || {});
20752
20758
  }
20753
20759
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CompanyFooterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
20754
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CompanyFooterComponent, isStandalone: true, selector: "val-company-footer", inputs: { props: "props" }, ngImport: i0, template: `
20755
- <footer [class.withPadding]="props.withPadding" [class.withBackground]="props.withBackground">
20760
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: CompanyFooterComponent, isStandalone: true, selector: "val-company-footer", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: `
20761
+ <footer [class.withPadding]="propsWithPadding()" [class.withBackground]="propsWithBackground()">
20756
20762
  <ion-grid>
20757
- <val-footer-links [props]="props.links" />
20763
+ <val-footer-links [props]="propsLinks()" />
20758
20764
  <ion-row>
20759
20765
  <ion-col size="12">
20760
- <val-rights-footer [props]="props.rights || {}" />
20766
+ <val-rights-footer [props]="propsRights()" />
20761
20767
  </ion-col>
20762
20768
  </ion-row>
20763
20769
  </ion-grid>
@@ -20767,20 +20773,18 @@ class CompanyFooterComponent {
20767
20773
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CompanyFooterComponent, decorators: [{
20768
20774
  type: Component,
20769
20775
  args: [{ selector: 'val-company-footer', standalone: true, imports: [CommonModule, IonGrid, IonRow, IonCol, RightsFooterComponent, FooterLinksComponent], template: `
20770
- <footer [class.withPadding]="props.withPadding" [class.withBackground]="props.withBackground">
20776
+ <footer [class.withPadding]="propsWithPadding()" [class.withBackground]="propsWithBackground()">
20771
20777
  <ion-grid>
20772
- <val-footer-links [props]="props.links" />
20778
+ <val-footer-links [props]="propsLinks()" />
20773
20779
  <ion-row>
20774
20780
  <ion-col size="12">
20775
- <val-rights-footer [props]="props.rights || {}" />
20781
+ <val-rights-footer [props]="propsRights()" />
20776
20782
  </ion-col>
20777
20783
  </ion-row>
20778
20784
  </ion-grid>
20779
20785
  </footer>
20780
20786
  `, styles: [".withPadding{padding:16px}.withBackground{background-color:var(--ion-background-color)}\n"] }]
20781
- }], propDecorators: { props: [{
20782
- type: Input
20783
- }] } });
20787
+ }] });
20784
20788
 
20785
20789
  /**
20786
20790
  * val-menu
@@ -22466,10 +22470,11 @@ class PageWrapperComponent {
22466
22470
  this.router = inject(Router);
22467
22471
  /**
22468
22472
  * Page wrapper configuration.
22473
+ * Signal-based input for full reactivity with computed().
22469
22474
  */
22470
- this.props = {
22475
+ this.props = input({
22471
22476
  scrollToTopOnNavigate: true,
22472
- };
22477
+ });
22473
22478
  /**
22474
22479
  * Emits when a header action is clicked.
22475
22480
  */
@@ -22507,15 +22512,23 @@ class PageWrapperComponent {
22507
22512
  ],
22508
22513
  },
22509
22514
  };
22510
- }
22511
- /**
22512
- * Gets header props, using cached default if not provided.
22513
- */
22514
- get headerProps() {
22515
- return this.props.header || this.defaultHeader;
22515
+ // Computed helpers for template bindings
22516
+ this.headerProps = computed(() => this.props()?.header || this.defaultHeader);
22517
+ this.contentId = computed(() => this.props()?.contentId || 'page-wrapper');
22518
+ this.propsFooter = computed(() => this.props()?.footer);
22519
+ this.background = computed(() => {
22520
+ if (this.theme.IsDark) {
22521
+ return 'var(--ion-background-color)';
22522
+ }
22523
+ const bg = this.props()?.background;
22524
+ if (!bg) {
22525
+ return 'var(--ion-background-color)';
22526
+ }
22527
+ return resolveColor(bg);
22528
+ });
22516
22529
  }
22517
22530
  ngOnInit() {
22518
- if (this.props.scrollToTopOnNavigate !== false) {
22531
+ if (this.props()?.scrollToTopOnNavigate !== false) {
22519
22532
  this.routerSubscription = this.router.events
22520
22533
  .pipe(filter((event) => event instanceof NavigationEnd))
22521
22534
  .subscribe(() => {
@@ -22534,25 +22547,12 @@ class PageWrapperComponent {
22534
22547
  ionViewWillEnter() {
22535
22548
  this.scrollToTop();
22536
22549
  }
22537
- /**
22538
- * Gets the background color based on theme.
22539
- */
22540
- getBackground() {
22541
- if (this.theme.IsDark) {
22542
- return 'var(--ion-background-color)';
22543
- }
22544
- const bg = this.props.background;
22545
- if (!bg) {
22546
- return 'var(--ion-background-color)';
22547
- }
22548
- return resolveColor(bg);
22549
- }
22550
22550
  /**
22551
22551
  * Scrolls the content to the top.
22552
22552
  */
22553
22553
  scrollToTop() {
22554
- const contentId = this.props.contentId || 'page-wrapper';
22555
- const ionContent = document.querySelector(`#${contentId}`);
22554
+ const id = this.props()?.contentId || 'page-wrapper';
22555
+ const ionContent = document.querySelector(`#${id}`);
22556
22556
  if (ionContent && ionContent.scrollToTop) {
22557
22557
  ionContent.scrollToTop(300);
22558
22558
  }
@@ -22565,30 +22565,31 @@ class PageWrapperComponent {
22565
22565
  */
22566
22566
  onHeaderClickHandler(token) {
22567
22567
  this.onHeaderClick.emit(token);
22568
- if (token === 'header-logo' && this.props.homeRoute) {
22569
- this.nav.navigateByUrl(this.props.homeRoute);
22568
+ const homeRoute = this.props()?.homeRoute;
22569
+ if (token === 'header-logo' && homeRoute) {
22570
+ this.nav.navigateByUrl(homeRoute);
22570
22571
  }
22571
22572
  }
22572
22573
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageWrapperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
22573
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageWrapperComponent, isStandalone: true, selector: "val-page-wrapper", inputs: { props: "props" }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
22574
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageWrapperComponent, isStandalone: true, selector: "val-page-wrapper", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
22574
22575
  <div class="ion-page">
22575
22576
  <val-header
22576
- [props]="headerProps"
22577
+ [props]="headerProps()"
22577
22578
  (onClick)="onHeaderClickHandler($event)"
22578
22579
  />
22579
22580
  <ion-content
22580
- [id]="props.contentId || 'page-wrapper'"
22581
+ [id]="contentId()"
22581
22582
  class="ion-padding"
22582
22583
  [fullscreen]="true"
22583
22584
  [ngStyle]="{
22584
- '--background': getBackground()
22585
+ '--background': background()
22585
22586
  }"
22586
22587
  >
22587
22588
  <main>
22588
22589
  <router-outlet></router-outlet>
22589
22590
  </main>
22590
- @if (props.footer) {
22591
- <val-company-footer [props]="props.footer" />
22591
+ @if (propsFooter()) {
22592
+ <val-company-footer [props]="propsFooter()" />
22592
22593
  }
22593
22594
  </ion-content>
22594
22595
  </div>
@@ -22599,29 +22600,27 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
22599
22600
  args: [{ selector: 'val-page-wrapper', standalone: true, imports: [CommonModule, HeaderComponent, CompanyFooterComponent, RouterOutlet, IonContent], template: `
22600
22601
  <div class="ion-page">
22601
22602
  <val-header
22602
- [props]="headerProps"
22603
+ [props]="headerProps()"
22603
22604
  (onClick)="onHeaderClickHandler($event)"
22604
22605
  />
22605
22606
  <ion-content
22606
- [id]="props.contentId || 'page-wrapper'"
22607
+ [id]="contentId()"
22607
22608
  class="ion-padding"
22608
22609
  [fullscreen]="true"
22609
22610
  [ngStyle]="{
22610
- '--background': getBackground()
22611
+ '--background': background()
22611
22612
  }"
22612
22613
  >
22613
22614
  <main>
22614
22615
  <router-outlet></router-outlet>
22615
22616
  </main>
22616
- @if (props.footer) {
22617
- <val-company-footer [props]="props.footer" />
22617
+ @if (propsFooter()) {
22618
+ <val-company-footer [props]="propsFooter()" />
22618
22619
  }
22619
22620
  </ion-content>
22620
22621
  </div>
22621
22622
  `, styles: ["main{min-height:60vh}\n"] }]
22622
- }], propDecorators: { props: [{
22623
- type: Input
22624
- }], onHeaderClick: [{
22623
+ }], propDecorators: { onHeaderClick: [{
22625
22624
  type: Output
22626
22625
  }] } });
22627
22626