valtech-components 2.0.566 → 2.0.567
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/atoms/avatar/avatar.component.mjs +51 -14
- package/esm2022/lib/components/atoms/fab/fab.component.mjs +63 -25
- package/esm2022/lib/components/atoms/progress-bar/progress-bar.component.mjs +64 -19
- package/esm2022/lib/components/molecules/accordion/accordion.component.mjs +45 -18
- package/esm2022/lib/components/molecules/breadcrumb/breadcrumb.component.mjs +43 -15
- package/esm2022/lib/components/molecules/card/card.component.mjs +2 -2
- package/esm2022/lib/components/molecules/check-input/check-input.component.mjs +52 -8
- package/esm2022/lib/components/molecules/chip-group/chip-group.component.mjs +54 -37
- package/esm2022/lib/components/molecules/comment/comment.component.mjs +1 -1
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +62 -27
- package/esm2022/lib/components/molecules/glow-card/glow-card.component.mjs +74 -46
- package/esm2022/lib/components/molecules/multi-select-search/multi-select-search.component.mjs +2 -2
- package/esm2022/lib/components/molecules/participant-card/participant-card.component.mjs +1 -1
- package/esm2022/lib/components/molecules/phone-input/phone-input.component.mjs +95 -59
- package/esm2022/lib/components/molecules/pill/pill.component.mjs +61 -13
- package/esm2022/lib/components/molecules/progress-status/progress-status.component.mjs +2 -2
- package/esm2022/lib/components/molecules/rating/rating.component.mjs +60 -43
- package/esm2022/lib/components/molecules/searchbar/searchbar.component.mjs +56 -26
- package/esm2022/lib/components/molecules/searchbar/types.mjs +2 -0
- package/esm2022/lib/components/molecules/segment-control/segment-control.component.mjs +52 -29
- package/esm2022/lib/components/molecules/select-search/select-search.component.mjs +2 -2
- package/esm2022/lib/components/molecules/stats-card/stats-card.component.mjs +67 -51
- package/esm2022/lib/components/molecules/stepper/stepper.component.mjs +51 -23
- package/esm2022/lib/components/molecules/tabs/tabs.component.mjs +59 -21
- package/esm2022/lib/components/molecules/textarea-input/textarea-input.component.mjs +93 -59
- package/esm2022/lib/components/molecules/toggle-input/toggle-input.component.mjs +49 -15
- package/esm2022/lib/components/molecules/winner-display/winner-display.component.mjs +1 -1
- package/esm2022/lib/components/organisms/comment-section/comment-section.component.mjs +2 -2
- package/esm2022/lib/components/organisms/form/form.component.mjs +2 -2
- package/esm2022/lib/components/organisms/tabbed-content/tabbed-content.component.mjs +1 -1
- package/esm2022/lib/components/organisms/toolbar/toolbar.component.mjs +2 -2
- package/esm2022/lib/components/organisms/wizard/wizard-footer/wizard-footer.component.mjs +2 -2
- package/fesm2022/valtech-components.mjs +1259 -675
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/avatar/avatar.component.d.ts +26 -5
- package/lib/components/atoms/fab/fab.component.d.ts +24 -5
- package/lib/components/atoms/progress-bar/progress-bar.component.d.ts +21 -6
- package/lib/components/molecules/accordion/accordion.component.d.ts +10 -4
- package/lib/components/molecules/breadcrumb/breadcrumb.component.d.ts +10 -4
- package/lib/components/molecules/check-input/check-input.component.d.ts +25 -4
- package/lib/components/molecules/chip-group/chip-group.component.d.ts +10 -4
- package/lib/components/molecules/date-input/date-input.component.d.ts +25 -7
- package/lib/components/molecules/glow-card/glow-card.component.d.ts +16 -5
- package/lib/components/molecules/phone-input/phone-input.component.d.ts +28 -5
- package/lib/components/molecules/pill/pill.component.d.ts +27 -3
- package/lib/components/molecules/rating/rating.component.d.ts +10 -4
- package/lib/components/molecules/searchbar/searchbar.component.d.ts +15 -11
- package/lib/components/molecules/searchbar/types.d.ts +33 -0
- package/lib/components/molecules/segment-control/segment-control.component.d.ts +17 -6
- package/lib/components/molecules/stats-card/stats-card.component.d.ts +10 -4
- package/lib/components/molecules/stepper/stepper.component.d.ts +20 -4
- package/lib/components/molecules/tabs/tabs.component.d.ts +27 -4
- package/lib/components/molecules/textarea-input/textarea-input.component.d.ts +27 -4
- package/lib/components/molecules/toggle-input/toggle-input.component.d.ts +25 -4
- package/package.json +1 -1
- package/src/lib/components/styles/_docs-page.scss +310 -0
|
@@ -1,6 +1,7 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
1
|
+
import { Component, inject, Input, Output, EventEmitter } from '@angular/core';
|
|
2
2
|
import { IonIcon } from '@ionic/angular/standalone';
|
|
3
3
|
import { CommonModule } from '@angular/common';
|
|
4
|
+
import { PresetService } from '../../../services/presets';
|
|
4
5
|
import { addIcons } from 'ionicons';
|
|
5
6
|
import { checkmark, close, ellipse } from 'ionicons/icons';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
@@ -37,14 +38,40 @@ addIcons({ checkmark, close, ellipse });
|
|
|
37
38
|
*/
|
|
38
39
|
export class StepperComponent {
|
|
39
40
|
constructor() {
|
|
41
|
+
this.presets = inject(PresetService);
|
|
42
|
+
/**
|
|
43
|
+
* Stepper configuration object. Values here override preset values.
|
|
44
|
+
*/
|
|
45
|
+
this.props = {};
|
|
46
|
+
/**
|
|
47
|
+
* Resolved props after merging preset + explicit props.
|
|
48
|
+
*/
|
|
49
|
+
this.resolvedProps = {};
|
|
40
50
|
this.stepChange = new EventEmitter();
|
|
41
51
|
}
|
|
52
|
+
ngOnInit() {
|
|
53
|
+
this.resolveProps();
|
|
54
|
+
}
|
|
55
|
+
ngOnChanges(changes) {
|
|
56
|
+
if (changes['preset'] || changes['props']) {
|
|
57
|
+
this.resolveProps();
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
resolveProps() {
|
|
61
|
+
const presetProps = this.preset
|
|
62
|
+
? this.presets.get('stepper', this.preset)
|
|
63
|
+
: {};
|
|
64
|
+
this.resolvedProps = {
|
|
65
|
+
...presetProps,
|
|
66
|
+
...this.props,
|
|
67
|
+
};
|
|
68
|
+
}
|
|
42
69
|
getCurrentIndex() {
|
|
43
|
-
if (this.
|
|
44
|
-
return this.
|
|
70
|
+
if (this.resolvedProps.currentIndex !== undefined) {
|
|
71
|
+
return this.resolvedProps.currentIndex;
|
|
45
72
|
}
|
|
46
|
-
if (this.
|
|
47
|
-
return this.
|
|
73
|
+
if (this.resolvedProps.currentStep) {
|
|
74
|
+
return this.resolvedProps.steps?.findIndex((s) => s.value === this.resolvedProps.currentStep) ?? 0;
|
|
48
75
|
}
|
|
49
76
|
return 0;
|
|
50
77
|
}
|
|
@@ -60,7 +87,7 @@ export class StepperComponent {
|
|
|
60
87
|
}
|
|
61
88
|
getStepColor(step, index) {
|
|
62
89
|
const status = this.getStepStatus(step, index);
|
|
63
|
-
const color = this.
|
|
90
|
+
const color = this.resolvedProps.color || 'primary';
|
|
64
91
|
switch (status) {
|
|
65
92
|
case 'completed':
|
|
66
93
|
return `var(--ion-color-${color})`;
|
|
@@ -75,10 +102,9 @@ export class StepperComponent {
|
|
|
75
102
|
isStepClickable(step, index) {
|
|
76
103
|
if (step.disabled)
|
|
77
104
|
return false;
|
|
78
|
-
if (!this.
|
|
105
|
+
if (!this.resolvedProps.linear)
|
|
79
106
|
return true;
|
|
80
107
|
const currentIndex = this.getCurrentIndex();
|
|
81
|
-
// In linear mode, can only go to completed steps or next step
|
|
82
108
|
return index <= currentIndex || (step.editable && index < currentIndex);
|
|
83
109
|
}
|
|
84
110
|
onStepClick(step, index) {
|
|
@@ -97,13 +123,13 @@ export class StepperComponent {
|
|
|
97
123
|
return step.label || step.contentFallback || '';
|
|
98
124
|
}
|
|
99
125
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StepperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
100
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: StepperComponent, isStandalone: true, selector: "val-stepper", inputs: { props: "props" }, outputs: { stepChange: "stepChange" }, ngImport: i0, template: `
|
|
126
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: StepperComponent, isStandalone: true, selector: "val-stepper", inputs: { preset: "preset", props: "props" }, outputs: { stepChange: "stepChange" }, usesOnChanges: true, ngImport: i0, template: `
|
|
101
127
|
<div
|
|
102
128
|
class="stepper-container"
|
|
103
|
-
[class.horizontal]="
|
|
104
|
-
[class.vertical]="
|
|
129
|
+
[class.horizontal]="resolvedProps.orientation !== 'vertical'"
|
|
130
|
+
[class.vertical]="resolvedProps.orientation === 'vertical'"
|
|
105
131
|
>
|
|
106
|
-
@for (step of
|
|
132
|
+
@for (step of resolvedProps.steps; track step.value; let i = $index; let isLast = $last) {
|
|
107
133
|
<div
|
|
108
134
|
class="step"
|
|
109
135
|
[class.current]="getStepStatus(step, i) === 'current'"
|
|
@@ -121,16 +147,16 @@ export class StepperComponent {
|
|
|
121
147
|
<ion-icon name="close"></ion-icon>
|
|
122
148
|
} @else if (step.icon) {
|
|
123
149
|
<ion-icon [name]="step.icon"></ion-icon>
|
|
124
|
-
} @else if (
|
|
150
|
+
} @else if (resolvedProps.showNumbers !== false) {
|
|
125
151
|
<span class="step-number">{{ i + 1 }}</span>
|
|
126
152
|
} @else {
|
|
127
153
|
<ion-icon name="ellipse"></ion-icon>
|
|
128
154
|
}
|
|
129
155
|
</div>
|
|
130
|
-
@if (!isLast &&
|
|
156
|
+
@if (!isLast && resolvedProps.connectorStyle !== 'none') {
|
|
131
157
|
<div
|
|
132
158
|
class="step-connector"
|
|
133
|
-
[class.dashed]="
|
|
159
|
+
[class.dashed]="resolvedProps.connectorStyle === 'dashed'"
|
|
134
160
|
[class.completed]="getStepStatus(step, i) === 'completed'"
|
|
135
161
|
></div>
|
|
136
162
|
}
|
|
@@ -156,10 +182,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
156
182
|
args: [{ selector: 'val-stepper', standalone: true, imports: [CommonModule, IonIcon], template: `
|
|
157
183
|
<div
|
|
158
184
|
class="stepper-container"
|
|
159
|
-
[class.horizontal]="
|
|
160
|
-
[class.vertical]="
|
|
185
|
+
[class.horizontal]="resolvedProps.orientation !== 'vertical'"
|
|
186
|
+
[class.vertical]="resolvedProps.orientation === 'vertical'"
|
|
161
187
|
>
|
|
162
|
-
@for (step of
|
|
188
|
+
@for (step of resolvedProps.steps; track step.value; let i = $index; let isLast = $last) {
|
|
163
189
|
<div
|
|
164
190
|
class="step"
|
|
165
191
|
[class.current]="getStepStatus(step, i) === 'current'"
|
|
@@ -177,16 +203,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
177
203
|
<ion-icon name="close"></ion-icon>
|
|
178
204
|
} @else if (step.icon) {
|
|
179
205
|
<ion-icon [name]="step.icon"></ion-icon>
|
|
180
|
-
} @else if (
|
|
206
|
+
} @else if (resolvedProps.showNumbers !== false) {
|
|
181
207
|
<span class="step-number">{{ i + 1 }}</span>
|
|
182
208
|
} @else {
|
|
183
209
|
<ion-icon name="ellipse"></ion-icon>
|
|
184
210
|
}
|
|
185
211
|
</div>
|
|
186
|
-
@if (!isLast &&
|
|
212
|
+
@if (!isLast && resolvedProps.connectorStyle !== 'none') {
|
|
187
213
|
<div
|
|
188
214
|
class="step-connector"
|
|
189
|
-
[class.dashed]="
|
|
215
|
+
[class.dashed]="resolvedProps.connectorStyle === 'dashed'"
|
|
190
216
|
[class.completed]="getStepStatus(step, i) === 'completed'"
|
|
191
217
|
></div>
|
|
192
218
|
}
|
|
@@ -206,9 +232,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
206
232
|
}
|
|
207
233
|
</div>
|
|
208
234
|
`, styles: [":host{display:block}.stepper-container{display:flex}.stepper-container.horizontal{flex-direction:row;align-items:flex-start}.stepper-container.horizontal .step{flex:1;flex-direction:column;align-items:center;text-align:center}.stepper-container.horizontal .step .step-indicator{flex-direction:row;width:100%;justify-content:center}.stepper-container.horizontal .step .step-connector{flex:1;height:2px;margin:0 8px;align-self:center}.stepper-container.vertical{flex-direction:column}.stepper-container.vertical .step{flex-direction:row;align-items:flex-start}.stepper-container.vertical .step .step-indicator{flex-direction:column;align-items:center}.stepper-container.vertical .step .step-connector{width:2px;flex:1;min-height:24px;margin:8px 0}.stepper-container.vertical .step .step-content{margin-left:12px;padding-bottom:24px}.step{display:flex;cursor:default}.step.clickable{cursor:pointer}.step.clickable:hover .step-circle{transform:scale(1.1)}.step.disabled{opacity:.5;pointer-events:none}.step.current .step-circle{box-shadow:0 0 0 4px rgba(var(--ion-color-primary-rgb),.2)}.step-indicator{display:flex;position:relative}.step-circle{width:32px;height:32px;border-radius:50%;display:flex;align-items:center;justify-content:center;background-color:var(--step-color, var(--ion-color-medium));color:#fff;font-weight:600;font-size:14px;transition:transform .2s,box-shadow .2s;flex-shrink:0}.step-circle ion-icon{font-size:18px}.step-connector{background-color:var(--ion-color-light-shade);transition:background-color .3s}.step-connector.completed{background-color:var(--ion-color-primary)}.step-connector.dashed{background:repeating-linear-gradient(90deg,var(--ion-color-light-shade),var(--ion-color-light-shade) 4px,transparent 4px,transparent 8px)}.step-connector.dashed.completed{background:repeating-linear-gradient(90deg,var(--ion-color-primary),var(--ion-color-primary) 4px,transparent 4px,transparent 8px)}.step-content{margin-top:8px}.step-label{font-weight:500;font-size:14px;color:var(--ion-text-color)}.step-label .optional-text{font-weight:400;font-size:12px;color:var(--ion-color-medium);margin-left:4px}.step-description{font-size:12px;color:var(--ion-color-medium);margin-top:4px}\n"] }]
|
|
209
|
-
}], propDecorators: {
|
|
235
|
+
}], propDecorators: { preset: [{
|
|
236
|
+
type: Input
|
|
237
|
+
}], props: [{
|
|
210
238
|
type: Input
|
|
211
239
|
}], stepChange: [{
|
|
212
240
|
type: Output
|
|
213
241
|
}] } });
|
|
214
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stepper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/stepper/stepper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter } from '@angular/core';\nimport { IonIcon } from '@ionic/angular/standalone';\nimport { CommonModule } from '@angular/common';\nimport { StepperMetadata, StepMetadata, StepChangeEvent, StepStatus } from './types';\nimport { addIcons } from 'ionicons';\nimport { checkmark, close, ellipse } from 'ionicons/icons';\n\naddIcons({ checkmark, close, ellipse });\n\n@Component({\n  selector: 'val-stepper',\n  standalone: true,\n  imports: [CommonModule, IonIcon],\n  template: `\n    <div\n      class=\"stepper-container\"\n      [class.horizontal]=\"props.orientation !== 'vertical'\"\n      [class.vertical]=\"props.orientation === 'vertical'\"\n    >\n      @for (step of props.steps; track step.value; let i = $index; let isLast = $last) {\n        <div\n          class=\"step\"\n          [class.current]=\"getStepStatus(step, i) === 'current'\"\n          [class.completed]=\"getStepStatus(step, i) === 'completed'\"\n          [class.error]=\"getStepStatus(step, i) === 'error'\"\n          [class.disabled]=\"step.disabled\"\n          [class.clickable]=\"isStepClickable(step, i)\"\n          (click)=\"onStepClick(step, i)\"\n        >\n          <div class=\"step-indicator\">\n            <div class=\"step-circle\" [style.--step-color]=\"getStepColor(step, i)\">\n              @if (getStepStatus(step, i) === 'completed') {\n                <ion-icon name=\"checkmark\"></ion-icon>\n              } @else if (getStepStatus(step, i) === 'error') {\n                <ion-icon name=\"close\"></ion-icon>\n              } @else if (step.icon) {\n                <ion-icon [name]=\"step.icon\"></ion-icon>\n              } @else if (props.showNumbers !== false) {\n                <span class=\"step-number\">{{ i + 1 }}</span>\n              } @else {\n                <ion-icon name=\"ellipse\"></ion-icon>\n              }\n            </div>\n            @if (!isLast && props.connectorStyle !== 'none') {\n              <div\n                class=\"step-connector\"\n                [class.dashed]=\"props.connectorStyle === 'dashed'\"\n                [class.completed]=\"getStepStatus(step, i) === 'completed'\"\n              ></div>\n            }\n          </div>\n          <div class=\"step-content\">\n            <div class=\"step-label\">\n              {{ getStepLabel(step) }}\n              @if (step.optional) {\n                <span class=\"optional-text\">(Optional)</span>\n              }\n            </div>\n            @if (step.description) {\n              <div class=\"step-description\">{{ step.description }}</div>\n            }\n          </div>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./stepper.component.scss'],\n})\n/**\n * val-stepper\n *\n * A stepper component for multi-step processes.\n * Supports horizontal and vertical orientations.\n *\n * @example Basic usage\n * <val-stepper [props]=\"{\n *   steps: [\n *     { value: 'info', label: 'Personal Info' },\n *     { value: 'address', label: 'Address' },\n *     { value: 'payment', label: 'Payment' },\n *     { value: 'confirm', label: 'Confirm' }\n *   ],\n *   currentIndex: 1\n * }\" (stepChange)=\"onStepChange($event)\"></val-stepper>\n *\n * @example Vertical with descriptions\n * <val-stepper [props]=\"{\n *   steps: [\n *     { value: 'step1', label: 'Step 1', description: 'Description here' },\n *     { value: 'step2', label: 'Step 2', optional: true }\n *   ],\n *   orientation: 'vertical',\n *   currentIndex: 0\n * }\"></val-stepper>\n *\n * @input props: StepperMetadata - Configuration for the stepper\n * @output stepChange: StepChangeEvent - Emits when step changes\n */\nexport class StepperComponent {\n  @Input() props: StepperMetadata;\n\n  @Output() stepChange = new EventEmitter<StepChangeEvent>();\n\n  getCurrentIndex(): number {\n    if (this.props.currentIndex !== undefined) {\n      return this.props.currentIndex;\n    }\n    if (this.props.currentStep) {\n      return this.props.steps.findIndex((s) => s.value === this.props.currentStep);\n    }\n    return 0;\n  }\n\n  getStepStatus(step: StepMetadata, index: number): StepStatus {\n    if (step.status) return step.status;\n\n    const currentIndex = this.getCurrentIndex();\n    if (index < currentIndex) return 'completed';\n    if (index === currentIndex) return 'current';\n    return 'pending';\n  }\n\n  getStepColor(step: StepMetadata, index: number): string {\n    const status = this.getStepStatus(step, index);\n    const color = this.props.color || 'primary';\n\n    switch (status) {\n      case 'completed':\n        return `var(--ion-color-${color})`;\n      case 'current':\n        return `var(--ion-color-${color})`;\n      case 'error':\n        return 'var(--ion-color-danger)';\n      default:\n        return 'var(--ion-color-medium)';\n    }\n  }\n\n  isStepClickable(step: StepMetadata, index: number): boolean {\n    if (step.disabled) return false;\n    if (!this.props.linear) return true;\n\n    const currentIndex = this.getCurrentIndex();\n    // In linear mode, can only go to completed steps or next step\n    return index <= currentIndex || (step.editable && index < currentIndex);\n  }\n\n  onStepClick(step: StepMetadata, index: number): void {\n    if (!this.isStepClickable(step, index)) return;\n\n    const currentIndex = this.getCurrentIndex();\n    if (index === currentIndex) return;\n\n    this.stepChange.emit({\n      step,\n      index,\n      previousIndex: currentIndex,\n    });\n  }\n\n  getStepLabel(step: StepMetadata): string {\n    return step.label || step.contentFallback || '';\n  }\n}\n"]}
|
|
242
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stepper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/stepper/stepper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnChanges, OnInit, Output, EventEmitter, SimpleChanges } from '@angular/core';\nimport { IonIcon } from '@ionic/angular/standalone';\nimport { CommonModule } from '@angular/common';\nimport { PresetService } from '../../../services/presets';\nimport { StepperMetadata, StepMetadata, StepChangeEvent, StepStatus } from './types';\nimport { addIcons } from 'ionicons';\nimport { checkmark, close, ellipse } from 'ionicons/icons';\n\naddIcons({ checkmark, close, ellipse });\n\n@Component({\n  selector: 'val-stepper',\n  standalone: true,\n  imports: [CommonModule, IonIcon],\n  template: `\n    <div\n      class=\"stepper-container\"\n      [class.horizontal]=\"resolvedProps.orientation !== 'vertical'\"\n      [class.vertical]=\"resolvedProps.orientation === 'vertical'\"\n    >\n      @for (step of resolvedProps.steps; track step.value; let i = $index; let isLast = $last) {\n        <div\n          class=\"step\"\n          [class.current]=\"getStepStatus(step, i) === 'current'\"\n          [class.completed]=\"getStepStatus(step, i) === 'completed'\"\n          [class.error]=\"getStepStatus(step, i) === 'error'\"\n          [class.disabled]=\"step.disabled\"\n          [class.clickable]=\"isStepClickable(step, i)\"\n          (click)=\"onStepClick(step, i)\"\n        >\n          <div class=\"step-indicator\">\n            <div class=\"step-circle\" [style.--step-color]=\"getStepColor(step, i)\">\n              @if (getStepStatus(step, i) === 'completed') {\n                <ion-icon name=\"checkmark\"></ion-icon>\n              } @else if (getStepStatus(step, i) === 'error') {\n                <ion-icon name=\"close\"></ion-icon>\n              } @else if (step.icon) {\n                <ion-icon [name]=\"step.icon\"></ion-icon>\n              } @else if (resolvedProps.showNumbers !== false) {\n                <span class=\"step-number\">{{ i + 1 }}</span>\n              } @else {\n                <ion-icon name=\"ellipse\"></ion-icon>\n              }\n            </div>\n            @if (!isLast && resolvedProps.connectorStyle !== 'none') {\n              <div\n                class=\"step-connector\"\n                [class.dashed]=\"resolvedProps.connectorStyle === 'dashed'\"\n                [class.completed]=\"getStepStatus(step, i) === 'completed'\"\n              ></div>\n            }\n          </div>\n          <div class=\"step-content\">\n            <div class=\"step-label\">\n              {{ getStepLabel(step) }}\n              @if (step.optional) {\n                <span class=\"optional-text\">(Optional)</span>\n              }\n            </div>\n            @if (step.description) {\n              <div class=\"step-description\">{{ step.description }}</div>\n            }\n          </div>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./stepper.component.scss'],\n})\n/**\n * val-stepper\n *\n * A stepper component for multi-step processes.\n * Supports horizontal and vertical orientations.\n *\n * @example Basic usage\n * <val-stepper [props]=\"{\n *   steps: [\n *     { value: 'info', label: 'Personal Info' },\n *     { value: 'address', label: 'Address' },\n *     { value: 'payment', label: 'Payment' },\n *     { value: 'confirm', label: 'Confirm' }\n *   ],\n *   currentIndex: 1\n * }\" (stepChange)=\"onStepChange($event)\"></val-stepper>\n *\n * @example Vertical with descriptions\n * <val-stepper [props]=\"{\n *   steps: [\n *     { value: 'step1', label: 'Step 1', description: 'Description here' },\n *     { value: 'step2', label: 'Step 2', optional: true }\n *   ],\n *   orientation: 'vertical',\n *   currentIndex: 0\n * }\"></val-stepper>\n *\n * @input props: StepperMetadata - Configuration for the stepper\n * @output stepChange: StepChangeEvent - Emits when step changes\n */\nexport class StepperComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  /**\n   * Preset name to apply. Presets define reusable stepper configurations\n   * that can be registered at app level via provideValtechPresets().\n   */\n  @Input() preset?: string;\n\n  /**\n   * Stepper configuration object. Values here override preset values.\n   */\n  @Input() props: Partial<StepperMetadata> = {};\n\n  /**\n   * Resolved props after merging preset + explicit props.\n   */\n  resolvedProps: StepperMetadata = {} as StepperMetadata;\n\n  @Output() stepChange = new EventEmitter<StepChangeEvent>();\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  private resolveProps(): void {\n    const presetProps = this.preset\n      ? (this.presets.get('stepper', this.preset) as Partial<StepperMetadata>)\n      : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as StepperMetadata;\n  }\n\n  getCurrentIndex(): number {\n    if (this.resolvedProps.currentIndex !== undefined) {\n      return this.resolvedProps.currentIndex;\n    }\n    if (this.resolvedProps.currentStep) {\n      return this.resolvedProps.steps?.findIndex((s) => s.value === this.resolvedProps.currentStep) ?? 0;\n    }\n    return 0;\n  }\n\n  getStepStatus(step: StepMetadata, index: number): StepStatus {\n    if (step.status) return step.status;\n\n    const currentIndex = this.getCurrentIndex();\n    if (index < currentIndex) return 'completed';\n    if (index === currentIndex) return 'current';\n    return 'pending';\n  }\n\n  getStepColor(step: StepMetadata, index: number): string {\n    const status = this.getStepStatus(step, index);\n    const color = this.resolvedProps.color || 'primary';\n\n    switch (status) {\n      case 'completed':\n        return `var(--ion-color-${color})`;\n      case 'current':\n        return `var(--ion-color-${color})`;\n      case 'error':\n        return 'var(--ion-color-danger)';\n      default:\n        return 'var(--ion-color-medium)';\n    }\n  }\n\n  isStepClickable(step: StepMetadata, index: number): boolean {\n    if (step.disabled) return false;\n    if (!this.resolvedProps.linear) return true;\n\n    const currentIndex = this.getCurrentIndex();\n    return index <= currentIndex || (step.editable && index < currentIndex);\n  }\n\n  onStepClick(step: StepMetadata, index: number): void {\n    if (!this.isStepClickable(step, index)) return;\n\n    const currentIndex = this.getCurrentIndex();\n    if (index === currentIndex) return;\n\n    this.stepChange.emit({\n      step,\n      index,\n      previousIndex: currentIndex,\n    });\n  }\n\n  getStepLabel(step: StepMetadata): string {\n    return step.label || step.contentFallback || '';\n  }\n}\n"]}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
1
|
+
import { Component, inject, Input, Output, EventEmitter } from '@angular/core';
|
|
2
2
|
import { IonTabBar, IonTabButton, IonIcon, IonLabel, IonBadge } from '@ionic/angular/standalone';
|
|
3
3
|
import { CommonModule } from '@angular/common';
|
|
4
|
+
import { PresetService } from '../../../services/presets';
|
|
4
5
|
import { addIcons } from 'ionicons';
|
|
5
6
|
import { home, settings, person, search, heart, star, notifications, mail, calendar, folder } from 'ionicons/icons';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
@@ -9,8 +10,12 @@ addIcons({ home, settings, person, search, heart, star, notifications, mail, cal
|
|
|
9
10
|
* val-tabs
|
|
10
11
|
*
|
|
11
12
|
* A tab bar component for navigation between views.
|
|
13
|
+
* Supports presets for reusable configurations.
|
|
12
14
|
*
|
|
13
|
-
* @example
|
|
15
|
+
* @example With preset (recommended):
|
|
16
|
+
* <val-tabs preset="main-nav" [props]="{ tabs: [...], selectedTab: 'home' }" (tabChange)="onTabChange($event)"></val-tabs>
|
|
17
|
+
*
|
|
18
|
+
* @example Static (backwards compatible):
|
|
14
19
|
* <val-tabs [props]="{
|
|
15
20
|
* tabs: [
|
|
16
21
|
* { value: 'home', label: 'Inicio', icon: 'home' },
|
|
@@ -20,13 +25,44 @@ addIcons({ home, settings, person, search, heart, star, notifications, mail, cal
|
|
|
20
25
|
* selectedTab: 'home'
|
|
21
26
|
* }" (tabChange)="onTabChange($event)"></val-tabs>
|
|
22
27
|
*
|
|
28
|
+
* @input preset: string - Name of preset to apply
|
|
23
29
|
* @input props: TabsMetadata - Configuration for the tabs
|
|
24
30
|
* @output tabChange: TabMetadata - Emits when a tab is selected
|
|
25
31
|
*/
|
|
26
32
|
export class TabsComponent {
|
|
27
33
|
constructor() {
|
|
34
|
+
this.presets = inject(PresetService);
|
|
35
|
+
/**
|
|
36
|
+
* Tabs configuration object. Values here override preset values.
|
|
37
|
+
*/
|
|
38
|
+
this.props = {};
|
|
39
|
+
/**
|
|
40
|
+
* Resolved props after merging preset + explicit props.
|
|
41
|
+
*/
|
|
42
|
+
this.resolvedProps = {};
|
|
28
43
|
this.tabChange = new EventEmitter();
|
|
29
44
|
}
|
|
45
|
+
ngOnInit() {
|
|
46
|
+
this.resolveProps();
|
|
47
|
+
}
|
|
48
|
+
ngOnChanges(changes) {
|
|
49
|
+
if (changes['preset'] || changes['props']) {
|
|
50
|
+
this.resolveProps();
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Merge preset configuration with explicit props.
|
|
55
|
+
* Explicit props take precedence over preset values.
|
|
56
|
+
*/
|
|
57
|
+
resolveProps() {
|
|
58
|
+
const presetProps = this.preset
|
|
59
|
+
? this.presets.get('tabs', this.preset)
|
|
60
|
+
: {};
|
|
61
|
+
this.resolvedProps = {
|
|
62
|
+
...presetProps,
|
|
63
|
+
...this.props,
|
|
64
|
+
};
|
|
65
|
+
}
|
|
30
66
|
onTabClick(tab) {
|
|
31
67
|
if (!tab.disabled) {
|
|
32
68
|
this.tabChange.emit(tab);
|
|
@@ -36,24 +72,24 @@ export class TabsComponent {
|
|
|
36
72
|
return tab.label || tab.contentFallback || '';
|
|
37
73
|
}
|
|
38
74
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TabsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
39
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: TabsComponent, isStandalone: true, selector: "val-tabs", inputs: { props: "props" }, outputs: { tabChange: "tabChange" }, ngImport: i0, template: `
|
|
75
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: TabsComponent, isStandalone: true, selector: "val-tabs", inputs: { preset: "preset", props: "props" }, outputs: { tabChange: "tabChange" }, usesOnChanges: true, ngImport: i0, template: `
|
|
40
76
|
<ion-tab-bar
|
|
41
|
-
[color]="
|
|
42
|
-
[mode]="
|
|
43
|
-
[translucent]="
|
|
44
|
-
[selectedTab]="
|
|
77
|
+
[color]="resolvedProps.color"
|
|
78
|
+
[mode]="resolvedProps.mode"
|
|
79
|
+
[translucent]="resolvedProps.translucent ?? false"
|
|
80
|
+
[selectedTab]="resolvedProps.selectedTab || resolvedProps.tabs?.[0]?.value"
|
|
45
81
|
>
|
|
46
|
-
@for (tab of
|
|
82
|
+
@for (tab of resolvedProps.tabs; track tab.value) {
|
|
47
83
|
<ion-tab-button
|
|
48
84
|
[tab]="tab.value"
|
|
49
85
|
[disabled]="tab.disabled"
|
|
50
|
-
[layout]="
|
|
86
|
+
[layout]="resolvedProps.layout || 'icon-top'"
|
|
51
87
|
(click)="onTabClick(tab)"
|
|
52
88
|
>
|
|
53
|
-
@if (tab.icon &&
|
|
89
|
+
@if (tab.icon && resolvedProps.layout !== 'icon-hide') {
|
|
54
90
|
<ion-icon [name]="tab.icon"></ion-icon>
|
|
55
91
|
}
|
|
56
|
-
@if (getTabLabel(tab) &&
|
|
92
|
+
@if (getTabLabel(tab) && resolvedProps.layout !== 'label-hide') {
|
|
57
93
|
<ion-label>{{ getTabLabel(tab) }}</ion-label>
|
|
58
94
|
}
|
|
59
95
|
@if (tab.badge) {
|
|
@@ -68,22 +104,22 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
68
104
|
type: Component,
|
|
69
105
|
args: [{ selector: 'val-tabs', standalone: true, imports: [CommonModule, IonTabBar, IonTabButton, IonIcon, IonLabel, IonBadge], template: `
|
|
70
106
|
<ion-tab-bar
|
|
71
|
-
[color]="
|
|
72
|
-
[mode]="
|
|
73
|
-
[translucent]="
|
|
74
|
-
[selectedTab]="
|
|
107
|
+
[color]="resolvedProps.color"
|
|
108
|
+
[mode]="resolvedProps.mode"
|
|
109
|
+
[translucent]="resolvedProps.translucent ?? false"
|
|
110
|
+
[selectedTab]="resolvedProps.selectedTab || resolvedProps.tabs?.[0]?.value"
|
|
75
111
|
>
|
|
76
|
-
@for (tab of
|
|
112
|
+
@for (tab of resolvedProps.tabs; track tab.value) {
|
|
77
113
|
<ion-tab-button
|
|
78
114
|
[tab]="tab.value"
|
|
79
115
|
[disabled]="tab.disabled"
|
|
80
|
-
[layout]="
|
|
116
|
+
[layout]="resolvedProps.layout || 'icon-top'"
|
|
81
117
|
(click)="onTabClick(tab)"
|
|
82
118
|
>
|
|
83
|
-
@if (tab.icon &&
|
|
119
|
+
@if (tab.icon && resolvedProps.layout !== 'icon-hide') {
|
|
84
120
|
<ion-icon [name]="tab.icon"></ion-icon>
|
|
85
121
|
}
|
|
86
|
-
@if (getTabLabel(tab) &&
|
|
122
|
+
@if (getTabLabel(tab) && resolvedProps.layout !== 'label-hide') {
|
|
87
123
|
<ion-label>{{ getTabLabel(tab) }}</ion-label>
|
|
88
124
|
}
|
|
89
125
|
@if (tab.badge) {
|
|
@@ -93,9 +129,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
93
129
|
}
|
|
94
130
|
</ion-tab-bar>
|
|
95
131
|
`, styles: [":host{display:block}ion-tab-bar{--background: var(--ion-background-color);--border: 1px solid var(--ion-color-light-shade);border-radius:0}ion-tab-button{--color: var(--ion-color-medium);--color-selected: var(--ion-color-primary);--padding-top: 8px;--padding-bottom: 8px}ion-tab-button ion-icon{font-size:22px}ion-tab-button ion-label{font-size:12px;font-weight:500;text-transform:none}ion-tab-button ion-badge{position:absolute;top:4px;right:calc(50% - 20px);font-size:10px;padding:2px 6px;min-width:16px}\n"] }]
|
|
96
|
-
}], propDecorators: {
|
|
132
|
+
}], propDecorators: { preset: [{
|
|
133
|
+
type: Input
|
|
134
|
+
}], props: [{
|
|
97
135
|
type: Input
|
|
98
136
|
}], tabChange: [{
|
|
99
137
|
type: Output
|
|
100
138
|
}] } });
|
|
101
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/tabs/tabs.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAqB,MAAM,EAAE,YAAY,EAAiB,MAAM,eAAe,CAAC;AACjH,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,2BAA2B,CAAC;AACjG,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAE1D,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,aAAa,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;;AAEpH,QAAQ,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,aAAa,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;AAmCjG;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,MAAM,OAAO,aAAa;IAxD1B;QAyDU,YAAO,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QAWxC;;WAEG;QACM,UAAK,GAA0B,EAAE,CAAC;QAE3C;;WAEG;QACH,kBAAa,GAAiB,EAAkB,CAAC;QAEvC,cAAS,GAAG,IAAI,YAAY,EAAe,CAAC;KAoCvD;IAlCC,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;YAC7B,CAAC,CAAE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,MAAM,CAA2B;YAClE,CAAC,CAAC,EAAE,CAAC;QAEP,IAAI,CAAC,aAAa,GAAG;YACnB,GAAG,WAAW;YACd,GAAG,IAAI,CAAC,KAAK;SACE,CAAC;IACpB,CAAC;IAED,UAAU,CAAC,GAAgB;QACzB,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;YAClB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAED,WAAW,CAAC,GAAgB;QAC1B,OAAO,GAAG,CAAC,KAAK,IAAI,GAAG,CAAC,eAAe,IAAI,EAAE,CAAC;IAChD,CAAC;+GAzDU,aAAa;mGAAb,aAAa,4KApDd;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BT,qkBA3BS,YAAY,+BAAE,SAAS,iHAAE,YAAY,2JAAE,OAAO,2JAAE,QAAQ,6FAAE,QAAQ;;4FAqDjE,aAAa;kBAxDzB,SAAS;+BACE,UAAU,cACR,IAAI,WACP,CAAC,YAAY,EAAE,SAAS,EAAE,YAAY,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,CAAC,YACnE;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BT;8BAoCQ,MAAM;sBAAd,KAAK;gBAKG,KAAK;sBAAb,KAAK;gBAOI,SAAS;sBAAlB,MAAM","sourcesContent":["import { Component, inject, Input, OnChanges, OnInit, Output, EventEmitter, SimpleChanges } from '@angular/core';\nimport { IonTabBar, IonTabButton, IonIcon, IonLabel, IonBadge } from '@ionic/angular/standalone';\nimport { CommonModule } from '@angular/common';\nimport { PresetService } from '../../../services/presets';\nimport { TabsMetadata, TabMetadata } from './types';\nimport { addIcons } from 'ionicons';\nimport { home, settings, person, search, heart, star, notifications, mail, calendar, folder } from 'ionicons/icons';\n\naddIcons({ home, settings, person, search, heart, star, notifications, mail, calendar, folder });\n\n@Component({\n  selector: 'val-tabs',\n  standalone: true,\n  imports: [CommonModule, IonTabBar, IonTabButton, IonIcon, IonLabel, IonBadge],\n  template: `\n    <ion-tab-bar\n      [color]=\"resolvedProps.color\"\n      [mode]=\"resolvedProps.mode\"\n      [translucent]=\"resolvedProps.translucent ?? false\"\n      [selectedTab]=\"resolvedProps.selectedTab || resolvedProps.tabs?.[0]?.value\"\n    >\n      @for (tab of resolvedProps.tabs; track tab.value) {\n        <ion-tab-button\n          [tab]=\"tab.value\"\n          [disabled]=\"tab.disabled\"\n          [layout]=\"resolvedProps.layout || 'icon-top'\"\n          (click)=\"onTabClick(tab)\"\n        >\n          @if (tab.icon && resolvedProps.layout !== 'icon-hide') {\n            <ion-icon [name]=\"tab.icon\"></ion-icon>\n          }\n          @if (getTabLabel(tab) && resolvedProps.layout !== 'label-hide') {\n            <ion-label>{{ getTabLabel(tab) }}</ion-label>\n          }\n          @if (tab.badge) {\n            <ion-badge [color]=\"tab.badgeColor || 'danger'\">{{ tab.badge }}</ion-badge>\n          }\n        </ion-tab-button>\n      }\n    </ion-tab-bar>\n  `,\n  styleUrls: ['./tabs.component.scss'],\n})\n/**\n * val-tabs\n *\n * A tab bar component for navigation between views.\n * Supports presets for reusable configurations.\n *\n * @example With preset (recommended):\n * <val-tabs preset=\"main-nav\" [props]=\"{ tabs: [...], selectedTab: 'home' }\" (tabChange)=\"onTabChange($event)\"></val-tabs>\n *\n * @example Static (backwards compatible):\n * <val-tabs [props]=\"{\n *   tabs: [\n *     { value: 'home', label: 'Inicio', icon: 'home' },\n *     { value: 'search', label: 'Buscar', icon: 'search' },\n *     { value: 'profile', label: 'Perfil', icon: 'person' }\n *   ],\n *   selectedTab: 'home'\n * }\" (tabChange)=\"onTabChange($event)\"></val-tabs>\n *\n * @input preset: string - Name of preset to apply\n * @input props: TabsMetadata - Configuration for the tabs\n * @output tabChange: TabMetadata - Emits when a tab is selected\n */\nexport class TabsComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  /**\n   * Preset name to apply. Presets define reusable tabs configurations\n   * that can be registered at app level via provideValtechPresets().\n   *\n   * @example\n   * <val-tabs preset=\"main-nav\" [props]=\"{ tabs: [...] }\"></val-tabs>\n   */\n  @Input() preset?: string;\n\n  /**\n   * Tabs configuration object. Values here override preset values.\n   */\n  @Input() props: Partial<TabsMetadata> = {};\n\n  /**\n   * Resolved props after merging preset + explicit props.\n   */\n  resolvedProps: TabsMetadata = {} as TabsMetadata;\n\n  @Output() tabChange = new EventEmitter<TabMetadata>();\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\n      ? (this.presets.get('tabs', this.preset) as Partial<TabsMetadata>)\n      : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as TabsMetadata;\n  }\n\n  onTabClick(tab: TabMetadata): void {\n    if (!tab.disabled) {\n      this.tabChange.emit(tab);\n    }\n  }\n\n  getTabLabel(tab: TabMetadata): string {\n    return tab.label || tab.contentFallback || '';\n  }\n}\n"]}
|