valtech-components 2.0.565 → 2.0.566

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,6 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, Input } from '@angular/core';
2
+ import { Component, inject, Input } from '@angular/core';
3
+ import { PresetService } from '../../../services/presets';
3
4
  import { BoxComponent } from '../../atoms/box/box.component';
4
5
  import { IconComponent } from '../../atoms/icon/icon.component';
5
6
  import { TextComponent } from '../../atoms/text/text.component';
@@ -8,28 +9,67 @@ import * as i0 from "@angular/core";
8
9
  * val-alert-box
9
10
  *
10
11
  * Displays an alert box with an icon and text, using a styled box container.
12
+ * Supports presets for reusable configurations.
11
13
  *
12
- * @example
14
+ * @example With preset (recommended):
15
+ * <val-alert-box preset="warning" [props]="{ text: 'Advertencia!' }"></val-alert-box>
16
+ *
17
+ * @example Static (backwards compatible):
13
18
  * <val-alert-box [props]="{ box: {...}, icon: {...}, text: { content: 'Alerta' } }"></val-alert-box>
14
19
  *
15
- * @input props: AlertBoxMetadata | ReactiveAlertBoxMetadata - Configuration for the alert box
20
+ * @input preset: string - Name of preset to apply (e.g., 'info', 'success', 'warning', 'error')
21
+ * @input props: AlertBoxMetadata | ReactiveAlertBoxMetadata - Configuration for the alert box (overrides preset values)
16
22
  */
17
23
  export class AlertBoxComponent {
24
+ constructor() {
25
+ this.presets = inject(PresetService);
26
+ /**
27
+ * Alert box configuration object. Values here override preset values.
28
+ * Supports both legacy AlertBoxMetadata and ReactiveAlertBoxMetadata patterns.
29
+ */
30
+ this.props = {};
31
+ /**
32
+ * Resolved props after merging preset + explicit props.
33
+ */
34
+ this.resolvedProps = {};
35
+ }
18
36
  /** Whether this is using the legacy props pattern */
19
37
  get isLegacyProps() {
20
- return 'text' in this.props && typeof this.props.text === 'object';
38
+ return 'text' in this.resolvedProps && typeof this.resolvedProps.text === 'object';
21
39
  }
22
40
  /** Get text props for legacy pattern */
23
41
  getLegacyTextProps() {
24
- return this.props.text;
42
+ return this.resolvedProps.text;
25
43
  }
26
44
  ngOnInit() {
45
+ this.resolveProps();
27
46
  if (!this.isLegacyProps) {
28
47
  this.initializeTextProps();
29
48
  }
30
49
  }
50
+ ngOnChanges(changes) {
51
+ if (changes['preset'] || changes['props']) {
52
+ this.resolveProps();
53
+ if (!this.isLegacyProps) {
54
+ this.initializeTextProps();
55
+ }
56
+ }
57
+ }
58
+ /**
59
+ * Merge preset configuration with explicit props.
60
+ * Explicit props take precedence over preset values.
61
+ */
62
+ resolveProps() {
63
+ const presetProps = this.preset
64
+ ? this.presets.get('alertBox', this.preset)
65
+ : {};
66
+ this.resolvedProps = {
67
+ ...presetProps,
68
+ ...this.props,
69
+ };
70
+ }
31
71
  initializeTextProps() {
32
- const reactiveProps = this.props;
72
+ const reactiveProps = this.resolvedProps;
33
73
  // Base text properties with styling
34
74
  this.computedTextProps = {
35
75
  size: reactiveProps.textStyle?.size || 'medium',
@@ -42,10 +82,10 @@ export class AlertBoxComponent {
42
82
  }
43
83
  }
44
84
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AlertBoxComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
45
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: AlertBoxComponent, isStandalone: true, selector: "val-alert-box", inputs: { props: "props" }, ngImport: i0, template: `
46
- <val-box [props]="props.box">
85
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: AlertBoxComponent, isStandalone: true, selector: "val-alert-box", inputs: { preset: "preset", props: "props" }, usesOnChanges: true, ngImport: i0, template: `
86
+ <val-box [props]="resolvedProps.box">
47
87
  <div class="content-container" body>
48
- <val-icon [props]="props.icon"></val-icon>
88
+ <val-icon [props]="resolvedProps.icon"></val-icon>
49
89
  <!-- Support both legacy and reactive patterns -->
50
90
  @if (isLegacyProps) {
51
91
  <val-text class="text" [props]="getLegacyTextProps()" />
@@ -59,9 +99,9 @@ export class AlertBoxComponent {
59
99
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AlertBoxComponent, decorators: [{
60
100
  type: Component,
61
101
  args: [{ selector: 'val-alert-box', standalone: true, imports: [CommonModule, BoxComponent, IconComponent, TextComponent], template: `
62
- <val-box [props]="props.box">
102
+ <val-box [props]="resolvedProps.box">
63
103
  <div class="content-container" body>
64
- <val-icon [props]="props.icon"></val-icon>
104
+ <val-icon [props]="resolvedProps.icon"></val-icon>
65
105
  <!-- Support both legacy and reactive patterns -->
66
106
  @if (isLegacyProps) {
67
107
  <val-text class="text" [props]="getLegacyTextProps()" />
@@ -71,7 +111,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
71
111
  </div>
72
112
  </val-box>
73
113
  `, 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)}.text{margin-left:.25rem}.content-container{display:flex;align-items:flex-start}\n"] }]
74
- }], propDecorators: { props: [{
114
+ }], propDecorators: { preset: [{
115
+ type: Input
116
+ }], props: [{
75
117
  type: Input
76
118
  }] } });
77
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWxlcnQtYm94LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9tb2xlY3VsZXMvYWxlcnQtYm94L2FsZXJ0LWJveC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ3pELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUM3RCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDaEUsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGlDQUFpQyxDQUFDOztBQXNCaEU7Ozs7Ozs7OztHQVNHO0FBQ0gsTUFBTSxPQUFPLGlCQUFpQjtJQVc1QixxREFBcUQ7SUFDckQsSUFBSSxhQUFhO1FBQ2YsT0FBTyxNQUFNLElBQUksSUFBSSxDQUFDLEtBQUssSUFBSSxPQUFRLElBQUksQ0FBQyxLQUFhLENBQUMsSUFBSSxLQUFLLFFBQVEsQ0FBQztJQUM5RSxDQUFDO0lBRUQsd0NBQXdDO0lBQ3hDLGtCQUFrQjtRQUNoQixPQUFRLElBQUksQ0FBQyxLQUEwQixDQUFDLElBQUksQ0FBQztJQUMvQyxDQUFDO0lBRUQsUUFBUTtRQUNOLElBQUksQ0FBQyxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDeEIsSUFBSSxDQUFDLG1CQUFtQixFQUFFLENBQUM7UUFDN0IsQ0FBQztJQUNILENBQUM7SUFFTyxtQkFBbUI7UUFDekIsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLEtBQWlDLENBQUM7UUFFN0Qsb0NBQW9DO1FBQ3BDLElBQUksQ0FBQyxpQkFBaUIsR0FBRztZQUN2QixJQUFJLEVBQUUsYUFBYSxDQUFDLFNBQVMsRUFBRSxJQUFJLElBQUksUUFBUTtZQUMvQyxLQUFLLEVBQUUsYUFBYSxDQUFDLFNBQVMsRUFBRSxLQUFLO1lBQ3JDLElBQUksRUFBRSxhQUFhLENBQUMsU0FBUyxFQUFFLElBQUksSUFBSSxLQUFLO1NBQzdDLENBQUM7UUFFRixxQ0FBcUM7UUFDckMsSUFBSSxhQUFhLENBQUMsSUFBSSxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQ3JDLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxPQUFPLEdBQUcsYUFBYSxDQUFDLElBQUksQ0FBQztRQUN0RCxDQUFDO0lBQ0gsQ0FBQzsrR0F6Q1UsaUJBQWlCO21HQUFqQixpQkFBaUIscUdBekJsQjs7Ozs7Ozs7Ozs7O0dBWVQsK3BGQWJTLFlBQVksK0JBQUUsWUFBWSw2RkFBRSxhQUFhLHdFQUFFLGFBQWE7OzRGQTBCdkQsaUJBQWlCO2tCQTdCN0IsU0FBUzsrQkFDRSxlQUFlLGNBQ2IsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLFlBQVksRUFBRSxhQUFhLEVBQUUsYUFBYSxDQUFDLFlBQ3pEOzs7Ozs7Ozs7Ozs7R0FZVDs4QkFtQkQsS0FBSztzQkFESixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIE9uSW5pdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQm94Q29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvYm94L2JveC5jb21wb25lbnQnO1xuaW1wb3J0IHsgSWNvbkNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2ljb24vaWNvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgVGV4dENvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL3RleHQvdGV4dC5jb21wb25lbnQnO1xuaW1wb3J0IHsgQWxlcnRCb3hNZXRhZGF0YSwgUmVhY3RpdmVBbGVydEJveE1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC1hbGVydC1ib3gnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBCb3hDb21wb25lbnQsIEljb25Db21wb25lbnQsIFRleHRDb21wb25lbnRdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDx2YWwtYm94IFtwcm9wc109XCJwcm9wcy5ib3hcIj5cbiAgICAgIDxkaXYgY2xhc3M9XCJjb250ZW50LWNvbnRhaW5lclwiIGJvZHk+XG4gICAgICAgIDx2YWwtaWNvbiBbcHJvcHNdPVwicHJvcHMuaWNvblwiPjwvdmFsLWljb24+XG4gICAgICAgIDwhLS0gU3VwcG9ydCBib3RoIGxlZ2FjeSBhbmQgcmVhY3RpdmUgcGF0dGVybnMgLS0+XG4gICAgICAgIEBpZiAoaXNMZWdhY3lQcm9wcykge1xuICAgICAgICAgIDx2YWwtdGV4dCBjbGFzcz1cInRleHRcIiBbcHJvcHNdPVwiZ2V0TGVnYWN5VGV4dFByb3BzKClcIiAvPlxuICAgICAgICB9IEBlbHNlIHtcbiAgICAgICAgICA8dmFsLXRleHQgY2xhc3M9XCJ0ZXh0XCIgW3Byb3BzXT1cImNvbXB1dGVkVGV4dFByb3BzXCIgLz5cbiAgICAgICAgfVxuICAgICAgPC9kaXY+XG4gICAgPC92YWwtYm94PlxuICBgLFxuICBzdHlsZVVybHM6IFsnLi9hbGVydC1ib3guY29tcG9uZW50LnNjc3MnXSxcbn0pXG4vKipcbiAqIHZhbC1hbGVydC1ib3hcbiAqXG4gKiBEaXNwbGF5cyBhbiBhbGVydCBib3ggd2l0aCBhbiBpY29uIGFuZCB0ZXh0LCB1c2luZyBhIHN0eWxlZCBib3ggY29udGFpbmVyLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLWFsZXJ0LWJveCBbcHJvcHNdPVwieyBib3g6IHsuLi59LCBpY29uOiB7Li4ufSwgdGV4dDogeyBjb250ZW50OiAnQWxlcnRhJyB9IH1cIj48L3ZhbC1hbGVydC1ib3g+XG4gKlxuICogQGlucHV0IHByb3BzOiBBbGVydEJveE1ldGFkYXRhIHwgUmVhY3RpdmVBbGVydEJveE1ldGFkYXRhIC0gQ29uZmlndXJhdGlvbiBmb3IgdGhlIGFsZXJ0IGJveFxuICovXG5leHBvcnQgY2xhc3MgQWxlcnRCb3hDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICAvKipcbiAgICogQWxlcnQgYm94IGNvbmZpZ3VyYXRpb24gb2JqZWN0LlxuICAgKiBTdXBwb3J0cyBib3RoIGxlZ2FjeSBBbGVydEJveE1ldGFkYXRhIGFuZCBSZWFjdGl2ZUFsZXJ0Qm94TWV0YWRhdGEgcGF0dGVybnMuXG4gICAqL1xuICBASW5wdXQoKVxuICBwcm9wczogQWxlcnRCb3hNZXRhZGF0YSB8IFJlYWN0aXZlQWxlcnRCb3hNZXRhZGF0YTtcblxuICAvKiogQ29tcHV0ZWQgdGV4dCBwcm9wZXJ0aWVzIGZvciByZWFjdGl2ZSBwYXR0ZXJuICovXG4gIGNvbXB1dGVkVGV4dFByb3BzOiBhbnk7XG5cbiAgLyoqIFdoZXRoZXIgdGhpcyBpcyB1c2luZyB0aGUgbGVnYWN5IHByb3BzIHBhdHRlcm4gKi9cbiAgZ2V0IGlzTGVnYWN5UHJvcHMoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuICd0ZXh0JyBpbiB0aGlzLnByb3BzICYmIHR5cGVvZiAodGhpcy5wcm9wcyBhcyBhbnkpLnRleHQgPT09ICdvYmplY3QnO1xuICB9XG5cbiAgLyoqIEdldCB0ZXh0IHByb3BzIGZvciBsZWdhY3kgcGF0dGVybiAqL1xuICBnZXRMZWdhY3lUZXh0UHJvcHMoKTogYW55IHtcbiAgICByZXR1cm4gKHRoaXMucHJvcHMgYXMgQWxlcnRCb3hNZXRhZGF0YSkudGV4dDtcbiAgfVxuXG4gIG5nT25Jbml0KCkge1xuICAgIGlmICghdGhpcy5pc0xlZ2FjeVByb3BzKSB7XG4gICAgICB0aGlzLmluaXRpYWxpemVUZXh0UHJvcHMoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGluaXRpYWxpemVUZXh0UHJvcHMoKSB7XG4gICAgY29uc3QgcmVhY3RpdmVQcm9wcyA9IHRoaXMucHJvcHMgYXMgUmVhY3RpdmVBbGVydEJveE1ldGFkYXRhO1xuXG4gICAgLy8gQmFzZSB0ZXh0IHByb3BlcnRpZXMgd2l0aCBzdHlsaW5nXG4gICAgdGhpcy5jb21wdXRlZFRleHRQcm9wcyA9IHtcbiAgICAgIHNpemU6IHJlYWN0aXZlUHJvcHMudGV4dFN0eWxlPy5zaXplIHx8ICdtZWRpdW0nLFxuICAgICAgY29sb3I6IHJlYWN0aXZlUHJvcHMudGV4dFN0eWxlPy5jb2xvcixcbiAgICAgIGJvbGQ6IHJlYWN0aXZlUHJvcHMudGV4dFN0eWxlPy5ib2xkIHx8IGZhbHNlLFxuICAgIH07XG5cbiAgICAvLyBBZGQgY29udGVudCAtIHVzZSBzdGF0aWMgdGV4dCBvbmx5XG4gICAgaWYgKHJlYWN0aXZlUHJvcHMudGV4dCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICB0aGlzLmNvbXB1dGVkVGV4dFByb3BzLmNvbnRlbnQgPSByZWFjdGl2ZVByb3BzLnRleHQ7XG4gICAgfVxuICB9XG59XG4iXX0=
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alert-box.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/alert-box/alert-box.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAoC,MAAM,eAAe,CAAC;AAC3F,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;AAC7D,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC;AAChE,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC;;AAsBhE;;;;;;;;;;;;;;GAcG;AACH,MAAM,OAAO,iBAAiB;IAlC9B;QAmCU,YAAO,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QAWxC;;;WAGG;QACM,UAAK,GAAyD,EAAE,CAAC;QAE1E;;WAEG;QACH,kBAAa,GAAgD,EAAsB,CAAC;KA6DrF;IAxDC,qDAAqD;IACrD,IAAI,aAAa;QACf,OAAO,MAAM,IAAI,IAAI,CAAC,aAAa,IAAI,OAAQ,IAAI,CAAC,aAAqB,CAAC,IAAI,KAAK,QAAQ,CAAC;IAC9F,CAAC;IAED,wCAAwC;IACxC,kBAAkB;QAChB,OAAQ,IAAI,CAAC,aAAkC,CAAC,IAAI,CAAC;IACvD,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;YACxB,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YAC1C,IAAI,CAAC,YAAY,EAAE,CAAC;YACpB,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;gBACxB,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC7B,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,YAAY;QAClB,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM;YAC7B,CAAC,CAAE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,UAAU,EAAE,IAAI,CAAC,MAAM,CAA0D;YACrG,CAAC,CAAC,EAAE,CAAC;QAEP,IAAI,CAAC,aAAa,GAAG;YACnB,GAAG,WAAW;YACd,GAAG,IAAI,CAAC,KAAK;SACiC,CAAC;IACnD,CAAC;IAEO,mBAAmB;QACzB,MAAM,aAAa,GAAG,IAAI,CAAC,aAAyC,CAAC;QAErE,oCAAoC;QACpC,IAAI,CAAC,iBAAiB,GAAG;YACvB,IAAI,EAAE,aAAa,CAAC,SAAS,EAAE,IAAI,IAAI,QAAQ;YAC/C,KAAK,EAAE,aAAa,CAAC,SAAS,EAAE,KAAK;YACrC,IAAI,EAAE,aAAa,CAAC,SAAS,EAAE,IAAI,IAAI,KAAK;SAC7C,CAAC;QAEF,qCAAqC;QACrC,IAAI,aAAa,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YACrC,IAAI,CAAC,iBAAiB,CAAC,OAAO,GAAG,aAAa,CAAC,IAAI,CAAC;QACtD,CAAC;IACH,CAAC;+GAjFU,iBAAiB;mGAAjB,iBAAiB,4IA9BlB;;;;;;;;;;;;GAYT,+pFAbS,YAAY,+BAAE,YAAY,6FAAE,aAAa,wEAAE,aAAa;;4FA+BvD,iBAAiB;kBAlC7B,SAAS;+BACE,eAAe,cACb,IAAI,WACP,CAAC,YAAY,EAAE,YAAY,EAAE,aAAa,EAAE,aAAa,CAAC,YACzD;;;;;;;;;;;;GAYT;8BA4BQ,MAAM;sBAAd,KAAK;gBAMG,KAAK;sBAAb,KAAK","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject, Input, OnChanges, OnInit, SimpleChanges } from '@angular/core';\nimport { PresetService } from '../../../services/presets';\nimport { BoxComponent } from '../../atoms/box/box.component';\nimport { IconComponent } from '../../atoms/icon/icon.component';\nimport { TextComponent } from '../../atoms/text/text.component';\nimport { AlertBoxMetadata, ReactiveAlertBoxMetadata } from './types';\n\n@Component({\n  selector: 'val-alert-box',\n  standalone: true,\n  imports: [CommonModule, BoxComponent, IconComponent, TextComponent],\n  template: `\n    <val-box [props]=\"resolvedProps.box\">\n      <div class=\"content-container\" body>\n        <val-icon [props]=\"resolvedProps.icon\"></val-icon>\n        <!-- Support both legacy and reactive patterns -->\n        @if (isLegacyProps) {\n          <val-text class=\"text\" [props]=\"getLegacyTextProps()\" />\n        } @else {\n          <val-text class=\"text\" [props]=\"computedTextProps\" />\n        }\n      </div>\n    </val-box>\n  `,\n  styleUrls: ['./alert-box.component.scss'],\n})\n/**\n * val-alert-box\n *\n * Displays an alert box with an icon and text, using a styled box container.\n * Supports presets for reusable configurations.\n *\n * @example With preset (recommended):\n * <val-alert-box preset=\"warning\" [props]=\"{ text: 'Advertencia!' }\"></val-alert-box>\n *\n * @example Static (backwards compatible):\n * <val-alert-box [props]=\"{ box: {...}, icon: {...}, text: { content: 'Alerta' } }\"></val-alert-box>\n *\n * @input preset: string - Name of preset to apply (e.g., 'info', 'success', 'warning', 'error')\n * @input props: AlertBoxMetadata | ReactiveAlertBoxMetadata - Configuration for the alert box (overrides preset values)\n */\nexport class AlertBoxComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  /**\n   * Preset name to apply. Presets define reusable alert configurations\n   * (color, icon, box style) that can be registered at app level.\n   *\n   * @example\n   * <val-alert-box preset=\"warning\" [props]=\"{ text: 'Warning message' }\"></val-alert-box>\n   */\n  @Input() preset?: string;\n\n  /**\n   * Alert box configuration object. Values here override preset values.\n   * Supports both legacy AlertBoxMetadata and ReactiveAlertBoxMetadata patterns.\n   */\n  @Input() props: Partial<AlertBoxMetadata | ReactiveAlertBoxMetadata> = {};\n\n  /**\n   * Resolved props after merging preset + explicit props.\n   */\n  resolvedProps: AlertBoxMetadata | ReactiveAlertBoxMetadata = {} as AlertBoxMetadata;\n\n  /** Computed text properties for reactive pattern */\n  computedTextProps: any;\n\n  /** Whether this is using the legacy props pattern */\n  get isLegacyProps(): boolean {\n    return 'text' in this.resolvedProps && typeof (this.resolvedProps as any).text === 'object';\n  }\n\n  /** Get text props for legacy pattern */\n  getLegacyTextProps(): any {\n    return (this.resolvedProps as AlertBoxMetadata).text;\n  }\n\n  ngOnInit() {\n    this.resolveProps();\n    if (!this.isLegacyProps) {\n      this.initializeTextProps();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['preset'] || changes['props']) {\n      this.resolveProps();\n      if (!this.isLegacyProps) {\n        this.initializeTextProps();\n      }\n    }\n  }\n\n  /**\n   * Merge preset configuration with explicit props.\n   * Explicit props take precedence over preset values.\n   */\n  private resolveProps(): void {\n    const presetProps = this.preset\n      ? (this.presets.get('alertBox', this.preset) as Partial<AlertBoxMetadata | ReactiveAlertBoxMetadata>)\n      : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as AlertBoxMetadata | ReactiveAlertBoxMetadata;\n  }\n\n  private initializeTextProps() {\n    const reactiveProps = this.resolvedProps as ReactiveAlertBoxMetadata;\n\n    // Base text properties with styling\n    this.computedTextProps = {\n      size: reactiveProps.textStyle?.size || 'medium',\n      color: reactiveProps.textStyle?.color,\n      bold: reactiveProps.textStyle?.bold || false,\n    };\n\n    // Add content - use static text only\n    if (reactiveProps.text !== undefined) {\n      this.computedTextProps.content = reactiveProps.text;\n    }\n  }\n}\n"]}
@@ -1,6 +1,7 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
3
3
  import { IonButton, IonButtons, IonCard, IonCardContent, IonCardHeader, IonCardSubtitle, IonCardTitle, IonCheckbox, IonIcon, } from '@ionic/angular/standalone';
4
+ import { PresetService } from '../../../services/presets';
4
5
  import { AvatarComponent } from '../../atoms/avatar/avatar.component';
5
6
  import { ButtonComponent } from '../../atoms/button/button.component';
6
7
  import { ImageComponent } from '../../atoms/image/image.component';
@@ -13,15 +14,37 @@ import * as i1 from "@angular/common";
13
14
  * val-card
14
15
  *
15
16
  * A flexible card component supporting images, titles, content, actions, and custom sections.
17
+ * Supports presets for reusable configurations.
16
18
  *
17
- * @example
18
- * <val-card [props]="{ type: 'native', title: 'Card', image: 'url', content: '...', actions: [...] }" (onClick)="handler($event)"></val-card>
19
+ * @example With preset (recommended):
20
+ * <val-card preset="feature" [props]="{ title: 'Card', image: 'url', content: '...' }" (onClick)="handler($event)"></val-card>
19
21
  *
20
- * @input props: CardMetadata - Configuration for the card (type, title, image, content, actions, etc.)
22
+ * @example Static (backwards compatible):
23
+ * <val-card [props]="{ type: 'native', title: 'Card', image: 'url', content: '...' }" (onClick)="handler($event)"></val-card>
24
+ *
25
+ * @input preset: string - Name of preset to apply (e.g., 'feature', 'compact')
26
+ * @input props: CardMetadata - Configuration for the card (overrides preset values)
21
27
  * @output onClick - Emits a CardClickEvent when the card or an action is clicked
22
28
  */
23
29
  export class CardComponent {
24
30
  constructor() {
31
+ this.presets = inject(PresetService);
32
+ /**
33
+ * Card configuration object. Values here override preset values.
34
+ * @type {CardMetadata}
35
+ * @property type - The card type (see CardType).
36
+ * @property title - The card title.
37
+ * @property image - The card image URL.
38
+ * @property content - The card content.
39
+ * @property actions - Array of action buttons (optional).
40
+ * @property sections - Custom card sections (optional).
41
+ * @property overtitle - Subtitle or overtitle (optional).
42
+ */
43
+ this.props = {};
44
+ /**
45
+ * Resolved props after merging preset + explicit props.
46
+ */
47
+ this.resolvedProps = {};
25
48
  /**
26
49
  * Event emitted when the card or an action is clicked.
27
50
  */
@@ -30,64 +53,82 @@ export class CardComponent {
30
53
  this.actionTypes = ToolbarActionType;
31
54
  this.sections = CardSection;
32
55
  }
33
- ngOnInit() { }
56
+ ngOnInit() {
57
+ this.resolveProps();
58
+ }
59
+ ngOnChanges(changes) {
60
+ if (changes['preset'] || changes['props']) {
61
+ this.resolveProps();
62
+ }
63
+ }
64
+ /**
65
+ * Merge preset configuration with explicit props.
66
+ * Explicit props take precedence over preset values.
67
+ */
68
+ resolveProps() {
69
+ const presetProps = this.preset ? this.presets.get('card', this.preset) : {};
70
+ this.resolvedProps = {
71
+ ...presetProps,
72
+ ...this.props,
73
+ };
74
+ }
34
75
  clickHandler(section, token) {
35
76
  this.onClick.emit({ section, token });
36
77
  }
37
78
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
38
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CardComponent, isStandalone: true, selector: "val-card", inputs: { props: "props" }, outputs: { onClick: "onClick" }, ngImport: i0, template: `
39
- <ion-card *ngIf="props.type === types.native">
40
- <img alt="image" [src]="props.image" />
41
- <ion-card-header *ngIf="props.title || props.overtitle">
42
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
43
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
79
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CardComponent, isStandalone: true, selector: "val-card", inputs: { preset: "preset", props: "props" }, outputs: { onClick: "onClick" }, usesOnChanges: true, ngImport: i0, template: `
80
+ <ion-card *ngIf="resolvedProps.type === types.native">
81
+ <img alt="image" [src]="resolvedProps.image" />
82
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle">
83
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
84
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
44
85
  </ion-card-header>
45
86
 
46
- <ion-card-content *ngIf="props.content">{{ props.content }}</ion-card-content>
87
+ <ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
47
88
 
48
89
  <val-button
49
- *ngFor="let b of props.footerActions"
90
+ *ngFor="let b of resolvedProps.footerActions"
50
91
  [props]="b"
51
92
  (onClick)="clickHandler(sections.footer, b.token)"
52
93
  ></val-button>
53
94
  </ion-card>
54
95
 
55
96
  <ion-card
56
- *ngIf="props.type === types.tappable"
57
- (click)="clickHandler(sections.content, props.token)"
97
+ *ngIf="resolvedProps.type === types.tappable"
98
+ (click)="clickHandler(sections.content, resolvedProps.token)"
58
99
  class="tapable"
59
100
  >
60
- <img alt="image" [src]="props.image" />
61
- <ion-card-header *ngIf="props.title || props.overtitle">
62
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
63
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
101
+ <img alt="image" [src]="resolvedProps.image" />
102
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle">
103
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
104
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
64
105
  </ion-card-header>
65
106
 
66
- <ion-card-content *ngIf="props.content">{{ props.content }}</ion-card-content>
107
+ <ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
67
108
  </ion-card>
68
109
 
69
110
  <ion-card
70
- *ngIf="props.type === types.checker"
71
- (click)="clickHandler(sections.content, props.token)"
111
+ *ngIf="resolvedProps.type === types.checker"
112
+ (click)="clickHandler(sections.content, resolvedProps.token)"
72
113
  class="tapable"
73
114
  >
74
- <ion-card-header *ngIf="props.title || props.overtitle" class="checker">
115
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle" class="checker">
75
116
  <div>
76
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
77
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
117
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
118
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
78
119
  </div>
79
120
  <div>
80
- <ion-checkbox [checked]="props.selected"></ion-checkbox>
121
+ <ion-checkbox [checked]="resolvedProps.selected"></ion-checkbox>
81
122
  </div>
82
123
  </ion-card-header>
83
124
 
84
- <ion-card-content *ngIf="props.content">{{ props.content }}</ion-card-content>
125
+ <ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
85
126
  </ion-card>
86
127
 
87
- <ion-card *ngIf="props.type === types.complex" class="complex">
128
+ <ion-card *ngIf="resolvedProps.type === types.complex" class="complex">
88
129
  <ion-card-header class="complex-header">
89
- <ion-buttons style="display: flex; align-items: center" *ngIf="props.leftActions.length > 0">
90
- <ng-container *ngFor="let action of props.leftActions">
130
+ <ion-buttons style="display: flex; align-items: center" *ngIf="resolvedProps.leftActions?.length > 0">
131
+ <ng-container *ngFor="let action of resolvedProps.leftActions">
91
132
  <ion-button
92
133
  *ngIf="action.type === actionTypes.ICON"
93
134
  (click)="clickHandler(sections.headerLeft, action.token)"
@@ -111,13 +152,13 @@ export class CardComponent {
111
152
  >
112
153
  {{ action.description }}
113
154
  </ion-button>
114
- <div *ngIf="props.headerText">
115
- <val-text [props]="{ content: props.headerText, color: 'dark', bold: true, size: 'medium' }" />
155
+ <div *ngIf="resolvedProps.headerText">
156
+ <val-text [props]="{ content: resolvedProps.headerText, color: 'dark', bold: true, size: 'medium' }" />
116
157
  </div>
117
158
  </ng-container>
118
159
  </ion-buttons>
119
- <ion-buttons style="display: flex; align-items: center" *ngIf="props.rightActions.length > 0">
120
- <ng-container *ngFor="let action of props.rightActions">
160
+ <ion-buttons style="display: flex; align-items: center" *ngIf="resolvedProps.rightActions?.length > 0">
161
+ <ng-container *ngFor="let action of resolvedProps.rightActions">
121
162
  <ion-button
122
163
  *ngIf="action.type === actionTypes.ICON"
123
164
  (click)="clickHandler(sections.headerRight, action.token)"
@@ -145,27 +186,27 @@ export class CardComponent {
145
186
  </ion-buttons>
146
187
  </ion-card-header>
147
188
 
148
- <div class="tapable" (click)="clickHandler(sections.content, props.token)">
149
- <ion-card-header *ngIf="props.title || props.overtitle" class="complex-header">
189
+ <div class="tapable" (click)="clickHandler(sections.content, resolvedProps.token)">
190
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle" class="complex-header">
150
191
  <div>
151
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
152
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
192
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
193
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
153
194
  </div>
154
195
  </ion-card-header>
155
196
 
156
- <img alt="image" [src]="props.image" />
157
- <ion-card-content *ngIf="props.content" class="complex-content">{{ props.content }}</ion-card-content>
197
+ <img alt="image" [src]="resolvedProps.image" />
198
+ <ion-card-content *ngIf="resolvedProps.content" class="complex-content">{{ resolvedProps.content }}</ion-card-content>
158
199
  </div>
159
200
  <val-button
160
- *ngFor="let b of props.footerActions"
201
+ *ngFor="let b of resolvedProps.footerActions"
161
202
  [props]="b"
162
203
  (onClick)="clickHandler(sections.footer, b.token)"
163
204
  ></val-button>
164
205
  <ion-buttons
165
206
  style="display: flex; align-items: center; justify-content: flex-end; margin: 8px"
166
- *ngIf="props.footerComplexActions.length > 0"
207
+ *ngIf="resolvedProps.footerComplexActions?.length > 0"
167
208
  >
168
- <ng-container *ngFor="let action of props.footerComplexActions">
209
+ <ng-container *ngFor="let action of resolvedProps.footerComplexActions">
169
210
  <ion-button
170
211
  *ngIf="action.type === actionTypes.ICON"
171
212
  (click)="clickHandler(sections.footerExtra, action.token)"
@@ -213,58 +254,58 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
213
254
  IonButton,
214
255
  IonIcon,
215
256
  ], template: `
216
- <ion-card *ngIf="props.type === types.native">
217
- <img alt="image" [src]="props.image" />
218
- <ion-card-header *ngIf="props.title || props.overtitle">
219
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
220
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
257
+ <ion-card *ngIf="resolvedProps.type === types.native">
258
+ <img alt="image" [src]="resolvedProps.image" />
259
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle">
260
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
261
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
221
262
  </ion-card-header>
222
263
 
223
- <ion-card-content *ngIf="props.content">{{ props.content }}</ion-card-content>
264
+ <ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
224
265
 
225
266
  <val-button
226
- *ngFor="let b of props.footerActions"
267
+ *ngFor="let b of resolvedProps.footerActions"
227
268
  [props]="b"
228
269
  (onClick)="clickHandler(sections.footer, b.token)"
229
270
  ></val-button>
230
271
  </ion-card>
231
272
 
232
273
  <ion-card
233
- *ngIf="props.type === types.tappable"
234
- (click)="clickHandler(sections.content, props.token)"
274
+ *ngIf="resolvedProps.type === types.tappable"
275
+ (click)="clickHandler(sections.content, resolvedProps.token)"
235
276
  class="tapable"
236
277
  >
237
- <img alt="image" [src]="props.image" />
238
- <ion-card-header *ngIf="props.title || props.overtitle">
239
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
240
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
278
+ <img alt="image" [src]="resolvedProps.image" />
279
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle">
280
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
281
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
241
282
  </ion-card-header>
242
283
 
243
- <ion-card-content *ngIf="props.content">{{ props.content }}</ion-card-content>
284
+ <ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
244
285
  </ion-card>
245
286
 
246
287
  <ion-card
247
- *ngIf="props.type === types.checker"
248
- (click)="clickHandler(sections.content, props.token)"
288
+ *ngIf="resolvedProps.type === types.checker"
289
+ (click)="clickHandler(sections.content, resolvedProps.token)"
249
290
  class="tapable"
250
291
  >
251
- <ion-card-header *ngIf="props.title || props.overtitle" class="checker">
292
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle" class="checker">
252
293
  <div>
253
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
254
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
294
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
295
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
255
296
  </div>
256
297
  <div>
257
- <ion-checkbox [checked]="props.selected"></ion-checkbox>
298
+ <ion-checkbox [checked]="resolvedProps.selected"></ion-checkbox>
258
299
  </div>
259
300
  </ion-card-header>
260
301
 
261
- <ion-card-content *ngIf="props.content">{{ props.content }}</ion-card-content>
302
+ <ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
262
303
  </ion-card>
263
304
 
264
- <ion-card *ngIf="props.type === types.complex" class="complex">
305
+ <ion-card *ngIf="resolvedProps.type === types.complex" class="complex">
265
306
  <ion-card-header class="complex-header">
266
- <ion-buttons style="display: flex; align-items: center" *ngIf="props.leftActions.length > 0">
267
- <ng-container *ngFor="let action of props.leftActions">
307
+ <ion-buttons style="display: flex; align-items: center" *ngIf="resolvedProps.leftActions?.length > 0">
308
+ <ng-container *ngFor="let action of resolvedProps.leftActions">
268
309
  <ion-button
269
310
  *ngIf="action.type === actionTypes.ICON"
270
311
  (click)="clickHandler(sections.headerLeft, action.token)"
@@ -288,13 +329,13 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
288
329
  >
289
330
  {{ action.description }}
290
331
  </ion-button>
291
- <div *ngIf="props.headerText">
292
- <val-text [props]="{ content: props.headerText, color: 'dark', bold: true, size: 'medium' }" />
332
+ <div *ngIf="resolvedProps.headerText">
333
+ <val-text [props]="{ content: resolvedProps.headerText, color: 'dark', bold: true, size: 'medium' }" />
293
334
  </div>
294
335
  </ng-container>
295
336
  </ion-buttons>
296
- <ion-buttons style="display: flex; align-items: center" *ngIf="props.rightActions.length > 0">
297
- <ng-container *ngFor="let action of props.rightActions">
337
+ <ion-buttons style="display: flex; align-items: center" *ngIf="resolvedProps.rightActions?.length > 0">
338
+ <ng-container *ngFor="let action of resolvedProps.rightActions">
298
339
  <ion-button
299
340
  *ngIf="action.type === actionTypes.ICON"
300
341
  (click)="clickHandler(sections.headerRight, action.token)"
@@ -322,27 +363,27 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
322
363
  </ion-buttons>
323
364
  </ion-card-header>
324
365
 
325
- <div class="tapable" (click)="clickHandler(sections.content, props.token)">
326
- <ion-card-header *ngIf="props.title || props.overtitle" class="complex-header">
366
+ <div class="tapable" (click)="clickHandler(sections.content, resolvedProps.token)">
367
+ <ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle" class="complex-header">
327
368
  <div>
328
- <ion-card-subtitle *ngIf="props.overtitle">{{ props.overtitle }}</ion-card-subtitle>
329
- <ion-card-title *ngIf="props.title">{{ props.title }}</ion-card-title>
369
+ <ion-card-subtitle *ngIf="resolvedProps.overtitle">{{ resolvedProps.overtitle }}</ion-card-subtitle>
370
+ <ion-card-title *ngIf="resolvedProps.title">{{ resolvedProps.title }}</ion-card-title>
330
371
  </div>
331
372
  </ion-card-header>
332
373
 
333
- <img alt="image" [src]="props.image" />
334
- <ion-card-content *ngIf="props.content" class="complex-content">{{ props.content }}</ion-card-content>
374
+ <img alt="image" [src]="resolvedProps.image" />
375
+ <ion-card-content *ngIf="resolvedProps.content" class="complex-content">{{ resolvedProps.content }}</ion-card-content>
335
376
  </div>
336
377
  <val-button
337
- *ngFor="let b of props.footerActions"
378
+ *ngFor="let b of resolvedProps.footerActions"
338
379
  [props]="b"
339
380
  (onClick)="clickHandler(sections.footer, b.token)"
340
381
  ></val-button>
341
382
  <ion-buttons
342
383
  style="display: flex; align-items: center; justify-content: flex-end; margin: 8px"
343
- *ngIf="props.footerComplexActions.length > 0"
384
+ *ngIf="resolvedProps.footerComplexActions?.length > 0"
344
385
  >
345
- <ng-container *ngFor="let action of props.footerComplexActions">
386
+ <ng-container *ngFor="let action of resolvedProps.footerComplexActions">
346
387
  <ion-button
347
388
  *ngIf="action.type === actionTypes.ICON"
348
389
  (click)="clickHandler(sections.footerExtra, action.token)"
@@ -371,9 +412,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
371
412
  </ion-buttons>
372
413
  </ion-card>
373
414
  `, 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-card.tapable{transition:transform .3s ease,box-shadow .3s ease}ion-card.tapable:hover{transform:scale(1.01);box-shadow:.1875rem .625rem .5rem #1219541a}.tapable{cursor:pointer}.checker{display:flex;flex-direction:row;justify-content:space-between;align-items:center}.complex-header{padding:10px;display:flex;flex-direction:row;justify-content:space-between;align-items:center}.complex-content{padding-left:10px;padding-top:4px;padding-bottom:10px}.complex{border-radius:16px}img{border-radius:24px}\n"] }]
374
- }], ctorParameters: () => [], propDecorators: { props: [{
415
+ }], ctorParameters: () => [], propDecorators: { preset: [{
416
+ type: Input
417
+ }], props: [{
375
418
  type: Input
376
419
  }], onClick: [{
377
420
  type: Output
378
421
  }] } });
379
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/card/card.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;AAC/E,OAAO,EACL,SAAS,EACT,UAAU,EACV,OAAO,EACP,cAAc,EACd,aAAa,EACb,eAAe,EACf,YAAY,EACZ,WAAW,EACX,OAAO,GACR,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAC;AACtE,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAC;AACtE,OAAO,EAAE,cAAc,EAAE,MAAM,mCAAmC,CAAC;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC;AAChE,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAChD,OAAO,EAAgC,WAAW,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;;;AAsL9E;;;;;;;;;;GAUG;AACH,MAAM,OAAO,aAAa;IAyBxB;QAVA;;WAEG;QAEH,YAAO,GAAG,IAAI,YAAY,EAAkB,CAAC;QAE7C,UAAK,GAAG,QAAQ,CAAC;QACjB,gBAAW,GAAG,iBAAiB,CAAC;QAChC,aAAQ,GAAG,WAAW,CAAC;IAER,CAAC;IAEhB,QAAQ,KAAI,CAAC;IAEb,YAAY,CAAC,OAAoB,EAAE,KAAc;QAC/C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;IACxC,CAAC;+GA/BU,aAAa;mGAAb,aAAa,iIA5Kd;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8JT,qkGA7KC,YAAY,gQACZ,eAAe,0GACf,eAAe,gGACf,cAAc,yEACd,aAAa,wEACb,OAAO,yLACP,cAAc,+EACd,aAAa,sGACb,YAAY,sFACZ,eAAe,yFACf,WAAW,qMACX,UAAU,8EACV,SAAS,oPACT,OAAO;;4FA8KE,aAAa;kBA/LzB,SAAS;+BACE,UAAU,cACR,IAAI,WACP;wBACP,YAAY;wBACZ,eAAe;wBACf,eAAe;wBACf,cAAc;wBACd,aAAa;wBACb,OAAO;wBACP,cAAc;wBACd,aAAa;wBACb,YAAY;wBACZ,eAAe;wBACf,WAAW;wBACX,UAAU;wBACV,SAAS;wBACT,OAAO;qBACR,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8JT;wDA2BD,KAAK;sBADJ,KAAK;gBAON,OAAO;sBADN,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport {\n  IonButton,\n  IonButtons,\n  IonCard,\n  IonCardContent,\n  IonCardHeader,\n  IonCardSubtitle,\n  IonCardTitle,\n  IonCheckbox,\n  IonIcon,\n} from '@ionic/angular/standalone';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { ButtonComponent } from '../../atoms/button/button.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport { TextComponent } from '../../atoms/text/text.component';\nimport { ToolbarActionType } from '../../types';\nimport { CardClickEvent, CardMetadata, CardSection, CardType } from './types';\n\n@Component({\n  selector: 'val-card',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ButtonComponent,\n    AvatarComponent,\n    ImageComponent,\n    TextComponent,\n    IonCard,\n    IonCardContent,\n    IonCardHeader,\n    IonCardTitle,\n    IonCardSubtitle,\n    IonCheckbox,\n    IonButtons,\n    IonButton,\n    IonIcon,\n  ],\n  template: `\n    <ion-card *ngIf=\"props.type === types.native\">\n      <img alt=\"image\" [src]=\"props.image\" />\n      <ion-card-header *ngIf=\"props.title || props.overtitle\">\n        <ion-card-title *ngIf=\"props.title\">{{ props.title }}</ion-card-title>\n        <ion-card-subtitle *ngIf=\"props.overtitle\">{{ props.overtitle }}</ion-card-subtitle>\n      </ion-card-header>\n\n      <ion-card-content *ngIf=\"props.content\">{{ props.content }}</ion-card-content>\n\n      <val-button\n        *ngFor=\"let b of props.footerActions\"\n        [props]=\"b\"\n        (onClick)=\"clickHandler(sections.footer, b.token)\"\n      ></val-button>\n    </ion-card>\n\n    <ion-card\n      *ngIf=\"props.type === types.tappable\"\n      (click)=\"clickHandler(sections.content, props.token)\"\n      class=\"tapable\"\n    >\n      <img alt=\"image\" [src]=\"props.image\" />\n      <ion-card-header *ngIf=\"props.title || props.overtitle\">\n        <ion-card-title *ngIf=\"props.title\">{{ props.title }}</ion-card-title>\n        <ion-card-subtitle *ngIf=\"props.overtitle\">{{ props.overtitle }}</ion-card-subtitle>\n      </ion-card-header>\n\n      <ion-card-content *ngIf=\"props.content\">{{ props.content }}</ion-card-content>\n    </ion-card>\n\n    <ion-card\n      *ngIf=\"props.type === types.checker\"\n      (click)=\"clickHandler(sections.content, props.token)\"\n      class=\"tapable\"\n    >\n      <ion-card-header *ngIf=\"props.title || props.overtitle\" class=\"checker\">\n        <div>\n          <ion-card-subtitle *ngIf=\"props.overtitle\">{{ props.overtitle }}</ion-card-subtitle>\n          <ion-card-title *ngIf=\"props.title\">{{ props.title }}</ion-card-title>\n        </div>\n        <div>\n          <ion-checkbox [checked]=\"props.selected\"></ion-checkbox>\n        </div>\n      </ion-card-header>\n\n      <ion-card-content *ngIf=\"props.content\">{{ props.content }}</ion-card-content>\n    </ion-card>\n\n    <ion-card *ngIf=\"props.type === types.complex\" class=\"complex\">\n      <ion-card-header class=\"complex-header\">\n        <ion-buttons style=\"display: flex; align-items: center\" *ngIf=\"props.leftActions.length > 0\">\n          <ng-container *ngFor=\"let action of props.leftActions\">\n            <ion-button\n              *ngIf=\"action.type === actionTypes.ICON\"\n              (click)=\"clickHandler(sections.headerLeft, action.token)\"\n            >\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" color=\"dark\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              style=\"margin-right: 4px; cursor: pointer\"\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(sections.headerLeft, action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(sections.headerLeft, action.token)\"\n            ></val-image>\n            <ion-button\n              *ngIf=\"action.type === actionTypes.BUTTON\"\n              (click)=\"clickHandler(sections.headerLeft, action.token)\"\n            >\n              {{ action.description }}\n            </ion-button>\n            <div *ngIf=\"props.headerText\">\n              <val-text [props]=\"{ content: props.headerText, color: 'dark', bold: true, size: 'medium' }\" />\n            </div>\n          </ng-container>\n        </ion-buttons>\n        <ion-buttons style=\"display: flex; align-items: center\" *ngIf=\"props.rightActions.length > 0\">\n          <ng-container *ngFor=\"let action of props.rightActions\">\n            <ion-button\n              *ngIf=\"action.type === actionTypes.ICON\"\n              (click)=\"clickHandler(sections.headerRight, action.token)\"\n            >\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" color=\"dark\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              style=\"margin-right: 4px; cursor: pointer\"\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(sections.headerRight, action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(sections.headerRight, action.token)\"\n            ></val-image>\n            <ion-button\n              *ngIf=\"action.type === actionTypes.BUTTON\"\n              (click)=\"clickHandler(sections.headerRight, action.token)\"\n            >\n              {{ action.description }}\n            </ion-button>\n          </ng-container>\n        </ion-buttons>\n      </ion-card-header>\n\n      <div class=\"tapable\" (click)=\"clickHandler(sections.content, props.token)\">\n        <ion-card-header *ngIf=\"props.title || props.overtitle\" class=\"complex-header\">\n          <div>\n            <ion-card-subtitle *ngIf=\"props.overtitle\">{{ props.overtitle }}</ion-card-subtitle>\n            <ion-card-title *ngIf=\"props.title\">{{ props.title }}</ion-card-title>\n          </div>\n        </ion-card-header>\n\n        <img alt=\"image\" [src]=\"props.image\" />\n        <ion-card-content *ngIf=\"props.content\" class=\"complex-content\">{{ props.content }}</ion-card-content>\n      </div>\n      <val-button\n        *ngFor=\"let b of props.footerActions\"\n        [props]=\"b\"\n        (onClick)=\"clickHandler(sections.footer, b.token)\"\n      ></val-button>\n      <ion-buttons\n        style=\"display: flex; align-items: center; justify-content: flex-end; margin: 8px\"\n        *ngIf=\"props.footerComplexActions.length > 0\"\n      >\n        <ng-container *ngFor=\"let action of props.footerComplexActions\">\n          <ion-button\n            *ngIf=\"action.type === actionTypes.ICON\"\n            (click)=\"clickHandler(sections.footerExtra, action.token)\"\n          >\n            <ion-icon slot=\"icon-only\" [name]=\"action.description\" color=\"dark\"></ion-icon>\n          </ion-button>\n          <val-avatar\n            style=\"margin-right: 4px; cursor: pointer\"\n            *ngIf=\"action.type === actionTypes.AVATAR\"\n            [props]=\"{ size: 'small', image: action.description, default: '' }\"\n            (onClick)=\"clickHandler(sections.footerExtra, action.token)\"\n          ></val-avatar>\n          <val-image\n            *ngIf=\"action.type === actionTypes.IMAGE\"\n            [props]=\"action.image\"\n            (click)=\"clickHandler(sections.footerExtra, action.token)\"\n          ></val-image>\n          <ion-button\n            *ngIf=\"action.type === actionTypes.BUTTON\"\n            (click)=\"clickHandler(sections.footerExtra, action.token)\"\n            color=\"dark\"\n          >\n            {{ action.description }}\n          </ion-button>\n        </ng-container>\n      </ion-buttons>\n    </ion-card>\n  `,\n  styleUrls: ['./card.component.scss'],\n})\n/**\n * val-card\n *\n * A flexible card component supporting images, titles, content, actions, and custom sections.\n *\n * @example\n * <val-card [props]=\"{ type: 'native', title: 'Card', image: 'url', content: '...', actions: [...] }\" (onClick)=\"handler($event)\"></val-card>\n *\n * @input props: CardMetadata - Configuration for the card (type, title, image, content, actions, etc.)\n * @output onClick - Emits a CardClickEvent when the card or an action is clicked\n */\nexport class CardComponent implements OnInit {\n  /**\n   * Card configuration object.\n   * @type {CardMetadata}\n   * @property type - The card type (see CardType).\n   * @property title - The card title.\n   * @property image - The card image URL.\n   * @property content - The card content.\n   * @property actions - Array of action buttons (optional).\n   * @property sections - Custom card sections (optional).\n   * @property overtitle - Subtitle or overtitle (optional).\n   */\n  @Input()\n  props: CardMetadata;\n\n  /**\n   * Event emitted when the card or an action is clicked.\n   */\n  @Output()\n  onClick = new EventEmitter<CardClickEvent>();\n\n  types = CardType;\n  actionTypes = ToolbarActionType;\n  sections = CardSection;\n\n  constructor() {}\n\n  ngOnInit() {}\n\n  clickHandler(section: CardSection, token?: string) {\n    this.onClick.emit({ section, token });\n  }\n}\n"]}
422
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/card/card.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAqB,MAAM,EAAiB,MAAM,eAAe,CAAC;AACjH,OAAO,EACL,SAAS,EACT,UAAU,EACV,OAAO,EACP,cAAc,EACd,aAAa,EACb,eAAe,EACf,YAAY,EACZ,WAAW,EACX,OAAO,GACR,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAC;AACtE,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAC;AACtE,OAAO,EAAE,cAAc,EAAE,MAAM,mCAAmC,CAAC;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC;AAChE,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAChD,OAAO,EAAgC,WAAW,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;;;AAsL9E;;;;;;;;;;;;;;;GAeG;AACH,MAAM,OAAO,aAAa;IAwCxB;QAvCQ,YAAO,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QAWxC;;;;;;;;;;WAUG;QACM,UAAK,GAA0B,EAAE,CAAC;QAE3C;;WAEG;QACH,kBAAa,GAAiB,EAAkB,CAAC;QAEjD;;WAEG;QAEH,YAAO,GAAG,IAAI,YAAY,EAAkB,CAAC;QAE7C,UAAK,GAAG,QAAQ,CAAC;QACjB,gBAAW,GAAG,iBAAiB,CAAC;QAChC,aAAQ,GAAG,WAAW,CAAC;IAER,CAAC;IAEhB,QAAQ;QACN,IAAI,CAAC,YAAY,EAAE,CAAC;IACtB,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YAC1C,IAAI,CAAC,YAAY,EAAE,CAAC;QACtB,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,YAAY;QAClB,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,MAAM,CAA2B,CAAC,CAAC,CAAC,EAAE,CAAC;QAExG,IAAI,CAAC,aAAa,GAAG;YACnB,GAAG,WAAW;YACd,GAAG,IAAI,CAAC,KAAK;SACE,CAAC;IACpB,CAAC;IAED,YAAY,CAAC,OAAoB,EAAE,KAAc;QAC/C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;IACxC,CAAC;+GAnEU,aAAa;mGAAb,aAAa,wKAjLd;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8JT,qkGA7KC,YAAY,gQACZ,eAAe,0GACf,eAAe,gGACf,cAAc,yEACd,aAAa,wEACb,OAAO,yLACP,cAAc,+EACd,aAAa,sGACb,YAAY,sFACZ,eAAe,yFACf,WAAW,qMACX,UAAU,8EACV,SAAS,oPACT,OAAO;;4FAmLE,aAAa;kBApMzB,SAAS;+BACE,UAAU,cACR,IAAI,WACP;wBACP,YAAY;wBACZ,eAAe;wBACf,eAAe;wBACf,cAAc;wBACd,aAAa;wBACb,OAAO;wBACP,cAAc;wBACd,aAAa;wBACb,YAAY;wBACZ,eAAe;wBACf,WAAW;wBACX,UAAU;wBACV,SAAS;wBACT,OAAO;qBACR,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8JT;wDA6BQ,MAAM;sBAAd,KAAK;gBAaG,KAAK;sBAAb,KAAK;gBAWN,OAAO;sBADN,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, OnChanges, OnInit, Output, SimpleChanges } from '@angular/core';\nimport {\n  IonButton,\n  IonButtons,\n  IonCard,\n  IonCardContent,\n  IonCardHeader,\n  IonCardSubtitle,\n  IonCardTitle,\n  IonCheckbox,\n  IonIcon,\n} from '@ionic/angular/standalone';\nimport { PresetService } from '../../../services/presets';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { ButtonComponent } from '../../atoms/button/button.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport { TextComponent } from '../../atoms/text/text.component';\nimport { ToolbarActionType } from '../../types';\nimport { CardClickEvent, CardMetadata, CardSection, CardType } from './types';\n\n@Component({\n  selector: 'val-card',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ButtonComponent,\n    AvatarComponent,\n    ImageComponent,\n    TextComponent,\n    IonCard,\n    IonCardContent,\n    IonCardHeader,\n    IonCardTitle,\n    IonCardSubtitle,\n    IonCheckbox,\n    IonButtons,\n    IonButton,\n    IonIcon,\n  ],\n  template: `\n    <ion-card *ngIf=\"resolvedProps.type === types.native\">\n      <img alt=\"image\" [src]=\"resolvedProps.image\" />\n      <ion-card-header *ngIf=\"resolvedProps.title || resolvedProps.overtitle\">\n        <ion-card-title *ngIf=\"resolvedProps.title\">{{ resolvedProps.title }}</ion-card-title>\n        <ion-card-subtitle *ngIf=\"resolvedProps.overtitle\">{{ resolvedProps.overtitle }}</ion-card-subtitle>\n      </ion-card-header>\n\n      <ion-card-content *ngIf=\"resolvedProps.content\">{{ resolvedProps.content }}</ion-card-content>\n\n      <val-button\n        *ngFor=\"let b of resolvedProps.footerActions\"\n        [props]=\"b\"\n        (onClick)=\"clickHandler(sections.footer, b.token)\"\n      ></val-button>\n    </ion-card>\n\n    <ion-card\n      *ngIf=\"resolvedProps.type === types.tappable\"\n      (click)=\"clickHandler(sections.content, resolvedProps.token)\"\n      class=\"tapable\"\n    >\n      <img alt=\"image\" [src]=\"resolvedProps.image\" />\n      <ion-card-header *ngIf=\"resolvedProps.title || resolvedProps.overtitle\">\n        <ion-card-title *ngIf=\"resolvedProps.title\">{{ resolvedProps.title }}</ion-card-title>\n        <ion-card-subtitle *ngIf=\"resolvedProps.overtitle\">{{ resolvedProps.overtitle }}</ion-card-subtitle>\n      </ion-card-header>\n\n      <ion-card-content *ngIf=\"resolvedProps.content\">{{ resolvedProps.content }}</ion-card-content>\n    </ion-card>\n\n    <ion-card\n      *ngIf=\"resolvedProps.type === types.checker\"\n      (click)=\"clickHandler(sections.content, resolvedProps.token)\"\n      class=\"tapable\"\n    >\n      <ion-card-header *ngIf=\"resolvedProps.title || resolvedProps.overtitle\" class=\"checker\">\n        <div>\n          <ion-card-subtitle *ngIf=\"resolvedProps.overtitle\">{{ resolvedProps.overtitle }}</ion-card-subtitle>\n          <ion-card-title *ngIf=\"resolvedProps.title\">{{ resolvedProps.title }}</ion-card-title>\n        </div>\n        <div>\n          <ion-checkbox [checked]=\"resolvedProps.selected\"></ion-checkbox>\n        </div>\n      </ion-card-header>\n\n      <ion-card-content *ngIf=\"resolvedProps.content\">{{ resolvedProps.content }}</ion-card-content>\n    </ion-card>\n\n    <ion-card *ngIf=\"resolvedProps.type === types.complex\" class=\"complex\">\n      <ion-card-header class=\"complex-header\">\n        <ion-buttons style=\"display: flex; align-items: center\" *ngIf=\"resolvedProps.leftActions?.length > 0\">\n          <ng-container *ngFor=\"let action of resolvedProps.leftActions\">\n            <ion-button\n              *ngIf=\"action.type === actionTypes.ICON\"\n              (click)=\"clickHandler(sections.headerLeft, action.token)\"\n            >\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" color=\"dark\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              style=\"margin-right: 4px; cursor: pointer\"\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(sections.headerLeft, action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(sections.headerLeft, action.token)\"\n            ></val-image>\n            <ion-button\n              *ngIf=\"action.type === actionTypes.BUTTON\"\n              (click)=\"clickHandler(sections.headerLeft, action.token)\"\n            >\n              {{ action.description }}\n            </ion-button>\n            <div *ngIf=\"resolvedProps.headerText\">\n              <val-text [props]=\"{ content: resolvedProps.headerText, color: 'dark', bold: true, size: 'medium' }\" />\n            </div>\n          </ng-container>\n        </ion-buttons>\n        <ion-buttons style=\"display: flex; align-items: center\" *ngIf=\"resolvedProps.rightActions?.length > 0\">\n          <ng-container *ngFor=\"let action of resolvedProps.rightActions\">\n            <ion-button\n              *ngIf=\"action.type === actionTypes.ICON\"\n              (click)=\"clickHandler(sections.headerRight, action.token)\"\n            >\n              <ion-icon slot=\"icon-only\" [name]=\"action.description\" color=\"dark\"></ion-icon>\n            </ion-button>\n            <val-avatar\n              style=\"margin-right: 4px; cursor: pointer\"\n              *ngIf=\"action.type === actionTypes.AVATAR\"\n              [props]=\"{ size: 'small', image: action.description, default: '' }\"\n              (onClick)=\"clickHandler(sections.headerRight, action.token)\"\n            ></val-avatar>\n            <val-image\n              *ngIf=\"action.type === actionTypes.IMAGE\"\n              [props]=\"action.image\"\n              (click)=\"clickHandler(sections.headerRight, action.token)\"\n            ></val-image>\n            <ion-button\n              *ngIf=\"action.type === actionTypes.BUTTON\"\n              (click)=\"clickHandler(sections.headerRight, action.token)\"\n            >\n              {{ action.description }}\n            </ion-button>\n          </ng-container>\n        </ion-buttons>\n      </ion-card-header>\n\n      <div class=\"tapable\" (click)=\"clickHandler(sections.content, resolvedProps.token)\">\n        <ion-card-header *ngIf=\"resolvedProps.title || resolvedProps.overtitle\" class=\"complex-header\">\n          <div>\n            <ion-card-subtitle *ngIf=\"resolvedProps.overtitle\">{{ resolvedProps.overtitle }}</ion-card-subtitle>\n            <ion-card-title *ngIf=\"resolvedProps.title\">{{ resolvedProps.title }}</ion-card-title>\n          </div>\n        </ion-card-header>\n\n        <img alt=\"image\" [src]=\"resolvedProps.image\" />\n        <ion-card-content *ngIf=\"resolvedProps.content\" class=\"complex-content\">{{ resolvedProps.content }}</ion-card-content>\n      </div>\n      <val-button\n        *ngFor=\"let b of resolvedProps.footerActions\"\n        [props]=\"b\"\n        (onClick)=\"clickHandler(sections.footer, b.token)\"\n      ></val-button>\n      <ion-buttons\n        style=\"display: flex; align-items: center; justify-content: flex-end; margin: 8px\"\n        *ngIf=\"resolvedProps.footerComplexActions?.length > 0\"\n      >\n        <ng-container *ngFor=\"let action of resolvedProps.footerComplexActions\">\n          <ion-button\n            *ngIf=\"action.type === actionTypes.ICON\"\n            (click)=\"clickHandler(sections.footerExtra, action.token)\"\n          >\n            <ion-icon slot=\"icon-only\" [name]=\"action.description\" color=\"dark\"></ion-icon>\n          </ion-button>\n          <val-avatar\n            style=\"margin-right: 4px; cursor: pointer\"\n            *ngIf=\"action.type === actionTypes.AVATAR\"\n            [props]=\"{ size: 'small', image: action.description, default: '' }\"\n            (onClick)=\"clickHandler(sections.footerExtra, action.token)\"\n          ></val-avatar>\n          <val-image\n            *ngIf=\"action.type === actionTypes.IMAGE\"\n            [props]=\"action.image\"\n            (click)=\"clickHandler(sections.footerExtra, action.token)\"\n          ></val-image>\n          <ion-button\n            *ngIf=\"action.type === actionTypes.BUTTON\"\n            (click)=\"clickHandler(sections.footerExtra, action.token)\"\n            color=\"dark\"\n          >\n            {{ action.description }}\n          </ion-button>\n        </ng-container>\n      </ion-buttons>\n    </ion-card>\n  `,\n  styleUrls: ['./card.component.scss'],\n})\n/**\n * val-card\n *\n * A flexible card component supporting images, titles, content, actions, and custom sections.\n * Supports presets for reusable configurations.\n *\n * @example With preset (recommended):\n * <val-card preset=\"feature\" [props]=\"{ title: 'Card', image: 'url', content: '...' }\" (onClick)=\"handler($event)\"></val-card>\n *\n * @example Static (backwards compatible):\n * <val-card [props]=\"{ type: 'native', title: 'Card', image: 'url', content: '...' }\" (onClick)=\"handler($event)\"></val-card>\n *\n * @input preset: string - Name of preset to apply (e.g., 'feature', 'compact')\n * @input props: CardMetadata - Configuration for the card (overrides preset values)\n * @output onClick - Emits a CardClickEvent when the card or an action is clicked\n */\nexport class CardComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  /**\n   * Preset name to apply. Presets define reusable card configurations\n   * (type, variant, padding, etc.) that can be registered at app level.\n   *\n   * @example\n   * <val-card preset=\"feature\" [props]=\"{ title: 'My Card' }\"></val-card>\n   */\n  @Input() preset?: string;\n\n  /**\n   * Card configuration object. Values here override preset values.\n   * @type {CardMetadata}\n   * @property type - The card type (see CardType).\n   * @property title - The card title.\n   * @property image - The card image URL.\n   * @property content - The card content.\n   * @property actions - Array of action buttons (optional).\n   * @property sections - Custom card sections (optional).\n   * @property overtitle - Subtitle or overtitle (optional).\n   */\n  @Input() props: Partial<CardMetadata> = {};\n\n  /**\n   * Resolved props after merging preset + explicit props.\n   */\n  resolvedProps: CardMetadata = {} as CardMetadata;\n\n  /**\n   * Event emitted when the card or an action is clicked.\n   */\n  @Output()\n  onClick = new EventEmitter<CardClickEvent>();\n\n  types = CardType;\n  actionTypes = ToolbarActionType;\n  sections = CardSection;\n\n  constructor() {}\n\n  ngOnInit() {\n    this.resolveProps();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['preset'] || changes['props']) {\n      this.resolveProps();\n    }\n  }\n\n  /**\n   * Merge preset configuration with explicit props.\n   * Explicit props take precedence over preset values.\n   */\n  private resolveProps(): void {\n    const presetProps = this.preset ? (this.presets.get('card', this.preset) as Partial<CardMetadata>) : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as CardMetadata;\n  }\n\n  clickHandler(section: CardSection, token?: string) {\n    this.onClick.emit({ section, token });\n  }\n}\n"]}