valtech-components 2.0.586 → 2.0.588

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,6 +1,7 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
2
+ import { Component, EventEmitter, inject, Input, Output, ViewChild } from '@angular/core';
3
3
  import { IonAccordion, IonAccordionGroup, IonItem, IonLabel, IonList } from '@ionic/angular/standalone';
4
+ import { I18nService } from '../../../services/i18n';
4
5
  import * as i0 from "@angular/core";
5
6
  import * as i1 from "@angular/router";
6
7
  import * as i2 from "@ionic/angular/standalone";
@@ -32,6 +33,7 @@ export class LinksAccordionComponent {
32
33
  constructor(router, menuCtrl) {
33
34
  this.router = router;
34
35
  this.menuCtrl = menuCtrl;
36
+ this.i18n = inject(I18nService);
35
37
  /**
36
38
  * Accordion configuration.
37
39
  */
@@ -46,6 +48,24 @@ export class LinksAccordionComponent {
46
48
  */
47
49
  this.navigate = new EventEmitter();
48
50
  }
51
+ /**
52
+ * Gets the section title, supporting i18n via titleKey.
53
+ */
54
+ getSectionTitle(section) {
55
+ if (section.titleKey && this.props.i18nNamespace) {
56
+ return this.i18n.t(section.titleKey, this.props.i18nNamespace);
57
+ }
58
+ return section.title || '';
59
+ }
60
+ /**
61
+ * Gets the link text, supporting i18n via textKey.
62
+ */
63
+ getLinkText(link) {
64
+ if (link.textKey && this.props.i18nNamespace) {
65
+ return this.i18n.t(link.textKey, this.props.i18nNamespace);
66
+ }
67
+ return link.text || '';
68
+ }
49
69
  /**
50
70
  * Handles link click - closes accordion/menu and navigates.
51
71
  */
@@ -83,7 +103,7 @@ export class LinksAccordionComponent {
83
103
  [button]="!!section.route"
84
104
  lines="none"
85
105
  >
86
- <ion-label class="accordion-label">{{ section.title }}</ion-label>
106
+ <ion-label class="accordion-label">{{ getSectionTitle(section) }}</ion-label>
87
107
  </ion-item>
88
108
  <div
89
109
  slot="content"
@@ -97,7 +117,7 @@ export class LinksAccordionComponent {
97
117
  (click)="onLinkClick(link.route, $event)"
98
118
  button
99
119
  >
100
- <ion-label class="accordion-item-label">{{ link.text }}</ion-label>
120
+ <ion-label class="accordion-item-label">{{ getLinkText(link) }}</ion-label>
101
121
  </ion-item>
102
122
  </ion-list>
103
123
  </div>
@@ -124,7 +144,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
124
144
  [button]="!!section.route"
125
145
  lines="none"
126
146
  >
127
- <ion-label class="accordion-label">{{ section.title }}</ion-label>
147
+ <ion-label class="accordion-label">{{ getSectionTitle(section) }}</ion-label>
128
148
  </ion-item>
129
149
  <div
130
150
  slot="content"
@@ -138,7 +158,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
138
158
  (click)="onLinkClick(link.route, $event)"
139
159
  button
140
160
  >
141
- <ion-label class="accordion-item-label">{{ link.text }}</ion-label>
161
+ <ion-label class="accordion-item-label">{{ getLinkText(link) }}</ion-label>
142
162
  </ion-item>
143
163
  </ion-list>
144
164
  </div>
@@ -154,4 +174,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
154
174
  type: ViewChild,
155
175
  args: ['accordionGroup']
156
176
  }] } });
157
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlua3MtYWNjb3JkaW9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9tb2xlY3VsZXMvbGlua3MtYWNjb3JkaW9uL2xpbmtzLWFjY29yZGlvbi5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRWxGLE9BQU8sRUFBRSxZQUFZLEVBQUUsaUJBQWlCLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxPQUFPLEVBQWtCLE1BQU0sMkJBQTJCLENBQUM7Ozs7O0FBR3hIOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JHO0FBNkNILE1BQU0sT0FBTyx1QkFBdUI7SUFrQmxDLFlBQ1UsTUFBYyxFQUNkLFFBQXdCO1FBRHhCLFdBQU0sR0FBTixNQUFNLENBQVE7UUFDZCxhQUFRLEdBQVIsUUFBUSxDQUFnQjtRQW5CbEM7O1dBRUc7UUFDTSxVQUFLLEdBQTJCO1lBQ3ZDLFFBQVEsRUFBRSxFQUFFO1lBQ1osZUFBZSxFQUFFLDZCQUE2QjtZQUM5QyxnQkFBZ0IsRUFBRSxJQUFJO1lBQ3RCLHFCQUFxQixFQUFFLElBQUk7U0FDNUIsQ0FBQztRQUVGOztXQUVHO1FBQ08sYUFBUSxHQUFHLElBQUksWUFBWSxFQUFZLENBQUM7SUFPL0MsQ0FBQztJQUVKOztPQUVHO0lBQ0gsS0FBSyxDQUFDLFdBQVcsQ0FBQyxLQUFlLEVBQUUsS0FBYTtRQUM5QyxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsS0FBSyxDQUFDLGVBQWUsRUFBRSxDQUFDO1FBQzFCLENBQUM7UUFFRCxvQ0FBb0M7UUFDcEMsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLHFCQUFxQixLQUFLLEtBQUssSUFBSSxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDdEUsSUFBSSxDQUFDLGNBQWMsQ0FBQyxLQUFLLEdBQUcsU0FBUyxDQUFDO1FBQ3hDLENBQUM7UUFFRCx3QkFBd0I7UUFDeEIsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLGdCQUFnQixLQUFLLEtBQUssRUFBRSxDQUFDO1lBQzFDLE1BQU0sSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUM5QixDQUFDO1FBRUQsYUFBYTtRQUNiLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRTFCLFdBQVc7UUFDWCxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUM5QixDQUFDOytHQTlDVSx1QkFBdUI7bUdBQXZCLHVCQUF1QixnUUF4Q3hCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBcUNULGdUQXRDUyxZQUFZLHFWQUFFLGlCQUFpQixxSkFBRSxZQUFZLDZJQUFFLE9BQU8sME5BQUUsUUFBUSw2RkFBRSxPQUFPOzs0RkF5Q3hFLHVCQUF1QjtrQkE1Q25DLFNBQVM7K0JBQ0UscUJBQXFCLGNBQ25CLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxpQkFBaUIsRUFBRSxZQUFZLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxPQUFPLENBQUMsWUFDMUU7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FxQ1Q7d0dBT1EsS0FBSztzQkFBYixLQUFLO2dCQVVJLFFBQVE7c0JBQWpCLE1BQU07Z0JBRXNCLGNBQWM7c0JBQTFDLFNBQVM7dUJBQUMsZ0JBQWdCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgT3V0cHV0LCBWaWV3Q2hpbGQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJvdXRlciB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBJb25BY2NvcmRpb24sIElvbkFjY29yZGlvbkdyb3VwLCBJb25JdGVtLCBJb25MYWJlbCwgSW9uTGlzdCwgTWVudUNvbnRyb2xsZXIgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IExpbmtzQWNjb3JkaW9uTWV0YWRhdGEsIExpbmtzQWNjb3JkaW9uU2VjdGlvbiB9IGZyb20gJy4vdHlwZXMnO1xuXG4vKipcbiAqIHZhbC1saW5rcy1hY2NvcmRpb25cbiAqXG4gKiBBIGhpZXJhcmNoaWNhbCBuYXZpZ2F0aW9uIGNvbXBvbmVudCB1c2luZyBJb25pYyBhY2NvcmRpb25zLlxuICogU3VwcG9ydHMgc2VjdGlvbnMgd2l0aCBvcHRpb25hbCBzdWItbGlua3MgYW5kIGF1dG9tYXRpYyBtZW51IGNsb3NpbmcuXG4gKlxuICogQGV4YW1wbGVcbiAqIDx2YWwtbGlua3MtYWNjb3JkaW9uXG4gKiAgIFtwcm9wc109XCJ7XG4gKiAgICAgc2VjdGlvbnM6IFtcbiAqICAgICAgIHsgdGl0bGU6ICdIb21lJywgcm91dGU6IFsnLyddIH0sXG4gKiAgICAgICB7IHRpdGxlOiAnUHJvZHVjdHMnLCBsaW5rczogW1xuICogICAgICAgICB7IHRleHQ6ICdDYXRhbG9nJywgcm91dGU6IFsnL3Byb2R1Y3RzJ10gfSxcbiAqICAgICAgICAgeyB0ZXh0OiAnTmV3IEFycml2YWxzJywgcm91dGU6IFsnL3Byb2R1Y3RzJywgJ25ldyddIH1cbiAqICAgICAgIF19XG4gKiAgICAgXVxuICogICB9XCJcbiAqICAgKG5hdmlnYXRlKT1cIm9uTmF2aWdhdGUoJGV2ZW50KVwiXG4gKiA+PC92YWwtbGlua3MtYWNjb3JkaW9uPlxuICpcbiAqIEBpbnB1dCBwcm9wcyAtIEFjY29yZGlvbiBjb25maWd1cmF0aW9uXG4gKiBAb3V0cHV0IG5hdmlnYXRlIC0gRW1pdHMgcm91dGUgd2hlbiBhIGxpbmsgaXMgY2xpY2tlZFxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtbGlua3MtYWNjb3JkaW9uJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgSW9uQWNjb3JkaW9uR3JvdXAsIElvbkFjY29yZGlvbiwgSW9uSXRlbSwgSW9uTGFiZWwsIElvbkxpc3RdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXYgY2xhc3M9XCJsaW5rcy1hY2NvcmRpb25cIj5cbiAgICAgIDxpb24tYWNjb3JkaW9uLWdyb3VwICNhY2NvcmRpb25Hcm91cD5cbiAgICAgICAgPGlvbi1hY2NvcmRpb25cbiAgICAgICAgICAqbmdGb3I9XCJsZXQgc2VjdGlvbiBvZiBwcm9wcy5zZWN0aW9uc1wiXG4gICAgICAgICAgW3ZhbHVlXT1cInNlY3Rpb24udGl0bGVcIlxuICAgICAgICAgIFt0b2dnbGVJY29uXT1cInNlY3Rpb24ubGlua3M/Lmxlbmd0aCA/ICdjaGV2cm9uLWRvd24tb3V0bGluZScgOiAnY2hldnJvbi1mb3J3YXJkLW91dGxpbmUnXCJcbiAgICAgICAgPlxuICAgICAgICAgIDxpb24taXRlbVxuICAgICAgICAgICAgc2xvdD1cImhlYWRlclwiXG4gICAgICAgICAgICBjb2xvcj1cImxpZ2h0XCJcbiAgICAgICAgICAgIFtuZ1N0eWxlXT1cInsgYmFja2dyb3VuZDogcHJvcHMuYmFja2dyb3VuZENvbG9yIHx8ICd2YXIoLS1pb24tYmFja2dyb3VuZC1jb2xvciknIH1cIlxuICAgICAgICAgICAgKGNsaWNrKT1cInNlY3Rpb24ucm91dGUgPyBvbkxpbmtDbGljayhzZWN0aW9uLnJvdXRlLCAkZXZlbnQpIDogbnVsbFwiXG4gICAgICAgICAgICBbYnV0dG9uXT1cIiEhc2VjdGlvbi5yb3V0ZVwiXG4gICAgICAgICAgICBsaW5lcz1cIm5vbmVcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIDxpb24tbGFiZWwgY2xhc3M9XCJhY2NvcmRpb24tbGFiZWxcIj57eyBzZWN0aW9uLnRpdGxlIH19PC9pb24tbGFiZWw+XG4gICAgICAgICAgPC9pb24taXRlbT5cbiAgICAgICAgICA8ZGl2XG4gICAgICAgICAgICBzbG90PVwiY29udGVudFwiXG4gICAgICAgICAgICBbbmdTdHlsZV09XCJ7IGJhY2tncm91bmQ6IHByb3BzLmJhY2tncm91bmRDb2xvciB8fCAndmFyKC0taW9uLWJhY2tncm91bmQtY29sb3IpJyB9XCJcbiAgICAgICAgICAgICpuZ0lmPVwic2VjdGlvbi5saW5rcz8ubGVuZ3RoXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICA8aW9uLWxpc3QgbGluZXM9XCJub25lXCI+XG4gICAgICAgICAgICAgIDxpb24taXRlbVxuICAgICAgICAgICAgICAgICpuZ0Zvcj1cImxldCBsaW5rIG9mIHNlY3Rpb24ubGlua3NcIlxuICAgICAgICAgICAgICAgIFtuZ1N0eWxlXT1cInsgYmFja2dyb3VuZDogcHJvcHMuYmFja2dyb3VuZENvbG9yIHx8ICd2YXIoLS1pb24tYmFja2dyb3VuZC1jb2xvciknIH1cIlxuICAgICAgICAgICAgICAgIChjbGljayk9XCJvbkxpbmtDbGljayhsaW5rLnJvdXRlLCAkZXZlbnQpXCJcbiAgICAgICAgICAgICAgICBidXR0b25cbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIDxpb24tbGFiZWwgY2xhc3M9XCJhY2NvcmRpb24taXRlbS1sYWJlbFwiPnt7IGxpbmsudGV4dCB9fTwvaW9uLWxhYmVsPlxuICAgICAgICAgICAgICA8L2lvbi1pdGVtPlxuICAgICAgICAgICAgPC9pb24tbGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9pb24tYWNjb3JkaW9uPlxuICAgICAgPC9pb24tYWNjb3JkaW9uLWdyb3VwPlxuICAgIDwvZGl2PlxuICBgLFxuICBzdHlsZVVybHM6IFsnLi9saW5rcy1hY2NvcmRpb24uY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgTGlua3NBY2NvcmRpb25Db21wb25lbnQge1xuICAvKipcbiAgICogQWNjb3JkaW9uIGNvbmZpZ3VyYXRpb24uXG4gICAqL1xuICBASW5wdXQoKSBwcm9wczogTGlua3NBY2NvcmRpb25NZXRhZGF0YSA9IHtcbiAgICBzZWN0aW9uczogW10sXG4gICAgYmFja2dyb3VuZENvbG9yOiAndmFyKC0taW9uLWJhY2tncm91bmQtY29sb3IpJyxcbiAgICBjbG9zZU1lbnVPbkNsaWNrOiB0cnVlLFxuICAgIGNsb3NlQWNjb3JkaW9uT25DbGljazogdHJ1ZSxcbiAgfTtcblxuICAvKipcbiAgICogRW1pdHMgdGhlIHJvdXRlIHdoZW4gYSBsaW5rIGlzIGNsaWNrZWQuXG4gICAqL1xuICBAT3V0cHV0KCkgbmF2aWdhdGUgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZ1tdPigpO1xuXG4gIEBWaWV3Q2hpbGQoJ2FjY29yZGlvbkdyb3VwJykgYWNjb3JkaW9uR3JvdXAhOiBJb25BY2NvcmRpb25Hcm91cDtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIHJvdXRlcjogUm91dGVyLFxuICAgIHByaXZhdGUgbWVudUN0cmw6IE1lbnVDb250cm9sbGVyXG4gICkge31cblxuICAvKipcbiAgICogSGFuZGxlcyBsaW5rIGNsaWNrIC0gY2xvc2VzIGFjY29yZGlvbi9tZW51IGFuZCBuYXZpZ2F0ZXMuXG4gICAqL1xuICBhc3luYyBvbkxpbmtDbGljayhyb3V0ZTogc3RyaW5nW10sIGV2ZW50PzogRXZlbnQpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBpZiAoZXZlbnQpIHtcbiAgICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xuICAgIH1cblxuICAgIC8vIENsb3NlIGFjY29yZGlvbiBiZWZvcmUgbmF2aWdhdGluZ1xuICAgIGlmICh0aGlzLnByb3BzLmNsb3NlQWNjb3JkaW9uT25DbGljayAhPT0gZmFsc2UgJiYgdGhpcy5hY2NvcmRpb25Hcm91cCkge1xuICAgICAgdGhpcy5hY2NvcmRpb25Hcm91cC52YWx1ZSA9IHVuZGVmaW5lZDtcbiAgICB9XG5cbiAgICAvLyBDbG9zZSBtZW51IGlmIGVuYWJsZWRcbiAgICBpZiAodGhpcy5wcm9wcy5jbG9zZU1lbnVPbkNsaWNrICE9PSBmYWxzZSkge1xuICAgICAgYXdhaXQgdGhpcy5tZW51Q3RybC5jbG9zZSgpO1xuICAgIH1cblxuICAgIC8vIEVtaXQgZXZlbnRcbiAgICB0aGlzLm5hdmlnYXRlLmVtaXQocm91dGUpO1xuXG4gICAgLy8gTmF2aWdhdGVcbiAgICB0aGlzLnJvdXRlci5uYXZpZ2F0ZShyb3V0ZSk7XG4gIH1cbn1cbiJdfQ==
177
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"links-accordion.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/links-accordion/links-accordion.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output, ViewChild } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { IonAccordion, IonAccordionGroup, IonItem, IonLabel, IonList, MenuController } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { LinksAccordionLink, LinksAccordionMetadata, LinksAccordionSection } from './types';\n\n/**\n * val-links-accordion\n *\n * A hierarchical navigation component using Ionic accordions.\n * Supports sections with optional sub-links and automatic menu closing.\n *\n * @example\n * <val-links-accordion\n *   [props]=\"{\n *     sections: [\n *       { title: 'Home', route: ['/'] },\n *       { title: 'Products', links: [\n *         { text: 'Catalog', route: ['/products'] },\n *         { text: 'New Arrivals', route: ['/products', 'new'] }\n *       ]}\n *     ]\n *   }\"\n *   (navigate)=\"onNavigate($event)\"\n * ></val-links-accordion>\n *\n * @input props - Accordion configuration\n * @output navigate - Emits route when a link is clicked\n */\n@Component({\n  selector: 'val-links-accordion',\n  standalone: true,\n  imports: [CommonModule, IonAccordionGroup, IonAccordion, IonItem, IonLabel, IonList],\n  template: `\n    <div class=\"links-accordion\">\n      <ion-accordion-group #accordionGroup>\n        <ion-accordion\n          *ngFor=\"let section of props.sections\"\n          [value]=\"section.title\"\n          [toggleIcon]=\"section.links?.length ? 'chevron-down-outline' : 'chevron-forward-outline'\"\n        >\n          <ion-item\n            slot=\"header\"\n            color=\"light\"\n            [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n            (click)=\"section.route ? onLinkClick(section.route, $event) : null\"\n            [button]=\"!!section.route\"\n            lines=\"none\"\n          >\n            <ion-label class=\"accordion-label\">{{ getSectionTitle(section) }}</ion-label>\n          </ion-item>\n          <div\n            slot=\"content\"\n            [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n            *ngIf=\"section.links?.length\"\n          >\n            <ion-list lines=\"none\">\n              <ion-item\n                *ngFor=\"let link of section.links\"\n                [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n                (click)=\"onLinkClick(link.route, $event)\"\n                button\n              >\n                <ion-label class=\"accordion-item-label\">{{ getLinkText(link) }}</ion-label>\n              </ion-item>\n            </ion-list>\n          </div>\n        </ion-accordion>\n      </ion-accordion-group>\n    </div>\n  `,\n  styleUrls: ['./links-accordion.component.scss'],\n})\nexport class LinksAccordionComponent {\n  private i18n = inject(I18nService);\n\n  /**\n   * Accordion configuration.\n   */\n  @Input() props: LinksAccordionMetadata = {\n    sections: [],\n    backgroundColor: 'var(--ion-background-color)',\n    closeMenuOnClick: true,\n    closeAccordionOnClick: true,\n  };\n\n  /**\n   * Emits the route when a link is clicked.\n   */\n  @Output() navigate = new EventEmitter<string[]>();\n\n  @ViewChild('accordionGroup') accordionGroup!: IonAccordionGroup;\n\n  constructor(\n    private router: Router,\n    private menuCtrl: MenuController\n  ) {}\n\n  /**\n   * Gets the section title, supporting i18n via titleKey.\n   */\n  getSectionTitle(section: LinksAccordionSection): string {\n    if (section.titleKey && this.props.i18nNamespace) {\n      return this.i18n.t(section.titleKey, this.props.i18nNamespace);\n    }\n    return section.title || '';\n  }\n\n  /**\n   * Gets the link text, supporting i18n via textKey.\n   */\n  getLinkText(link: LinksAccordionLink): string {\n    if (link.textKey && this.props.i18nNamespace) {\n      return this.i18n.t(link.textKey, this.props.i18nNamespace);\n    }\n    return link.text || '';\n  }\n\n  /**\n   * Handles link click - closes accordion/menu and navigates.\n   */\n  async onLinkClick(route: string[], event?: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    // Close accordion before navigating\n    if (this.props.closeAccordionOnClick !== false && this.accordionGroup) {\n      this.accordionGroup.value = undefined;\n    }\n\n    // Close menu if enabled\n    if (this.props.closeMenuOnClick !== false) {\n      await this.menuCtrl.close();\n    }\n\n    // Emit event\n    this.navigate.emit(route);\n\n    // Navigate\n    this.router.navigate(route);\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2xpbmtzLWFjY29yZGlvbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBBIGxpbmsgaXRlbSB3aXRoaW4gYW4gYWNjb3JkaW9uIHNlY3Rpb24uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGlua3NBY2NvcmRpb25MaW5rIHtcbiAgLyoqIERpc3BsYXkgdGV4dCBmb3IgdGhlIGxpbmsgKi9cbiAgdGV4dDogc3RyaW5nO1xuICAvKiogUm91dGUgc2VnbWVudHMgZm9yIG5hdmlnYXRpb24gKi9cbiAgcm91dGU6IHN0cmluZ1tdO1xufVxuXG4vKipcbiAqIEFuIGFjY29yZGlvbiBzZWN0aW9uIHdpdGggb3B0aW9uYWwgc3ViLWxpbmtzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIExpbmtzQWNjb3JkaW9uU2VjdGlvbiB7XG4gIC8qKiBTZWN0aW9uIHRpdGxlICovXG4gIHRpdGxlOiBzdHJpbmc7XG4gIC8qKiBPcHRpb25hbCByb3V0ZSBpZiB0aGUgc2VjdGlvbiBpdHNlbGYgaXMgY2xpY2thYmxlICovXG4gIHJvdXRlPzogc3RyaW5nW107XG4gIC8qKiBPcHRpb25hbCBzdWItbGlua3Mgd2l0aGluIHRoaXMgc2VjdGlvbiAqL1xuICBsaW5rcz86IExpbmtzQWNjb3JkaW9uTGlua1tdO1xufVxuXG4vKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBsaW5rcyBhY2NvcmRpb24gY29tcG9uZW50LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIExpbmtzQWNjb3JkaW9uTWV0YWRhdGEge1xuICAvKiogQXJyYXkgb2YgYWNjb3JkaW9uIHNlY3Rpb25zICovXG4gIHNlY3Rpb25zOiBMaW5rc0FjY29yZGlvblNlY3Rpb25bXTtcbiAgLyoqIEJhY2tncm91bmQgY29sb3IgZm9yIHRoZSBhY2NvcmRpb24uIERlZmF1bHQ6ICd2YXIoLS1pb24tYmFja2dyb3VuZC1jb2xvciknICovXG4gIGJhY2tncm91bmRDb2xvcj86IHN0cmluZztcbiAgLyoqIFdoZXRoZXIgdG8gY2xvc2UgdGhlIG1lbnUgd2hlbiBhIGxpbmsgaXMgY2xpY2tlZC4gRGVmYXVsdDogdHJ1ZSAqL1xuICBjbG9zZU1lbnVPbkNsaWNrPzogYm9vbGVhbjtcbiAgLyoqIFdoZXRoZXIgdG8gY2xvc2UgdGhlIGFjY29yZGlvbiB3aGVuIG5hdmlnYXRpbmcuIERlZmF1bHQ6IHRydWUgKi9cbiAgY2xvc2VBY2NvcmRpb25PbkNsaWNrPzogYm9vbGVhbjtcbn1cbiJdfQ==
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2xpbmtzLWFjY29yZGlvbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBBIGxpbmsgaXRlbSB3aXRoaW4gYW4gYWNjb3JkaW9uIHNlY3Rpb24uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGlua3NBY2NvcmRpb25MaW5rIHtcbiAgLyoqIERpc3BsYXkgdGV4dCBmb3IgdGhlIGxpbmsgKHN0YXRpYykgKi9cbiAgdGV4dD86IHN0cmluZztcbiAgLyoqIGkxOG4ga2V5IGZvciB0aGUgbGluayB0ZXh0ICh1c2Ugd2l0aCBuYW1lc3BhY2UgaW4gTGlua3NBY2NvcmRpb25NZXRhZGF0YSkgKi9cbiAgdGV4dEtleT86IHN0cmluZztcbiAgLyoqIFJvdXRlIHNlZ21lbnRzIGZvciBuYXZpZ2F0aW9uICovXG4gIHJvdXRlOiBzdHJpbmdbXTtcbn1cblxuLyoqXG4gKiBBbiBhY2NvcmRpb24gc2VjdGlvbiB3aXRoIG9wdGlvbmFsIHN1Yi1saW5rcy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBMaW5rc0FjY29yZGlvblNlY3Rpb24ge1xuICAvKiogU2VjdGlvbiB0aXRsZSAoc3RhdGljKSAqL1xuICB0aXRsZT86IHN0cmluZztcbiAgLyoqIGkxOG4ga2V5IGZvciB0aGUgc2VjdGlvbiB0aXRsZSAodXNlIHdpdGggbmFtZXNwYWNlIGluIExpbmtzQWNjb3JkaW9uTWV0YWRhdGEpICovXG4gIHRpdGxlS2V5Pzogc3RyaW5nO1xuICAvKiogT3B0aW9uYWwgcm91dGUgaWYgdGhlIHNlY3Rpb24gaXRzZWxmIGlzIGNsaWNrYWJsZSAqL1xuICByb3V0ZT86IHN0cmluZ1tdO1xuICAvKiogT3B0aW9uYWwgc3ViLWxpbmtzIHdpdGhpbiB0aGlzIHNlY3Rpb24gKi9cbiAgbGlua3M/OiBMaW5rc0FjY29yZGlvbkxpbmtbXTtcbn1cblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgbGlua3MgYWNjb3JkaW9uIGNvbXBvbmVudC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBMaW5rc0FjY29yZGlvbk1ldGFkYXRhIHtcbiAgLyoqIEFycmF5IG9mIGFjY29yZGlvbiBzZWN0aW9ucyAqL1xuICBzZWN0aW9uczogTGlua3NBY2NvcmRpb25TZWN0aW9uW107XG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIGZvciB0aGUgYWNjb3JkaW9uLiBEZWZhdWx0OiAndmFyKC0taW9uLWJhY2tncm91bmQtY29sb3IpJyAqL1xuICBiYWNrZ3JvdW5kQ29sb3I/OiBzdHJpbmc7XG4gIC8qKiBXaGV0aGVyIHRvIGNsb3NlIHRoZSBtZW51IHdoZW4gYSBsaW5rIGlzIGNsaWNrZWQuIERlZmF1bHQ6IHRydWUgKi9cbiAgY2xvc2VNZW51T25DbGljaz86IGJvb2xlYW47XG4gIC8qKiBXaGV0aGVyIHRvIGNsb3NlIHRoZSBhY2NvcmRpb24gd2hlbiBuYXZpZ2F0aW5nLiBEZWZhdWx0OiB0cnVlICovXG4gIGNsb3NlQWNjb3JkaW9uT25DbGljaz86IGJvb2xlYW47XG4gIC8qKiBpMThuIG5hbWVzcGFjZSBmb3IgdHJhbnNsYXRpbmcgdGl0bGVLZXkgYW5kIHRleHRLZXkgdmFsdWVzICovXG4gIGkxOG5OYW1lc3BhY2U/OiBzdHJpbmc7XG59XG4iXX0=
@@ -5,6 +5,7 @@ import { addIcons } from 'ionicons';
5
5
  import { chevronBackOutline } from 'ionicons/icons';
6
6
  import { AvatarComponent } from '../../atoms/avatar/avatar.component';
7
7
  import { ImageComponent } from '../../atoms/image/image.component';
8
+ import { LanguageSelectorComponent } from '../../molecules/language-selector/language-selector.component';
8
9
  import { ToolbarActionType } from '../../types';
9
10
  import * as i0 from "@angular/core";
10
11
  import * as i1 from "@ionic/angular";
@@ -70,6 +71,16 @@ export class ToolbarComponent {
70
71
  leftActions() {
71
72
  return this.props.actions.filter(x => x.position === 'left');
72
73
  }
74
+ /**
75
+ * Language selector props with 'icon' mode as default for toolbar display.
76
+ */
77
+ get languageSelectorProps() {
78
+ return {
79
+ mode: 'icon',
80
+ color: this.props.textColor || 'dark',
81
+ ...this.props.languageSelector,
82
+ };
83
+ }
73
84
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ToolbarComponent, deps: [{ token: i1.NavController }, { token: i2.IconService }], target: i0.ɵɵFactoryTarget.Component }); }
74
85
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ToolbarComponent, isStandalone: true, selector: "val-toolbar", inputs: { props: "props" }, outputs: { onClick: "onClick" }, ngImport: i0, template: `
75
86
  <ion-toolbar [color]="props.color" [class.background]="props.color === 'background'">
@@ -104,6 +115,9 @@ export class ToolbarComponent {
104
115
  >{{ action.description }}</ion-button
105
116
  >
106
117
  </ng-container>
118
+ @if (props.languageSelector) {
119
+ <val-language-selector [props]="languageSelectorProps"></val-language-selector>
120
+ }
107
121
  @if (props.withMenu) {
108
122
  <ion-menu-button color="dark"></ion-menu-button>
109
123
  }
@@ -133,7 +147,7 @@ export class ToolbarComponent {
133
147
  <!-- experimental -->
134
148
  <ng-content select="[toolbar-bottom]"></ng-content>
135
149
  </ion-toolbar>
136
- `, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.left-buttons{margin-left:-1rem}.background{background:var(--ion-background-color)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: AvatarComponent, selector: "val-avatar", inputs: ["preset", "props"], outputs: ["onClick"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }, { kind: "component", type: IonMenuButton, selector: "ion-menu-button", inputs: ["autoHide", "color", "disabled", "menu", "mode", "type"] }] }); }
150
+ `, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.left-buttons{margin-left:-1rem}.background{background:var(--ion-background-color)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: AvatarComponent, selector: "val-avatar", inputs: ["preset", "props"], outputs: ["onClick"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }, { kind: "component", type: IonMenuButton, selector: "ion-menu-button", inputs: ["autoHide", "color", "disabled", "menu", "mode", "type"] }, { kind: "component", type: LanguageSelectorComponent, selector: "val-language-selector", inputs: ["props"], outputs: ["languageChange"] }] }); }
137
151
  }
138
152
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ToolbarComponent, decorators: [{
139
153
  type: Component,
@@ -148,6 +162,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
148
162
  AvatarComponent,
149
163
  ImageComponent,
150
164
  IonMenuButton,
165
+ LanguageSelectorComponent,
151
166
  ], template: `
152
167
  <ion-toolbar [color]="props.color" [class.background]="props.color === 'background'">
153
168
  <ng-container *ngIf="props.withBack">
@@ -181,6 +196,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
181
196
  >{{ action.description }}</ion-button
182
197
  >
183
198
  </ng-container>
199
+ @if (props.languageSelector) {
200
+ <val-language-selector [props]="languageSelectorProps"></val-language-selector>
201
+ }
184
202
  @if (props.withMenu) {
185
203
  <ion-menu-button color="dark"></ion-menu-button>
186
204
  }
@@ -216,4 +234,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
216
234
  }], onClick: [{
217
235
  type: Output
218
236
  }] } });
219
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"toolbar.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/toolbar/toolbar.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAE/E,OAAO,EACL,SAAS,EACT,UAAU,EACV,OAAO,EACP,aAAa,EACb,OAAO,EACP,QAAQ,EACR,UAAU,GACX,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AAEpD,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAC;AACtE,OAAO,EAAE,cAAc,EAAE,MAAM,mCAAmC,CAAC;AACnE,OAAO,EAAiB,iBAAiB,EAAE,MAAM,aAAa,CAAC;;;;;AAG/D;;;;;;;;;;;GAWG;AAiFH,MAAM,OAAO,gBAAgB;IAe3B,YACU,OAAsB,EAC9B,IAAiB;QADT,YAAO,GAAP,OAAO,CAAe;QAThC;;WAEG;QAEH,YAAO,GAAG,IAAI,YAAY,EAAU,CAAC;QAErC,gBAAW,GAAG,iBAAiB,CAAC;QAM9B,QAAQ,CAAC,EAAE,kBAAkB,EAAE,CAAC,CAAC;IACnC,CAAC;IAED,QAAQ,KAAI,CAAC;IAEb;;;OAGG;IACH,YAAY,CAAC,KAAc;QACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED;;OAEG;IACH,MAAM;QACJ,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;IACtB,CAAC;IAED;;OAEG;IACH,WAAW;QACT,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC;IAChE,CAAC;IAED;;OAEG;IACH,UAAU;QACR,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,MAAM,CAAC,CAAC;IAC/D,CAAC;IAED;;OAEG;IACH,YAAY;QACV,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC;IAChE,CAAC;IAED;;OAEG;IACH,WAAW;QACT,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,MAAM,CAAC,CAAC;IAC/D,CAAC;+GAjEU,gBAAgB;mGAAhB,gBAAgB,oIAjEjB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8DT,kuFAzEC,YAAY,gQACZ,UAAU,mFACV,UAAU,8EACV,SAAS,oPACT,OAAO,2JACP,OAAO,gFACP,QAAQ,iFACR,eAAe,0GACf,cAAc,yEACd,aAAa;;4FAmEJ,gBAAgB;kBAhF5B,SAAS;+BACE,aAAa,cACX,IAAI,WACP;wBACP,YAAY;wBACZ,UAAU;wBACV,UAAU;wBACV,SAAS;wBACT,OAAO;wBACP,OAAO;wBACP,QAAQ;wBACR,eAAe;wBACf,cAAc;wBACd,aAAa;qBACd,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8DT;4GAQD,KAAK;sBADJ,KAAK;gBAON,OAAO;sBADN,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { NavController } from '@ionic/angular';\nimport {\n  IonButton,\n  IonButtons,\n  IonIcon,\n  IonMenuButton,\n  IonText,\n  IonTitle,\n  IonToolbar,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { chevronBackOutline } from 'ionicons/icons';\nimport { IconService } from '../../../services/icons.service';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport { ToolbarAction, ToolbarActionType } from '../../types';\nimport { ToolbarMetadata } from './types';\n\n/**\n * ToolbarComponent\n *\n * Componente de barra de herramientas reutilizable para encabezados y pies de página.\n * Permite acciones, botón de retroceso, menú y personalización de colores y título.\n *\n * @example\n * <val-toolbar [props]=\"{ title: 'Mi App', withBack: true, actions: [...] }\" (onClick)=\"handleToolbarAction($event)\"></val-toolbar>\n *\n * @input props {ToolbarMetadata} - Metadatos de la barra de herramientas.\n * @output onClick - Emite el token de la acción clicada.\n */\n@Component({\n  selector: 'val-toolbar',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonToolbar,\n    IonButtons,\n    IonButton,\n    IonIcon,\n    IonText,\n    IonTitle,\n    AvatarComponent,\n    ImageComponent,\n    IonMenuButton,\n  ],\n  template: `\n    <ion-toolbar [color]=\"props.color\" [class.background]=\"props.color === 'background'\">\n      <ng-container *ngIf=\"props.withBack\">\n        <ion-buttons class=\"left-buttons\" slot=\"start\" *ngIf=\"props.withBack\">\n          <ion-button fill=\"clear\" (click)=\"goBack()\" [color]=\"props.textColor\" style=\"margin-left: 8px;\">\n            <ion-icon name=\"chevron-back-outline\" [slot]=\"props.backText ? 'start' : 'icon-only'\"></ion-icon>\n            <ion-text *ngIf=\"props.backText\">{{ props.backText }}</ion-text>\n          </ion-button>\n        </ion-buttons>\n      </ng-container>\n      <ng-container *ngIf=\"props.withActions\">\n        <ion-buttons slot=\"end\" *ngIf=\"someInRight() || props.withMenu\">\n          <ng-container *ngFor=\"let action of rightActions()\">\n            <ion-button *ngIf=\"action.type === actionTypes.ICON\" (click)=\"clickHandler(action.token)\">\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" [color]=\"props.textColor\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(action.token)\"\n            ></val-image>\n            <ion-button\n              [color]=\"props.textColor\"\n              *ngIf=\"action.type === actionTypes.BUTTON\"\n              (click)=\"clickHandler(action.token)\"\n              >{{ action.description }}</ion-button\n            >\n          </ng-container>\n          @if (props.withMenu) {\n            <ion-menu-button color=\"dark\"></ion-menu-button>\n          }\n        </ion-buttons>\n        <ion-buttons slot=\"start\" *ngIf=\"someInLeft()\" style=\"padding-left: 4px;\">\n          <ng-container *ngFor=\"let action of leftActions()\">\n            <ion-button *ngIf=\"action.type === actionTypes.ICON\" (click)=\"clickHandler(action.token)\">\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" [color]=\"props.textColor\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(action.token)\"\n            ></val-image>\n            <ion-button *ngIf=\"action.type === actionTypes.BUTTON\" (click)=\"clickHandler(action.token)\">{{\n              action.description\n            }}</ion-button>\n          </ng-container>\n        </ion-buttons>\n      </ng-container>\n      <ion-title *ngIf=\"props.title\" [color]=\"props.textColor\">{{ props.title }}</ion-title>\n      <!-- experimental -->\n      <ng-content select=\"[toolbar-bottom]\"></ng-content>\n    </ion-toolbar>\n  `,\n  styleUrls: ['./toolbar.component.scss'],\n})\nexport class ToolbarComponent implements OnInit {\n  /**\n   * Metadatos de la barra de herramientas.\n   */\n  @Input()\n  props: ToolbarMetadata;\n\n  /**\n   * Evento emitido al hacer click en una acción o botón de la barra.\n   */\n  @Output()\n  onClick = new EventEmitter<string>();\n\n  actionTypes = ToolbarActionType;\n\n  constructor(\n    private navCtrl: NavController,\n    icon: IconService\n  ) {\n    addIcons({ chevronBackOutline });\n  }\n\n  ngOnInit() {}\n\n  /**\n   * Emite el token de la acción clicada.\n   * @param token Token de la acción\n   */\n  clickHandler(token?: string) {\n    this.onClick.emit(token);\n  }\n\n  /**\n   * Navega hacia atrás usando NavController.\n   */\n  goBack() {\n    this.navCtrl.back();\n  }\n\n  /**\n   * Determina si hay acciones a la derecha.\n   */\n  someInRight(): boolean {\n    return !!this.props.actions.find(x => x.position === 'right');\n  }\n\n  /**\n   * Determina si hay acciones a la izquierda.\n   */\n  someInLeft(): boolean {\n    return !!this.props.actions.find(x => x.position === 'left');\n  }\n\n  /**\n   * Devuelve las acciones de la derecha.\n   */\n  rightActions(): ToolbarAction[] {\n    return this.props.actions.filter(x => x.position === 'right');\n  }\n\n  /**\n   * Devuelve las acciones de la izquierda.\n   */\n  leftActions(): ToolbarAction[] {\n    return this.props.actions.filter(x => x.position === 'left');\n  }\n}\n"]}
237
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"toolbar.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/toolbar/toolbar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { NavController } from '@ionic/angular';\nimport {\n  IonButton,\n  IonButtons,\n  IonIcon,\n  IonMenuButton,\n  IonText,\n  IonTitle,\n  IonToolbar,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { chevronBackOutline } from 'ionicons/icons';\nimport { IconService } from '../../../services/icons.service';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport { LanguageSelectorComponent } from '../../molecules/language-selector/language-selector.component';\nimport { ToolbarAction, ToolbarActionType } from '../../types';\nimport { ToolbarMetadata } from './types';\n\n/**\n * ToolbarComponent\n *\n * Componente de barra de herramientas reutilizable para encabezados y pies de página.\n * Permite acciones, botón de retroceso, menú y personalización de colores y título.\n *\n * @example\n * <val-toolbar [props]=\"{ title: 'Mi App', withBack: true, actions: [...] }\" (onClick)=\"handleToolbarAction($event)\"></val-toolbar>\n *\n * @input props {ToolbarMetadata} - Metadatos de la barra de herramientas.\n * @output onClick - Emite el token de la acción clicada.\n */\n@Component({\n  selector: 'val-toolbar',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonToolbar,\n    IonButtons,\n    IonButton,\n    IonIcon,\n    IonText,\n    IonTitle,\n    AvatarComponent,\n    ImageComponent,\n    IonMenuButton,\n    LanguageSelectorComponent,\n  ],\n  template: `\n    <ion-toolbar [color]=\"props.color\" [class.background]=\"props.color === 'background'\">\n      <ng-container *ngIf=\"props.withBack\">\n        <ion-buttons class=\"left-buttons\" slot=\"start\" *ngIf=\"props.withBack\">\n          <ion-button fill=\"clear\" (click)=\"goBack()\" [color]=\"props.textColor\" style=\"margin-left: 8px;\">\n            <ion-icon name=\"chevron-back-outline\" [slot]=\"props.backText ? 'start' : 'icon-only'\"></ion-icon>\n            <ion-text *ngIf=\"props.backText\">{{ props.backText }}</ion-text>\n          </ion-button>\n        </ion-buttons>\n      </ng-container>\n      <ng-container *ngIf=\"props.withActions\">\n        <ion-buttons slot=\"end\" *ngIf=\"someInRight() || props.withMenu\">\n          <ng-container *ngFor=\"let action of rightActions()\">\n            <ion-button *ngIf=\"action.type === actionTypes.ICON\" (click)=\"clickHandler(action.token)\">\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" [color]=\"props.textColor\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(action.token)\"\n            ></val-image>\n            <ion-button\n              [color]=\"props.textColor\"\n              *ngIf=\"action.type === actionTypes.BUTTON\"\n              (click)=\"clickHandler(action.token)\"\n              >{{ action.description }}</ion-button\n            >\n          </ng-container>\n          @if (props.languageSelector) {\n            <val-language-selector [props]=\"languageSelectorProps\"></val-language-selector>\n          }\n          @if (props.withMenu) {\n            <ion-menu-button color=\"dark\"></ion-menu-button>\n          }\n        </ion-buttons>\n        <ion-buttons slot=\"start\" *ngIf=\"someInLeft()\" style=\"padding-left: 4px;\">\n          <ng-container *ngFor=\"let action of leftActions()\">\n            <ion-button *ngIf=\"action.type === actionTypes.ICON\" (click)=\"clickHandler(action.token)\">\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" [color]=\"props.textColor\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(action.token)\"\n            ></val-image>\n            <ion-button *ngIf=\"action.type === actionTypes.BUTTON\" (click)=\"clickHandler(action.token)\">{{\n              action.description\n            }}</ion-button>\n          </ng-container>\n        </ion-buttons>\n      </ng-container>\n      <ion-title *ngIf=\"props.title\" [color]=\"props.textColor\">{{ props.title }}</ion-title>\n      <!-- experimental -->\n      <ng-content select=\"[toolbar-bottom]\"></ng-content>\n    </ion-toolbar>\n  `,\n  styleUrls: ['./toolbar.component.scss'],\n})\nexport class ToolbarComponent implements OnInit {\n  /**\n   * Metadatos de la barra de herramientas.\n   */\n  @Input()\n  props: ToolbarMetadata;\n\n  /**\n   * Evento emitido al hacer click en una acción o botón de la barra.\n   */\n  @Output()\n  onClick = new EventEmitter<string>();\n\n  actionTypes = ToolbarActionType;\n\n  constructor(\n    private navCtrl: NavController,\n    icon: IconService\n  ) {\n    addIcons({ chevronBackOutline });\n  }\n\n  ngOnInit() {}\n\n  /**\n   * Emite el token de la acción clicada.\n   * @param token Token de la acción\n   */\n  clickHandler(token?: string) {\n    this.onClick.emit(token);\n  }\n\n  /**\n   * Navega hacia atrás usando NavController.\n   */\n  goBack() {\n    this.navCtrl.back();\n  }\n\n  /**\n   * Determina si hay acciones a la derecha.\n   */\n  someInRight(): boolean {\n    return !!this.props.actions.find(x => x.position === 'right');\n  }\n\n  /**\n   * Determina si hay acciones a la izquierda.\n   */\n  someInLeft(): boolean {\n    return !!this.props.actions.find(x => x.position === 'left');\n  }\n\n  /**\n   * Devuelve las acciones de la derecha.\n   */\n  rightActions(): ToolbarAction[] {\n    return this.props.actions.filter(x => x.position === 'right');\n  }\n\n  /**\n   * Devuelve las acciones de la izquierda.\n   */\n  leftActions(): ToolbarAction[] {\n    return this.props.actions.filter(x => x.position === 'left');\n  }\n\n  /**\n   * Language selector props with 'icon' mode as default for toolbar display.\n   */\n  get languageSelectorProps() {\n    return {\n      mode: 'icon' as const,\n      color: this.props.textColor || 'dark',\n      ...this.props.languageSelector,\n    };\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL3Rvb2xiYXIvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVGlwb3MgeSBtZXRhZGF0b3MgcGFyYSBlbCBjb21wb25lbnRlIFRvb2xiYXJDb21wb25lbnQuXG4gKiBQZXJtaXRlbiBkZWZpbmlyIGVsIHTDrXR1bG8sIGFjY2lvbmVzLCBjb2xvcmVzIHkgb3BjaW9uZXMgZGUgbmF2ZWdhY2nDs24gZGUgbGEgYmFycmEgZGUgaGVycmFtaWVudGFzLlxuICovXG5pbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcbmltcG9ydCB7IFRvb2xiYXJBY3Rpb24gfSBmcm9tICcuLi8uLi90eXBlcyc7XG5cbi8qKlxuICogTWV0YWRhdG9zIHBhcmEgbGEgYmFycmEgZGUgaGVycmFtaWVudGFzLlxuICogQHByb3BlcnR5IHdpdGhCYWNrIFNpIG11ZXN0cmEgYm90w7NuIGRlIHJldHJvY2Vzby5cbiAqIEBwcm9wZXJ0eSB3aXRoTWVudSAob3BjaW9uYWwpIFNpIG11ZXN0cmEgYm90w7NuIGRlIG1lbsO6LlxuICogQHByb3BlcnR5IGJhY2tUZXh0IChvcGNpb25hbCkgVGV4dG8gZGVsIGJvdMOzbiBkZSByZXRyb2Nlc28uXG4gKiBAcHJvcGVydHkgd2l0aEFjdGlvbnMgU2kgbXVlc3RyYSBhY2Npb25lcy5cbiAqIEBwcm9wZXJ0eSBjb2xvciAob3BjaW9uYWwpIENvbG9yIGRlIGZvbmRvLlxuICogQHByb3BlcnR5IHRleHRDb2xvciAob3BjaW9uYWwpIENvbG9yIGRlbCB0ZXh0by5cbiAqIEBwcm9wZXJ0eSB0aXRsZSBUw610dWxvIGRlIGxhIGJhcnJhLlxuICogQHByb3BlcnR5IGFjdGlvbnMgQWNjaW9uZXMgZGlzcG9uaWJsZXMgZW4gbGEgYmFycmEuXG4gKi9cbmV4cG9ydCB0eXBlIFRvb2xiYXJNZXRhZGF0YSA9IHtcbiAgd2l0aEJhY2s6IGJvb2xlYW47XG4gIHdpdGhNZW51PzogYm9vbGVhbjtcbiAgYmFja1RleHQ/OiBzdHJpbmc7XG4gIHdpdGhBY3Rpb25zOiBib29sZWFuO1xuICBjb2xvcj86IENvbG9yO1xuICB0ZXh0Q29sb3I/OiBDb2xvcjtcbiAgdGl0bGU6IHN0cmluZztcbiAgYWN0aW9uczogVG9vbGJhckFjdGlvbltdO1xufTtcbiJdfQ==
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL3Rvb2xiYXIvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVGlwb3MgeSBtZXRhZGF0b3MgcGFyYSBlbCBjb21wb25lbnRlIFRvb2xiYXJDb21wb25lbnQuXG4gKiBQZXJtaXRlbiBkZWZpbmlyIGVsIHTDrXR1bG8sIGFjY2lvbmVzLCBjb2xvcmVzIHkgb3BjaW9uZXMgZGUgbmF2ZWdhY2nDs24gZGUgbGEgYmFycmEgZGUgaGVycmFtaWVudGFzLlxuICovXG5pbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcbmltcG9ydCB7IFRvb2xiYXJBY3Rpb24gfSBmcm9tICcuLi8uLi90eXBlcyc7XG5pbXBvcnQgeyBMYW5ndWFnZVNlbGVjdG9yTWV0YWRhdGEgfSBmcm9tICcuLi8uLi9tb2xlY3VsZXMvbGFuZ3VhZ2Utc2VsZWN0b3IvdHlwZXMnO1xuXG4vKipcbiAqIE1ldGFkYXRvcyBwYXJhIGxhIGJhcnJhIGRlIGhlcnJhbWllbnRhcy5cbiAqIEBwcm9wZXJ0eSB3aXRoQmFjayBTaSBtdWVzdHJhIGJvdMOzbiBkZSByZXRyb2Nlc28uXG4gKiBAcHJvcGVydHkgd2l0aE1lbnUgKG9wY2lvbmFsKSBTaSBtdWVzdHJhIGJvdMOzbiBkZSBtZW7Dui5cbiAqIEBwcm9wZXJ0eSBiYWNrVGV4dCAob3BjaW9uYWwpIFRleHRvIGRlbCBib3TDs24gZGUgcmV0cm9jZXNvLlxuICogQHByb3BlcnR5IHdpdGhBY3Rpb25zIFNpIG11ZXN0cmEgYWNjaW9uZXMuXG4gKiBAcHJvcGVydHkgY29sb3IgKG9wY2lvbmFsKSBDb2xvciBkZSBmb25kby5cbiAqIEBwcm9wZXJ0eSB0ZXh0Q29sb3IgKG9wY2lvbmFsKSBDb2xvciBkZWwgdGV4dG8uXG4gKiBAcHJvcGVydHkgdGl0bGUgVMOtdHVsbyBkZSBsYSBiYXJyYS5cbiAqIEBwcm9wZXJ0eSBhY3Rpb25zIEFjY2lvbmVzIGRpc3BvbmlibGVzIGVuIGxhIGJhcnJhLlxuICogQHByb3BlcnR5IGxhbmd1YWdlU2VsZWN0b3IgKG9wY2lvbmFsKSBDb25maWd1cmFjacOzbiBkZWwgc2VsZWN0b3IgZGUgaWRpb21hLlxuICovXG5leHBvcnQgdHlwZSBUb29sYmFyTWV0YWRhdGEgPSB7XG4gIHdpdGhCYWNrOiBib29sZWFuO1xuICB3aXRoTWVudT86IGJvb2xlYW47XG4gIGJhY2tUZXh0Pzogc3RyaW5nO1xuICB3aXRoQWN0aW9uczogYm9vbGVhbjtcbiAgY29sb3I/OiBDb2xvcjtcbiAgdGV4dENvbG9yPzogQ29sb3I7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIGFjdGlvbnM6IFRvb2xiYXJBY3Rpb25bXTtcbiAgLyoqIExhbmd1YWdlIHNlbGVjdG9yIGNvbmZpZ3VyYXRpb24uIERpc3BsYXlzIHRvIHRoZSBsZWZ0IG9mIG1lbnUgYnV0dG9uIHdoZW4gcHJvdmlkZWQuICovXG4gIGxhbmd1YWdlU2VsZWN0b3I/OiBMYW5ndWFnZVNlbGVjdG9yTWV0YWRhdGE7XG59O1xuIl19
@@ -57,6 +57,7 @@ export class PageContentComponent {
57
57
  textColor: 'dark',
58
58
  withMenu: true,
59
59
  title: '',
60
+ languageSelector: undefined,
60
61
  actions: [
61
62
  {
62
63
  token: 'header-logo',
@@ -81,9 +82,21 @@ export class PageContentComponent {
81
82
  }
82
83
  /**
83
84
  * Gets header props, using cached default if not provided.
85
+ * Injects languageSelector into toolbar when provided at page level.
84
86
  */
85
87
  get headerProps() {
86
- return this.props.header || this.defaultHeader;
88
+ const header = this.props.header || this.defaultHeader;
89
+ // Inject languageSelector into toolbar if provided at page level
90
+ if (this.props.languageSelector && !header.toolbar.languageSelector) {
91
+ return {
92
+ ...header,
93
+ toolbar: {
94
+ ...header.toolbar,
95
+ languageSelector: this.props.languageSelector,
96
+ },
97
+ };
98
+ }
99
+ return header;
87
100
  }
88
101
  /**
89
102
  * Gets the background color based on theme.
@@ -157,4 +170,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
157
170
  }], onHeaderClick: [{
158
171
  type: Output
159
172
  }] } });
160
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACvE,OAAO,EAAE,UAAU,EAAE,MAAM,2BAA2B,CAAC;AACvD,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAI1E,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;;;;;AAE5D;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AA+BH,MAAM,OAAO,oBAAoB;IAM/B,YACU,KAAmB,EACnB,GAAsB;QADtB,UAAK,GAAL,KAAK,CAAc;QACnB,QAAG,GAAH,GAAG,CAAmB;QAPhC;;WAEG;QACM,UAAK,GAAwB,EAAE,CAAC;QAOzC;;WAEG;QACO,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;QAErD;;WAEG;QACc,kBAAa,GAAG;YAC/B,QAAQ,EAAE,IAAI;YACd,WAAW,EAAE,IAAI;YACjB,OAAO,EAAE;gBACP,QAAQ,EAAE,KAAK;gBACf,WAAW,EAAE,IAAI;gBACjB,SAAS,EAAE,MAAe;gBAC1B,QAAQ,EAAE,IAAI;gBACd,KAAK,EAAE,EAAE;gBACT,OAAO,EAAE;oBACP;wBACE,KAAK,EAAE,aAAa;wBACpB,WAAW,EAAE,EAAE;wBACf,QAAQ,EAAE,MAAe;wBACzB,IAAI,EAAE,OAAgB;wBACtB,KAAK,EAAE;4BACL,KAAK,EAAE,EAAE;4BACT,GAAG,EAAE,aAAa;4BAClB,GAAG,EAAE,aAAa;4BAClB,IAAI,EAAE,KAAc;4BACpB,MAAM,EAAE,KAAK;4BACb,QAAQ,EAAE,KAAK;4BACf,IAAI,EAAE,OAAgB;4BACtB,OAAO,EAAE,KAAK;4BACd,IAAI,EAAE,IAAI;yBACX;qBACF;iBACF;aACF;SACF,CAAC;IAvCC,CAAC;IAyCJ;;OAEG;IACH,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,aAAa,CAAC;IACjD,CAAC;IAED;;OAEG;IACH,aAAa;QACX,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACtB,OAAO,6BAA6B,CAAC;QACvC,CAAC;QAED,MAAM,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC;QACjC,IAAI,CAAC,EAAE,EAAE,CAAC;YACR,OAAO,6BAA6B,CAAC;QACvC,CAAC;QAED,OAAO,YAAY,CAAC,EAAE,CAAC,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,oBAAoB,CAAC,KAAa;QAChC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAE/B,4DAA4D;QAC5D,IAAI,KAAK,KAAK,aAAa,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACpD,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC;+GAnFU,oBAAoB;mGAApB,oBAAoB,qJA1BrB;;;;;;;;;;;;;;;;;;;GAmBT,gGApBS,YAAY,oHAAE,eAAe,gGAAE,UAAU;;4FA2BxC,oBAAoB;kBA9BhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,EAAE,eAAe,EAAE,UAAU,CAAC,YAC1C;;;;;;;;;;;;;;;;;;;GAmBT;iHAWQ,KAAK;sBAAb,KAAK;gBAUI,aAAa;sBAAtB,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <ng-content select=\"[content]\"></ng-content>\n        </main>\n        <ng-content select=\"[footer]\"></ng-content>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageContentComponent {\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  constructor(\n    private theme: ThemeService,\n    private nav: NavigationService\n  ) {}\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   */\n  get headerProps() {\n    return this.props.header || this.defaultHeader;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
173
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <ng-content select=\"[content]\"></ng-content>\n        </main>\n        <ng-content select=\"[footer]\"></ng-content>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageContentComponent {\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  constructor(\n    private theme: ThemeService,\n    private nav: NavigationService\n  ) {}\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      languageSelector: undefined as undefined,\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   * Injects languageSelector into toolbar when provided at page level.\n   */\n  get headerProps() {\n    const header = this.props.header || this.defaultHeader;\n\n    // Inject languageSelector into toolbar if provided at page level\n    if (this.props.languageSelector && !header.toolbar.languageSelector) {\n      return {\n        ...header,\n        toolbar: {\n          ...header.toolbar,\n          languageSelector: this.props.languageSelector,\n        },\n      };\n    }\n\n    return header;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtY29udGVudC90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSGVhZGVyTWV0YWRhdGEgfSBmcm9tICcuLi8uLi9vcmdhbmlzbXMvaGVhZGVyL3R5cGVzJztcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgcGFnZSBjb250ZW50IGNvbXBvbmVudC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBQYWdlQ29udGVudE1ldGFkYXRhIHtcbiAgLyoqIEhlYWRlciBjb25maWd1cmF0aW9uICovXG4gIGhlYWRlcj86IEhlYWRlck1ldGFkYXRhO1xuICAvKiogQmFja2dyb3VuZCBjb2xvciBvciBDU1MgdmFyaWFibGUgKi9cbiAgYmFja2dyb3VuZD86IHN0cmluZztcbiAgLyoqIEJhY2tncm91bmQgY29sb3IgZm9yIGRhcmsgbW9kZSAqL1xuICBiYWNrZ3JvdW5kRGFyaz86IHN0cmluZztcbiAgLyoqIFJvdXRlIHRvIG5hdmlnYXRlIHRvIHdoZW4gaGVhZGVyIGxvZ28gaXMgY2xpY2tlZCAqL1xuICBob21lUm91dGU/OiBzdHJpbmc7XG59XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtY29udGVudC90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSGVhZGVyTWV0YWRhdGEgfSBmcm9tICcuLi8uLi9vcmdhbmlzbXMvaGVhZGVyL3R5cGVzJztcbmltcG9ydCB7IExhbmd1YWdlU2VsZWN0b3JNZXRhZGF0YSB9IGZyb20gJy4uLy4uL21vbGVjdWxlcy9sYW5ndWFnZS1zZWxlY3Rvci90eXBlcyc7XG5cbi8qKlxuICogQ29uZmlndXJhdGlvbiBmb3IgdGhlIHBhZ2UgY29udGVudCBjb21wb25lbnQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUGFnZUNvbnRlbnRNZXRhZGF0YSB7XG4gIC8qKiBIZWFkZXIgY29uZmlndXJhdGlvbiAqL1xuICBoZWFkZXI/OiBIZWFkZXJNZXRhZGF0YTtcbiAgLyoqIEJhY2tncm91bmQgY29sb3Igb3IgQ1NTIHZhcmlhYmxlICovXG4gIGJhY2tncm91bmQ/OiBzdHJpbmc7XG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIGZvciBkYXJrIG1vZGUgKi9cbiAgYmFja2dyb3VuZERhcms/OiBzdHJpbmc7XG4gIC8qKiBSb3V0ZSB0byBuYXZpZ2F0ZSB0byB3aGVuIGhlYWRlciBsb2dvIGlzIGNsaWNrZWQgKi9cbiAgaG9tZVJvdXRlPzogc3RyaW5nO1xuICAvKipcbiAgICogTGFuZ3VhZ2Ugc2VsZWN0b3IgY29uZmlndXJhdGlvbi5cbiAgICogV2hlbiBwcm92aWRlZCwgZGlzcGxheXMgYSBsYW5ndWFnZSBzZWxlY3RvciBpY29uIGluIHRoZSBoZWFkZXIgKGxlZnQgb2YgbWVudSBidXR0b24pLlxuICAgKiBVc2VzICdpY29uJyBtb2RlIGJ5IGRlZmF1bHQgZm9yIGNvbXBhY3QgZGlzcGxheS5cbiAgICovXG4gIGxhbmd1YWdlU2VsZWN0b3I/OiBMYW5ndWFnZVNlbGVjdG9yTWV0YWRhdGE7XG59XG4iXX0=