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.
- package/esm2022/lib/components/molecules/alert-box/alert-box.component.mjs +55 -13
- package/esm2022/lib/components/molecules/card/card.component.mjs +125 -82
- package/esm2022/lib/components/molecules/docs-code-example/docs-code-example.component.mjs +3 -3
- package/esm2022/lib/components/molecules/email-input/email-input.component.mjs +48 -18
- package/esm2022/lib/components/molecules/password-input/password-input.component.mjs +49 -21
- package/esm2022/lib/components/molecules/select-input/select-input.component.mjs +68 -18
- package/esm2022/lib/components/molecules/text-input/text-input.component.mjs +49 -19
- package/esm2022/lib/components/organisms/form/form.component.mjs +2 -2
- package/fesm2022/valtech-components.mjs +380 -163
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/alert-box/alert-box.component.d.ts +24 -5
- package/lib/components/molecules/card/card.component.d.ts +24 -5
- package/lib/components/molecules/email-input/email-input.component.d.ts +25 -6
- package/lib/components/molecules/password-input/password-input.component.d.ts +25 -6
- package/lib/components/molecules/select-input/select-input.component.d.ts +26 -6
- package/lib/components/molecules/text-input/text-input.component.d.ts +26 -26
- package/package.json +1 -1
|
@@ -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
|
|
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.
|
|
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.
|
|
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.
|
|
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]="
|
|
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]="
|
|
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]="
|
|
102
|
+
<val-box [props]="resolvedProps.box">
|
|
63
103
|
<div class="content-container" body>
|
|
64
|
-
<val-icon [props]="
|
|
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: {
|
|
114
|
+
}], propDecorators: { preset: [{
|
|
115
|
+
type: Input
|
|
116
|
+
}], props: [{
|
|
75
117
|
type: Input
|
|
76
118
|
}] } });
|
|
77
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
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]="{
|
|
19
|
+
* @example With preset (recommended):
|
|
20
|
+
* <val-card preset="feature" [props]="{ title: 'Card', image: 'url', content: '...' }" (onClick)="handler($event)"></val-card>
|
|
19
21
|
*
|
|
20
|
-
* @
|
|
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="
|
|
40
|
-
<img alt="image" [src]="
|
|
41
|
-
<ion-card-header *ngIf="
|
|
42
|
-
<ion-card-title *ngIf="
|
|
43
|
-
<ion-card-subtitle *ngIf="
|
|
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="
|
|
87
|
+
<ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
|
|
47
88
|
|
|
48
89
|
<val-button
|
|
49
|
-
*ngFor="let b of
|
|
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="
|
|
57
|
-
(click)="clickHandler(sections.content,
|
|
97
|
+
*ngIf="resolvedProps.type === types.tappable"
|
|
98
|
+
(click)="clickHandler(sections.content, resolvedProps.token)"
|
|
58
99
|
class="tapable"
|
|
59
100
|
>
|
|
60
|
-
<img alt="image" [src]="
|
|
61
|
-
<ion-card-header *ngIf="
|
|
62
|
-
<ion-card-title *ngIf="
|
|
63
|
-
<ion-card-subtitle *ngIf="
|
|
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="
|
|
107
|
+
<ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
|
|
67
108
|
</ion-card>
|
|
68
109
|
|
|
69
110
|
<ion-card
|
|
70
|
-
*ngIf="
|
|
71
|
-
(click)="clickHandler(sections.content,
|
|
111
|
+
*ngIf="resolvedProps.type === types.checker"
|
|
112
|
+
(click)="clickHandler(sections.content, resolvedProps.token)"
|
|
72
113
|
class="tapable"
|
|
73
114
|
>
|
|
74
|
-
<ion-card-header *ngIf="
|
|
115
|
+
<ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle" class="checker">
|
|
75
116
|
<div>
|
|
76
|
-
<ion-card-subtitle *ngIf="
|
|
77
|
-
<ion-card-title *ngIf="
|
|
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]="
|
|
121
|
+
<ion-checkbox [checked]="resolvedProps.selected"></ion-checkbox>
|
|
81
122
|
</div>
|
|
82
123
|
</ion-card-header>
|
|
83
124
|
|
|
84
|
-
<ion-card-content *ngIf="
|
|
125
|
+
<ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
|
|
85
126
|
</ion-card>
|
|
86
127
|
|
|
87
|
-
<ion-card *ngIf="
|
|
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="
|
|
90
|
-
<ng-container *ngFor="let action of
|
|
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="
|
|
115
|
-
<val-text [props]="{ content:
|
|
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="
|
|
120
|
-
<ng-container *ngFor="let action of
|
|
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,
|
|
149
|
-
<ion-card-header *ngIf="
|
|
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="
|
|
152
|
-
<ion-card-title *ngIf="
|
|
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]="
|
|
157
|
-
<ion-card-content *ngIf="
|
|
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
|
|
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="
|
|
207
|
+
*ngIf="resolvedProps.footerComplexActions?.length > 0"
|
|
167
208
|
>
|
|
168
|
-
<ng-container *ngFor="let action of
|
|
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="
|
|
217
|
-
<img alt="image" [src]="
|
|
218
|
-
<ion-card-header *ngIf="
|
|
219
|
-
<ion-card-title *ngIf="
|
|
220
|
-
<ion-card-subtitle *ngIf="
|
|
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="
|
|
264
|
+
<ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
|
|
224
265
|
|
|
225
266
|
<val-button
|
|
226
|
-
*ngFor="let b of
|
|
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="
|
|
234
|
-
(click)="clickHandler(sections.content,
|
|
274
|
+
*ngIf="resolvedProps.type === types.tappable"
|
|
275
|
+
(click)="clickHandler(sections.content, resolvedProps.token)"
|
|
235
276
|
class="tapable"
|
|
236
277
|
>
|
|
237
|
-
<img alt="image" [src]="
|
|
238
|
-
<ion-card-header *ngIf="
|
|
239
|
-
<ion-card-title *ngIf="
|
|
240
|
-
<ion-card-subtitle *ngIf="
|
|
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="
|
|
284
|
+
<ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
|
|
244
285
|
</ion-card>
|
|
245
286
|
|
|
246
287
|
<ion-card
|
|
247
|
-
*ngIf="
|
|
248
|
-
(click)="clickHandler(sections.content,
|
|
288
|
+
*ngIf="resolvedProps.type === types.checker"
|
|
289
|
+
(click)="clickHandler(sections.content, resolvedProps.token)"
|
|
249
290
|
class="tapable"
|
|
250
291
|
>
|
|
251
|
-
<ion-card-header *ngIf="
|
|
292
|
+
<ion-card-header *ngIf="resolvedProps.title || resolvedProps.overtitle" class="checker">
|
|
252
293
|
<div>
|
|
253
|
-
<ion-card-subtitle *ngIf="
|
|
254
|
-
<ion-card-title *ngIf="
|
|
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]="
|
|
298
|
+
<ion-checkbox [checked]="resolvedProps.selected"></ion-checkbox>
|
|
258
299
|
</div>
|
|
259
300
|
</ion-card-header>
|
|
260
301
|
|
|
261
|
-
<ion-card-content *ngIf="
|
|
302
|
+
<ion-card-content *ngIf="resolvedProps.content">{{ resolvedProps.content }}</ion-card-content>
|
|
262
303
|
</ion-card>
|
|
263
304
|
|
|
264
|
-
<ion-card *ngIf="
|
|
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="
|
|
267
|
-
<ng-container *ngFor="let action of
|
|
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="
|
|
292
|
-
<val-text [props]="{ content:
|
|
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="
|
|
297
|
-
<ng-container *ngFor="let action of
|
|
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,
|
|
326
|
-
<ion-card-header *ngIf="
|
|
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="
|
|
329
|
-
<ion-card-title *ngIf="
|
|
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]="
|
|
334
|
-
<ion-card-content *ngIf="
|
|
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
|
|
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="
|
|
384
|
+
*ngIf="resolvedProps.footerComplexActions?.length > 0"
|
|
344
385
|
>
|
|
345
|
-
<ng-container *ngFor="let action of
|
|
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: {
|
|
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"]}
|