valtech-components 2.0.434 → 2.0.435
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/fab/fab.component.mjs +5 -4
- package/esm2022/lib/components/molecules/comment-input/comment-input.component.mjs +13 -5
- package/esm2022/lib/components/molecules/comment-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +23 -4
- package/esm2022/lib/components/molecules/date-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/feedback-form/feedback-form.component.mjs +352 -0
- package/esm2022/lib/components/molecules/feedback-form/types.mjs +2 -0
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +7 -7
- package/esm2022/lib/components/molecules/file-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/number-from-to/number-from-to.component.mjs +23 -9
- package/esm2022/lib/components/molecules/number-from-to/types.mjs +2 -0
- package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +2 -2
- package/esm2022/lib/components/molecules/pin-input/types.mjs +2 -0
- package/esm2022/lib/components/organisms/form/form.component.mjs +108 -30
- package/esm2022/lib/services/content/content-types/blog.mjs +275 -0
- package/esm2022/lib/services/content/content-types/documentation.mjs +303 -0
- package/esm2022/lib/services/content/content-types/news.mjs +277 -0
- package/esm2022/lib/services/content/index.mjs +51 -0
- package/esm2022/lib/services/content/transformer.mjs +265 -0
- package/esm2022/lib/services/content/types.mjs +41 -0
- package/esm2022/lib/services/feedback/config.mjs +49 -0
- package/esm2022/lib/services/feedback/feedback.service.mjs +174 -0
- package/esm2022/lib/services/feedback/index.mjs +44 -0
- package/esm2022/lib/services/feedback/types.mjs +30 -0
- package/esm2022/lib/services/firebase/index.mjs +3 -1
- package/esm2022/lib/services/firebase/shared-config.mjs +132 -0
- package/esm2022/public-api.mjs +14 -1
- package/fesm2022/valtech-components.mjs +2225 -177
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/fab/fab.component.d.ts +2 -0
- package/lib/components/molecules/comment-input/comment-input.component.d.ts +4 -4
- package/lib/components/molecules/comment-input/types.d.ts +59 -0
- package/lib/components/molecules/date-input/date-input.component.d.ts +4 -3
- package/lib/components/molecules/date-input/types.d.ts +74 -0
- package/lib/components/molecules/feedback-form/feedback-form.component.d.ts +56 -0
- package/lib/components/molecules/feedback-form/types.d.ts +54 -0
- package/lib/components/molecules/file-input/file-input.component.d.ts +5 -3
- package/lib/components/molecules/file-input/types.d.ts +72 -0
- package/lib/components/molecules/number-from-to/number-from-to.component.d.ts +8 -2
- package/lib/components/molecules/number-from-to/types.d.ts +76 -0
- package/lib/components/molecules/pin-input/pin-input.component.d.ts +4 -3
- package/lib/components/molecules/pin-input/types.d.ts +63 -0
- package/lib/components/organisms/form/form.component.d.ts +16 -2
- package/lib/services/content/content-types/blog.d.ts +148 -0
- package/lib/services/content/content-types/documentation.d.ts +183 -0
- package/lib/services/content/content-types/news.d.ts +162 -0
- package/lib/services/content/index.d.ts +49 -0
- package/lib/services/content/transformer.d.ts +96 -0
- package/lib/services/content/types.d.ts +220 -0
- package/lib/services/feedback/config.d.ts +35 -0
- package/lib/services/feedback/feedback.service.d.ts +76 -0
- package/lib/services/feedback/index.d.ts +40 -0
- package/lib/services/feedback/types.d.ts +107 -0
- package/lib/services/firebase/index.d.ts +1 -0
- package/lib/services/firebase/shared-config.d.ts +120 -0
- package/package.json +1 -1
- package/public-api.d.ts +9 -0
|
@@ -4,6 +4,7 @@ import { CommonModule } from '@angular/common';
|
|
|
4
4
|
import { addIcons } from 'ionicons';
|
|
5
5
|
import { add, close, share, create, trash, heart, star, camera, mic, send } from 'ionicons/icons';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "../../../services/icons.service";
|
|
7
8
|
addIcons({ add, close, share, create, trash, heart, star, camera, mic, send });
|
|
8
9
|
/**
|
|
9
10
|
* val-fab
|
|
@@ -35,7 +36,7 @@ addIcons({ add, close, share, create, trash, heart, star, camera, mic, send });
|
|
|
35
36
|
* @output actionClick: FabActionMetadata - Emits when an action is clicked
|
|
36
37
|
*/
|
|
37
38
|
export class FabComponent {
|
|
38
|
-
constructor() {
|
|
39
|
+
constructor(icon) {
|
|
39
40
|
this.fabClick = new EventEmitter();
|
|
40
41
|
this.actionClick = new EventEmitter();
|
|
41
42
|
this.isActivated = false;
|
|
@@ -76,7 +77,7 @@ export class FabComponent {
|
|
|
76
77
|
this.isActivated = false;
|
|
77
78
|
}
|
|
78
79
|
}
|
|
79
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FabComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
80
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FabComponent, deps: [{ token: i1.IconService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
80
81
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: FabComponent, isStandalone: true, selector: "val-fab", inputs: { props: "props" }, outputs: { fabClick: "fabClick", actionClick: "actionClick" }, ngImport: i0, template: `
|
|
81
82
|
<ion-fab
|
|
82
83
|
[vertical]="getVertical()"
|
|
@@ -142,11 +143,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
142
143
|
}
|
|
143
144
|
</ion-fab>
|
|
144
145
|
`, styles: [":host{display:block}ion-fab{position:relative}ion-fab-button{--box-shadow: 0 4px 12px rgba(0, 0, 0, .15);transition:transform .2s}ion-fab-button:hover:not([disabled]){transform:scale(1.05)}ion-fab-button:active:not([disabled]){transform:scale(.95)}ion-fab-list ion-fab-button{--box-shadow: 0 2px 8px rgba(0, 0, 0, .1)}\n"] }]
|
|
145
|
-
}], propDecorators: { props: [{
|
|
146
|
+
}], ctorParameters: () => [{ type: i1.IconService }], propDecorators: { props: [{
|
|
146
147
|
type: Input
|
|
147
148
|
}], fabClick: [{
|
|
148
149
|
type: Output
|
|
149
150
|
}], actionClick: [{
|
|
150
151
|
type: Output
|
|
151
152
|
}] } });
|
|
152
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
153
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fab.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/atoms/fab/fab.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter } from '@angular/core';\nimport { IonFab, IonFabButton, IonFabList, IonIcon, IonLabel } from '@ionic/angular/standalone';\nimport { CommonModule } from '@angular/common';\nimport { FabMetadata, FabActionMetadata } from './types';\nimport { addIcons } from 'ionicons';\nimport { add, close, share, create, trash, heart, star, camera, mic, send } from 'ionicons/icons';\nimport { IconService } from '../../../services/icons.service';\n\naddIcons({ add, close, share, create, trash, heart, star, camera, mic, send });\n\n@Component({\n  selector: 'val-fab',\n  standalone: true,\n  imports: [CommonModule, IonFab, IonFabButton, IonFabList, IonIcon, IonLabel],\n  template: `\n    <ion-fab\n      [vertical]=\"getVertical()\"\n      [horizontal]=\"getHorizontal()\"\n      [activated]=\"isActivated\"\n    >\n      <ion-fab-button\n        [color]=\"props.color || 'primary'\"\n        [size]=\"props.size\"\n        [disabled]=\"props.disabled\"\n        [translucent]=\"props.translucent\"\n        (click)=\"onMainClick()\"\n      >\n        <ion-icon [name]=\"getMainIcon()\"></ion-icon>\n      </ion-fab-button>\n\n      @if (props.actions && props.actions.length > 0) {\n        <ion-fab-list [side]=\"props.side || 'top'\">\n          @for (action of props.actions; track action.token || $index) {\n            <ion-fab-button\n              [color]=\"action.color || 'light'\"\n              [disabled]=\"action.disabled\"\n              (click)=\"onActionClick(action)\"\n            >\n              <ion-icon [name]=\"action.icon\"></ion-icon>\n            </ion-fab-button>\n          }\n        </ion-fab-list>\n      }\n    </ion-fab>\n  `,\n  styleUrls: ['./fab.component.scss'],\n})\n/**\n * val-fab\n *\n * A Floating Action Button component with optional speed dial actions.\n *\n * @example Simple FAB\n * <val-fab [props]=\"{\n *   icon: 'add',\n *   color: 'primary',\n *   position: 'bottom-end'\n * }\" (fabClick)=\"onCreate()\"></val-fab>\n *\n * @example Speed dial FAB\n * <val-fab [props]=\"{\n *   icon: 'add',\n *   iconActivated: 'close',\n *   position: 'bottom-end',\n *   actions: [\n *     { icon: 'camera', label: 'Photo', token: 'photo' },\n *     { icon: 'mic', label: 'Audio', token: 'audio' },\n *     { icon: 'create', label: 'Note', token: 'note' }\n *   ],\n *   side: 'top'\n * }\" (actionClick)=\"onAction($event)\"></val-fab>\n *\n * @input props: FabMetadata - Configuration for the FAB\n * @output fabClick: void - Emits when main FAB is clicked (without actions)\n * @output actionClick: FabActionMetadata - Emits when an action is clicked\n */\nexport class FabComponent {\n  @Input() props: FabMetadata;\n\n  @Output() fabClick = new EventEmitter<void>();\n  @Output() actionClick = new EventEmitter<FabActionMetadata>();\n\n  isActivated = false;\n\n  constructor(icon: IconService) {}\n\n  getVertical(): 'top' | 'bottom' | 'center' {\n    const pos = this.props.position || 'bottom-end';\n    if (pos.startsWith('top')) return 'top';\n    if (pos.startsWith('center')) return 'center';\n    return 'bottom';\n  }\n\n  getHorizontal(): 'start' | 'end' | 'center' {\n    const pos = this.props.position || 'bottom-end';\n    if (pos.endsWith('start')) return 'start';\n    if (pos.endsWith('center')) return 'center';\n    return 'end';\n  }\n\n  getMainIcon(): string {\n    if (this.isActivated && this.props.iconActivated) {\n      return this.props.iconActivated;\n    }\n    return this.props.icon;\n  }\n\n  onMainClick(): void {\n    if (this.props.actions && this.props.actions.length > 0) {\n      this.isActivated = !this.isActivated;\n    } else {\n      this.fabClick.emit();\n    }\n  }\n\n  onActionClick(action: FabActionMetadata): void {\n    this.actionClick.emit(action);\n    if (this.props.closeOnAction !== false) {\n      this.isActivated = false;\n    }\n  }\n}\n"]}
|
|
@@ -10,24 +10,32 @@ import * as i1 from "@angular/forms";
|
|
|
10
10
|
* A textarea input for multi-line comments, integrated with Angular forms.
|
|
11
11
|
*
|
|
12
12
|
* @example
|
|
13
|
-
* <val-comment-input [props]="{ control: myControl,
|
|
13
|
+
* <val-comment-input [props]="{ control: myControl, maxLength: 200 }"></val-comment-input>
|
|
14
14
|
*
|
|
15
|
-
* @input props:
|
|
15
|
+
* @input props: CommentInputMetadata - Configuration for the textarea (form control, maxLength, etc.)
|
|
16
16
|
*/
|
|
17
17
|
export class CommentInputComponent {
|
|
18
18
|
constructor() { }
|
|
19
19
|
ngOnInit() { }
|
|
20
20
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CommentInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
21
21
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CommentInputComponent, isStandalone: true, selector: "val-comment-input", inputs: { props: "props" }, ngImport: i0, template: `
|
|
22
|
-
<ion-textarea
|
|
22
|
+
<ion-textarea
|
|
23
|
+
[formControl]="props.control"
|
|
24
|
+
[counter]="props.showCounter !== false"
|
|
25
|
+
[maxlength]="props.maxLength"
|
|
26
|
+
></ion-textarea>
|
|
23
27
|
`, isInline: true, styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: IonTextarea, selector: "ion-textarea", inputs: ["autoGrow", "autocapitalize", "autofocus", "clearOnEdit", "color", "cols", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "maxlength", "minlength", "mode", "name", "placeholder", "readonly", "required", "rows", "shape", "spellcheck", "value", "wrap"] }] }); }
|
|
24
28
|
}
|
|
25
29
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CommentInputComponent, decorators: [{
|
|
26
30
|
type: Component,
|
|
27
31
|
args: [{ selector: 'val-comment-input', standalone: true, imports: [CommonModule, ReactiveFormsModule, IonCheckbox, IonTextarea], template: `
|
|
28
|
-
<ion-textarea
|
|
32
|
+
<ion-textarea
|
|
33
|
+
[formControl]="props.control"
|
|
34
|
+
[counter]="props.showCounter !== false"
|
|
35
|
+
[maxlength]="props.maxLength"
|
|
36
|
+
></ion-textarea>
|
|
29
37
|
` }]
|
|
30
38
|
}], ctorParameters: () => [], propDecorators: { props: [{
|
|
31
39
|
type: Input
|
|
32
40
|
}] } });
|
|
33
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbWVudC1pbnB1dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2NvbW1lbnQtaW5wdXQvY29tbWVudC1pbnB1dC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ3pELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3JELE9BQU8sRUFBRSxXQUFXLEVBQUUsV0FBVyxFQUFFLE1BQU0sMkJBQTJCLENBQUM7OztBQWdCckU7Ozs7Ozs7OztHQVNHO0FBQ0gsTUFBTSxPQUFPLHFCQUFxQjtJQVNoQyxnQkFBZSxDQUFDO0lBRWhCLFFBQVEsS0FBSSxDQUFDOytHQVhGLHFCQUFxQjttR0FBckIscUJBQXFCLHlHQW5CdEI7Ozs7OztHQU1ULHlFQVBTLFlBQVksOEJBQUUsbUJBQW1CLDZkQUFlLFdBQVc7OzRGQW9CMUQscUJBQXFCO2tCQXZCakMsU0FBUzsrQkFDRSxtQkFBbUIsY0FDakIsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLG1CQUFtQixFQUFFLFdBQVcsRUFBRSxXQUFXLENBQUMsWUFDNUQ7Ozs7OztHQU1UO3dEQW9CUSxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBSZWFjdGl2ZUZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgSW9uQ2hlY2tib3gsIElvblRleHRhcmVhIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5pbXBvcnQgeyBDb21tZW50SW5wdXRNZXRhZGF0YSB9IGZyb20gJy4vdHlwZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtY29tbWVudC1pbnB1dCcsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUsIElvbkNoZWNrYm94LCBJb25UZXh0YXJlYV0sXG4gIHRlbXBsYXRlOiBgXG4gICAgPGlvbi10ZXh0YXJlYVxuICAgICAgW2Zvcm1Db250cm9sXT1cInByb3BzLmNvbnRyb2xcIlxuICAgICAgW2NvdW50ZXJdPVwicHJvcHMuc2hvd0NvdW50ZXIgIT09IGZhbHNlXCJcbiAgICAgIFttYXhsZW5ndGhdPVwicHJvcHMubWF4TGVuZ3RoXCJcbiAgICA+PC9pb24tdGV4dGFyZWE+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL2NvbW1lbnQtaW5wdXQuY29tcG9uZW50LnNjc3MnXSxcbn0pXG4vKipcbiAqIHZhbC1jb21tZW50LWlucHV0XG4gKlxuICogQSB0ZXh0YXJlYSBpbnB1dCBmb3IgbXVsdGktbGluZSBjb21tZW50cywgaW50ZWdyYXRlZCB3aXRoIEFuZ3VsYXIgZm9ybXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIDx2YWwtY29tbWVudC1pbnB1dCBbcHJvcHNdPVwieyBjb250cm9sOiBteUNvbnRyb2wsIG1heExlbmd0aDogMjAwIH1cIj48L3ZhbC1jb21tZW50LWlucHV0PlxuICpcbiAqIEBpbnB1dCBwcm9wczogQ29tbWVudElucHV0TWV0YWRhdGEgLSBDb25maWd1cmF0aW9uIGZvciB0aGUgdGV4dGFyZWEgKGZvcm0gY29udHJvbCwgbWF4TGVuZ3RoLCBldGMuKVxuICovXG5leHBvcnQgY2xhc3MgQ29tbWVudElucHV0Q29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcbiAgLyoqXG4gICAqIElucHV0IGNvbmZpZ3VyYXRpb24gb2JqZWN0LlxuICAgKiBAdHlwZSB7Q29tbWVudElucHV0TWV0YWRhdGF9XG4gICAqIEBwcm9wZXJ0eSBjb250cm9sIC0gVGhlIEFuZ3VsYXIgRm9ybUNvbnRyb2wgZm9yIHRoZSB0ZXh0YXJlYS5cbiAgICogQHByb3BlcnR5IG1heExlbmd0aCAtIE1heGltdW0gY2hhcmFjdGVyIGxlbmd0aC5cbiAgICovXG4gIEBJbnB1dCgpIHByb3BzOiBDb21tZW50SW5wdXRNZXRhZGF0YTtcblxuICBjb25zdHJ1Y3RvcigpIHt9XG5cbiAgbmdPbkluaXQoKSB7fVxufVxuIl19
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2NvbW1lbnQtaW5wdXQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEZvcm1Db250cm9sIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgQ29sb3IgfSBmcm9tICdAaW9uaWMvY29yZSc7XG5pbXBvcnQgeyBDb21wb25lbnRTdGF0ZSB9IGZyb20gJy4uLy4uL3R5cGVzJztcblxuLyoqXG4gKiBNZXRhZGF0YSBmb3IgdGhlIGNvbW1lbnQgaW5wdXQgY29tcG9uZW50LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvbW1lbnRJbnB1dE1ldGFkYXRhIHtcbiAgLyoqIEZvcm0gY29udHJvbCBmb3IgdGhlIGNvbW1lbnQgdmFsdWUgKi9cbiAgY29udHJvbDogRm9ybUNvbnRyb2w8c3RyaW5nIHwgbnVsbD47XG4gIC8qKiBVbmlxdWUgdG9rZW4gZm9yIHRoZSBpbnB1dCAqL1xuICB0b2tlbj86IHN0cmluZztcbiAgLyoqIERpc3BsYXkgbGFiZWwgKi9cbiAgbGFiZWw/OiBzdHJpbmc7XG4gIC8qKiBGaWVsZCBuYW1lICovXG4gIG5hbWU/OiBzdHJpbmc7XG4gIC8qKiBIZWxwIHRleHQgKi9cbiAgaGludD86IHN0cmluZztcbiAgLyoqIFBsYWNlaG9sZGVyIHRleHQgKi9cbiAgcGxhY2Vob2xkZXI/OiBzdHJpbmc7XG4gIC8qKiBGaWVsZCBzdGF0ZSAqL1xuICBzdGF0ZT86IENvbXBvbmVudFN0YXRlO1xuXG4gIC8vID09PSBUZXh0YXJlYSBjb25maWd1cmF0aW9uID09PVxuICAvKiogTWF4aW11bSBjaGFyYWN0ZXIgbGVuZ3RoICovXG4gIG1heExlbmd0aD86IG51bWJlcjtcbiAgLyoqIE1pbmltdW0gY2hhcmFjdGVyIGxlbmd0aCAqL1xuICBtaW5MZW5ndGg/OiBudW1iZXI7XG4gIC8qKiBOdW1iZXIgb2YgdmlzaWJsZSByb3dzICovXG4gIHJvd3M/OiBudW1iZXI7XG4gIC8qKiBTaG93IGNoYXJhY3RlciBjb3VudGVyICovXG4gIHNob3dDb3VudGVyPzogYm9vbGVhbjtcbiAgLyoqIENvdW50ZXIgZm9ybWF0IChlLmcuLCAne2N1cnJlbnR9L3ttYXh9JykgKi9cbiAgY291bnRlckZvcm1hdD86IHN0cmluZztcblxuICAvLyA9PT0gU3R5bGluZyA9PT1cbiAgLyoqIENvbXBvbmVudCBjb2xvciAqL1xuICBjb2xvcj86IENvbG9yO1xuICAvKiogQ3VzdG9tIENTUyBjbGFzcyAqL1xuICBjc3NDbGFzcz86IHN0cmluZztcbiAgLyoqIEZpbGwgc3R5bGUgKi9cbiAgZmlsbD86ICdzb2xpZCcgfCAnb3V0bGluZSc7XG5cbiAgLy8gPT09IFJlYWN0aXZlIGNvbnRlbnQgPT09XG4gIC8qKiBDb250ZW50IGtleSBmb3IgcmVhY3RpdmUgbGFiZWwgKi9cbiAgY29udGVudEtleT86IHN0cmluZztcbiAgLyoqIENvbXBvbmVudCBjbGFzcyBuYW1lIGZvciBjb250ZW50IGxvb2t1cCAqL1xuICBjb250ZW50Q2xhc3M/OiBzdHJpbmc7XG4gIC8qKiBGYWxsYmFjayB0ZXh0IGlmIGNvbnRlbnQga2V5IGlzIG5vdCBmb3VuZCAqL1xuICBjb250ZW50RmFsbGJhY2s/OiBzdHJpbmc7XG5cbiAgLy8gPT09IFZhbGlkYXRpb24gZGlzcGxheSA9PT1cbiAgLyoqIEN1c3RvbSBlcnJvciBtZXNzYWdlcyAqL1xuICBlcnJvcnM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICAvKiogU2hvdyB2YWxpZGF0aW9uIGVycm9ycyAqL1xuICBzaG93RXJyb3JzPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBFdmVudCBlbWl0dGVkIHdoZW4gY29tbWVudCBpbnB1dCBjaGFuZ2VzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvbW1lbnRJbnB1dENoYW5nZUV2ZW50IHtcbiAgLyoqIEN1cnJlbnQgY29tbWVudCB2YWx1ZSAqL1xuICB2YWx1ZTogc3RyaW5nO1xuICAvKiogQ3VycmVudCBjaGFyYWN0ZXIgY291bnQgKi9cbiAgbGVuZ3RoOiBudW1iZXI7XG4gIC8qKiBSZW1haW5pbmcgY2hhcmFjdGVycyAoaWYgbWF4TGVuZ3RoIGlzIHNldCkgKi9cbiAgcmVtYWluaW5nOiBudW1iZXIgfCBudWxsO1xufVxuIl19
|
|
@@ -2,7 +2,6 @@ import { CommonModule } from '@angular/common';
|
|
|
2
2
|
import { Component, Input } from '@angular/core';
|
|
3
3
|
import { ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { IonDatetime, IonDatetimeButton, IonModal } from '@ionic/angular/standalone';
|
|
5
|
-
import { applyDefaultValueToControl } from '../../../shared/utils/form-defaults';
|
|
6
5
|
import * as i0 from "@angular/core";
|
|
7
6
|
import * as i1 from "@angular/forms";
|
|
8
7
|
/**
|
|
@@ -13,14 +12,34 @@ import * as i1 from "@angular/forms";
|
|
|
13
12
|
* @example
|
|
14
13
|
* <val-date-input [props]="{ control: myControl, hint: 'Select a date' }"></val-date-input>
|
|
15
14
|
*
|
|
16
|
-
* @input props:
|
|
15
|
+
* @input props: DateInputMetadata - Configuration for the date input (form control, hint, etc.)
|
|
17
16
|
*/
|
|
18
17
|
export class DateInputComponent {
|
|
19
18
|
constructor() { }
|
|
20
19
|
ngOnInit() {
|
|
21
20
|
// Apply default values on initialization
|
|
22
21
|
if (this.props?.withDefault || this.props?.value) {
|
|
23
|
-
|
|
22
|
+
this.applyDefaultValue();
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
applyDefaultValue() {
|
|
26
|
+
if (!this.props.control)
|
|
27
|
+
return;
|
|
28
|
+
let defaultValue = null;
|
|
29
|
+
// Explicit value takes precedence
|
|
30
|
+
if (this.props.value) {
|
|
31
|
+
defaultValue = new Date(this.props.value).toISOString();
|
|
32
|
+
}
|
|
33
|
+
else if (typeof this.props.withDefault === 'string') {
|
|
34
|
+
defaultValue = new Date(this.props.withDefault).toISOString();
|
|
35
|
+
}
|
|
36
|
+
else if (this.props.withDefault === true) {
|
|
37
|
+
defaultValue = new Date().toISOString();
|
|
38
|
+
}
|
|
39
|
+
if (defaultValue) {
|
|
40
|
+
this.props.control.setValue(defaultValue);
|
|
41
|
+
this.props.control.markAsPristine();
|
|
42
|
+
this.props.control.updateValueAndValidity();
|
|
24
43
|
}
|
|
25
44
|
}
|
|
26
45
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DateInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
@@ -80,4 +99,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
80
99
|
}], ctorParameters: () => [], propDecorators: { props: [{
|
|
81
100
|
type: Input
|
|
82
101
|
}] } });
|
|
83
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0ZS1pbnB1dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2RhdGUtaW5wdXQvZGF0ZS1pbnB1dC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ3pELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3JELE9BQU8sRUFBRSxXQUFXLEVBQUUsaUJBQWlCLEVBQUUsUUFBUSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7OztBQWtDckY7Ozs7Ozs7OztHQVNHO0FBQ0gsTUFBTSxPQUFPLGtCQUFrQjtJQVM3QixnQkFBZSxDQUFDO0lBRWhCLFFBQVE7UUFDTix5Q0FBeUM7UUFDekMsSUFBSSxJQUFJLENBQUMsS0FBSyxFQUFFLFdBQVcsSUFBSSxJQUFJLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxDQUFDO1lBQ2pELElBQUksQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO1FBQzNCLENBQUM7SUFDSCxDQUFDO0lBRU8saUJBQWlCO1FBQ3ZCLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU87WUFBRSxPQUFPO1FBRWhDLElBQUksWUFBWSxHQUFrQixJQUFJLENBQUM7UUFFdkMsa0NBQWtDO1FBQ2xDLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNyQixZQUFZLEdBQUcsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUMxRCxDQUFDO2FBQU0sSUFBSSxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQ3RELFlBQVksR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQ2hFLENBQUM7YUFBTSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxLQUFLLElBQUksRUFBRSxDQUFDO1lBQzNDLFlBQVksR0FBRyxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzFDLENBQUM7UUFFRCxJQUFJLFlBQVksRUFBRSxDQUFDO1lBQ2pCLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxZQUFZLENBQUMsQ0FBQztZQUMxQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxjQUFjLEVBQUUsQ0FBQztZQUNwQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxzQkFBc0IsRUFBRSxDQUFDO1FBQzlDLENBQUM7SUFDSCxDQUFDOytHQXJDVSxrQkFBa0I7bUdBQWxCLGtCQUFrQixzR0FyQ25COzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F3QlQsc3hGQXpCUyxZQUFZLDhCQUFFLG1CQUFtQiwwVEFBRSxXQUFXLG9oQkFBRSxpQkFBaUIsbUhBQUUsUUFBUTs7NEZBc0MxRSxrQkFBa0I7a0JBekM5QixTQUFTOytCQUNFLGdCQUFnQixjQUNkLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxtQkFBbUIsRUFBRSxXQUFXLEVBQUUsaUJBQWlCLEVBQUUsUUFBUSxDQUFDLFlBQzVFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F3QlQ7d0RBb0JRLEtBQUs7c0JBQWIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPbkluaXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBJb25EYXRldGltZSwgSW9uRGF0ZXRpbWVCdXR0b24sIElvbk1vZGFsIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5pbXBvcnQgeyBEYXRlSW5wdXRNZXRhZGF0YSB9IGZyb20gJy4vdHlwZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtZGF0ZS1pbnB1dCcsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUsIElvbkRhdGV0aW1lLCBJb25EYXRldGltZUJ1dHRvbiwgSW9uTW9kYWxdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXYgY2xhc3M9XCJidXR0b24tY29udGFpbmVyXCI+XG4gICAgICA8aW9uLWRhdGV0aW1lLWJ1dHRvbiBjbGFzcz1cImFjdGlvblwiIFtkYXRldGltZV09XCJwcm9wcy50b2tlblwiPjwvaW9uLWRhdGV0aW1lLWJ1dHRvbj5cbiAgICA8L2Rpdj5cbiAgICA8aW9uLW1vZGFsIFtrZWVwQ29udGVudHNNb3VudGVkXT1cInRydWVcIj5cbiAgICAgIDxuZy10ZW1wbGF0ZT5cbiAgICAgICAgPGlvbi1kYXRldGltZVxuICAgICAgICAgIFtmb3JtQ29udHJvbF09XCJwcm9wcy5jb250cm9sXCJcbiAgICAgICAgICBbaWRdPVwicHJvcHMudG9rZW5cIlxuICAgICAgICAgIHByZXNlbnRhdGlvbj1cImRhdGVcIlxuICAgICAgICAgIGxvY2FsZT1cImVzLUVTXCJcbiAgICAgICAgICBbZmlyc3REYXlPZldlZWtdPVwiMVwiXG4gICAgICAgICAgW3Nob3dEZWZhdWx0QnV0dG9uc109XCJ0cnVlXCJcbiAgICAgICAgICBkb25lVGV4dD1cIkFjZXB0YXJcIlxuICAgICAgICAgIGNhbmNlbFRleHQ9XCJDYW5jZWxhclwiXG4gICAgICAgICAgZm9ybWF0T3B0aW9ucz1cIntcbiAgICAgICAgICAgIGRhdGU6IHsgZGF0ZVN0eWxlOiAnbWVkaXVtJyB9LFxuICAgICAgICAgICAgdGltZTogeyB0aW1lU3R5bGU6ICdzaG9ydCcgfVxuICAgICAgICAgIH1cIlxuICAgICAgICA+XG4gICAgICAgICAgPHNwYW4gc2xvdD1cInRpdGxlXCI+e3sgcHJvcHMuaGludCB9fTwvc3Bhbj5cbiAgICAgICAgPC9pb24tZGF0ZXRpbWU+XG4gICAgICA8L25nLXRlbXBsYXRlPlxuICAgIDwvaW9uLW1vZGFsPlxuICBgLFxuICBzdHlsZVVybHM6IFsnLi9kYXRlLWlucHV0LmNvbXBvbmVudC5zY3NzJ10sXG59KVxuLyoqXG4gKiB2YWwtZGF0ZS1pbnB1dFxuICpcbiAqIEEgZGF0ZSBwaWNrZXIgaW5wdXQgaW50ZWdyYXRlZCB3aXRoIEFuZ3VsYXIgZm9ybXMsIHVzaW5nIElvbmljJ3MgZGF0ZXRpbWUgY29tcG9uZW50LlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLWRhdGUtaW5wdXQgW3Byb3BzXT1cInsgY29udHJvbDogbXlDb250cm9sLCBoaW50OiAnU2VsZWN0IGEgZGF0ZScgfVwiPjwvdmFsLWRhdGUtaW5wdXQ+XG4gKlxuICogQGlucHV0IHByb3BzOiBEYXRlSW5wdXRNZXRhZGF0YSAtIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBkYXRlIGlucHV0IChmb3JtIGNvbnRyb2wsIGhpbnQsIGV0Yy4pXG4gKi9cbmV4cG9ydCBjbGFzcyBEYXRlSW5wdXRDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICAvKipcbiAgICogSW5wdXQgY29uZmlndXJhdGlvbiBvYmplY3QuXG4gICAqIEB0eXBlIHtEYXRlSW5wdXRNZXRhZGF0YX1cbiAgICogQHByb3BlcnR5IGNvbnRyb2wgLSBUaGUgQW5ndWxhciBGb3JtQ29udHJvbCBmb3IgdGhlIGRhdGUgaW5wdXQuXG4gICAqIEBwcm9wZXJ0eSBoaW50IC0gVGhlIGhpbnQgdGV4dCBmb3IgdGhlIGlucHV0LlxuICAgKi9cbiAgQElucHV0KCkgcHJvcHM6IERhdGVJbnB1dE1ldGFkYXRhO1xuXG4gIGNvbnN0cnVjdG9yKCkge31cblxuICBuZ09uSW5pdCgpIHtcbiAgICAvLyBBcHBseSBkZWZhdWx0IHZhbHVlcyBvbiBpbml0aWFsaXphdGlvblxuICAgIGlmICh0aGlzLnByb3BzPy53aXRoRGVmYXVsdCB8fCB0aGlzLnByb3BzPy52YWx1ZSkge1xuICAgICAgdGhpcy5hcHBseURlZmF1bHRWYWx1ZSgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYXBwbHlEZWZhdWx0VmFsdWUoKTogdm9pZCB7XG4gICAgaWYgKCF0aGlzLnByb3BzLmNvbnRyb2wpIHJldHVybjtcblxuICAgIGxldCBkZWZhdWx0VmFsdWU6IHN0cmluZyB8IG51bGwgPSBudWxsO1xuXG4gICAgLy8gRXhwbGljaXQgdmFsdWUgdGFrZXMgcHJlY2VkZW5jZVxuICAgIGlmICh0aGlzLnByb3BzLnZhbHVlKSB7XG4gICAgICBkZWZhdWx0VmFsdWUgPSBuZXcgRGF0ZSh0aGlzLnByb3BzLnZhbHVlKS50b0lTT1N0cmluZygpO1xuICAgIH0gZWxzZSBpZiAodHlwZW9mIHRoaXMucHJvcHMud2l0aERlZmF1bHQgPT09ICdzdHJpbmcnKSB7XG4gICAgICBkZWZhdWx0VmFsdWUgPSBuZXcgRGF0ZSh0aGlzLnByb3BzLndpdGhEZWZhdWx0KS50b0lTT1N0cmluZygpO1xuICAgIH0gZWxzZSBpZiAodGhpcy5wcm9wcy53aXRoRGVmYXVsdCA9PT0gdHJ1ZSkge1xuICAgICAgZGVmYXVsdFZhbHVlID0gbmV3IERhdGUoKS50b0lTT1N0cmluZygpO1xuICAgIH1cblxuICAgIGlmIChkZWZhdWx0VmFsdWUpIHtcbiAgICAgIHRoaXMucHJvcHMuY29udHJvbC5zZXRWYWx1ZShkZWZhdWx0VmFsdWUpO1xuICAgICAgdGhpcy5wcm9wcy5jb250cm9sLm1hcmtBc1ByaXN0aW5lKCk7XG4gICAgICB0aGlzLnByb3BzLmNvbnRyb2wudXBkYXRlVmFsdWVBbmRWYWxpZGl0eSgpO1xuICAgIH1cbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2RhdGUtaW5wdXQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEZvcm1Db250cm9sIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgQ29sb3IgfSBmcm9tICdAaW9uaWMvY29yZSc7XG5pbXBvcnQgeyBDb21wb25lbnRTdGF0ZSB9IGZyb20gJy4uLy4uL3R5cGVzJztcblxuLyoqXG4gKiBNZXRhZGF0YSBmb3IgdGhlIGRhdGUgaW5wdXQgY29tcG9uZW50LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIERhdGVJbnB1dE1ldGFkYXRhIHtcbiAgLyoqIEZvcm0gY29udHJvbCBmb3IgdGhlIGRhdGUgdmFsdWUgKElTTyBzdHJpbmcpICovXG4gIGNvbnRyb2w6IEZvcm1Db250cm9sPHN0cmluZyB8IG51bGw+O1xuICAvKiogVW5pcXVlIHRva2VuIGZvciB0aGUgaW5wdXQgKi9cbiAgdG9rZW4/OiBzdHJpbmc7XG4gIC8qKiBEaXNwbGF5IGxhYmVsICovXG4gIGxhYmVsPzogc3RyaW5nO1xuICAvKiogRmllbGQgbmFtZSAqL1xuICBuYW1lPzogc3RyaW5nO1xuICAvKiogSGVscCB0ZXh0IC8gdGl0bGUgZGlzcGxheWVkIGluIG1vZGFsICovXG4gIGhpbnQ/OiBzdHJpbmc7XG4gIC8qKiBQbGFjZWhvbGRlciB0ZXh0ICovXG4gIHBsYWNlaG9sZGVyPzogc3RyaW5nO1xuICAvKiogRmllbGQgc3RhdGUgKi9cbiAgc3RhdGU/OiBDb21wb25lbnRTdGF0ZTtcblxuICAvLyA9PT0gRGF0ZSBjb25maWd1cmF0aW9uID09PVxuICAvKiogTWluaW11bSBzZWxlY3RhYmxlIGRhdGUgKElTTyBzdHJpbmcpICovXG4gIG1pbj86IHN0cmluZztcbiAgLyoqIE1heGltdW0gc2VsZWN0YWJsZSBkYXRlIChJU08gc3RyaW5nKSAqL1xuICBtYXg/OiBzdHJpbmc7XG4gIC8qKiBMb2NhbGUgZm9yIGRhdGUgZGlzcGxheSAoZS5nLiwgJ2VzLUVTJykgKi9cbiAgbG9jYWxlPzogc3RyaW5nO1xuICAvKiogRmlyc3QgZGF5IG9mIHdlZWsgKDAgPSBTdW5kYXksIDEgPSBNb25kYXkpICovXG4gIGZpcnN0RGF5T2ZXZWVrPzogbnVtYmVyO1xuICAvKiogRGF0ZSBwcmVzZW50YXRpb24gZm9ybWF0ICovXG4gIHByZXNlbnRhdGlvbj86ICdkYXRlJyB8ICdkYXRlLXRpbWUnIHwgJ21vbnRoJyB8ICdtb250aC15ZWFyJyB8ICd5ZWFyJyB8ICd0aW1lJztcbiAgLyoqIEN1c3RvbSBmb3JtYXQgb3B0aW9ucyAqL1xuICBmb3JtYXRPcHRpb25zPzoge1xuICAgIGRhdGU/OiB7IGRhdGVTdHlsZT86ICdmdWxsJyB8ICdsb25nJyB8ICdtZWRpdW0nIHwgJ3Nob3J0JyB9O1xuICAgIHRpbWU/OiB7IHRpbWVTdHlsZT86ICdmdWxsJyB8ICdsb25nJyB8ICdtZWRpdW0nIHwgJ3Nob3J0JyB9O1xuICB9O1xuXG4gIC8vID09PSBNb2RhbCBjb25maWd1cmF0aW9uID09PVxuICAvKiogRG9uZSBidXR0b24gdGV4dCAqL1xuICBkb25lVGV4dD86IHN0cmluZztcbiAgLyoqIENhbmNlbCBidXR0b24gdGV4dCAqL1xuICBjYW5jZWxUZXh0Pzogc3RyaW5nO1xuICAvKiogU2hvdyBkZWZhdWx0IGJ1dHRvbnMgKi9cbiAgc2hvd0RlZmF1bHRCdXR0b25zPzogYm9vbGVhbjtcblxuICAvLyA9PT0gU3R5bGluZyA9PT1cbiAgLyoqIENvbXBvbmVudCBjb2xvciAqL1xuICBjb2xvcj86IENvbG9yO1xuICAvKiogQ3VzdG9tIENTUyBjbGFzcyAqL1xuICBjc3NDbGFzcz86IHN0cmluZztcblxuICAvLyA9PT0gRGVmYXVsdCB2YWx1ZSA9PT1cbiAgLyoqIEluaXRpYWwgdmFsdWUgZm9yIHRoZSBmaWVsZCAqL1xuICB2YWx1ZT86IHN0cmluZztcbiAgLyoqIERlZmF1bHQgdmFsdWUgY29uZmlndXJhdGlvbiAtIHN0cmluZyBmb3IgY3VzdG9tIGRlZmF1bHRzLCB0cnVlIGZvciBhdXRvIGRlZmF1bHRzICovXG4gIHdpdGhEZWZhdWx0Pzogc3RyaW5nIHwgYm9vbGVhbjtcblxuICAvLyA9PT0gUmVhY3RpdmUgY29udGVudCA9PT1cbiAgLyoqIENvbnRlbnQga2V5IGZvciByZWFjdGl2ZSBsYWJlbCAqL1xuICBjb250ZW50S2V5Pzogc3RyaW5nO1xuICAvKiogQ29tcG9uZW50IGNsYXNzIG5hbWUgZm9yIGNvbnRlbnQgbG9va3VwICovXG4gIGNvbnRlbnRDbGFzcz86IHN0cmluZztcbiAgLyoqIEZhbGxiYWNrIHRleHQgaWYgY29udGVudCBrZXkgaXMgbm90IGZvdW5kICovXG4gIGNvbnRlbnRGYWxsYmFjaz86IHN0cmluZztcblxuICAvLyA9PT0gVmFsaWRhdGlvbiBkaXNwbGF5ID09PVxuICAvKiogQ3VzdG9tIGVycm9yIG1lc3NhZ2VzICovXG4gIGVycm9ycz86IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG4gIC8qKiBTaG93IHZhbGlkYXRpb24gZXJyb3JzICovXG4gIHNob3dFcnJvcnM/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIEV2ZW50IGVtaXR0ZWQgd2hlbiBkYXRlIGNoYW5nZXMuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRGF0ZUlucHV0Q2hhbmdlRXZlbnQge1xuICAvKiogU2VsZWN0ZWQgZGF0ZSB2YWx1ZSAoSVNPIHN0cmluZykgKi9cbiAgdmFsdWU6IHN0cmluZyB8IG51bGw7XG4gIC8qKiBGb3JtYXR0ZWQgZGF0ZSBzdHJpbmcgZm9yIGRpc3BsYXkgKi9cbiAgZm9ybWF0dGVkVmFsdWU6IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -0,0 +1,352 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, EventEmitter, Input, Output, inject, signal, } from '@angular/core';
|
|
3
|
+
import { FormBuilder, ReactiveFormsModule, Validators, } from '@angular/forms';
|
|
4
|
+
import { IonButton, IonIcon, IonItem, IonLabel, IonList, IonNote, IonSelect, IonSelectOption, IonSpinner, IonText, IonTextarea, } from '@ionic/angular/standalone';
|
|
5
|
+
import { addIcons } from 'ionicons';
|
|
6
|
+
import { bugOutline, bulbOutline, chatbubbleOutline, checkmarkCircleOutline, closeCircleOutline, documentTextOutline, } from 'ionicons/icons';
|
|
7
|
+
import { FeedbackService, DEFAULT_FEEDBACK_TYPE_OPTIONS, } from '../../../services/feedback';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/common";
|
|
10
|
+
import * as i2 from "@angular/forms";
|
|
11
|
+
/**
|
|
12
|
+
* val-feedback-form
|
|
13
|
+
*
|
|
14
|
+
* Formulario reutilizable para enviar feedback desde cualquier parte de la aplicación.
|
|
15
|
+
*
|
|
16
|
+
* @example
|
|
17
|
+
* ```html
|
|
18
|
+
* <!-- Feedback general -->
|
|
19
|
+
* <val-feedback-form
|
|
20
|
+
* [props]="{ defaultType: 'feedback', showTypeSelector: true }"
|
|
21
|
+
* (onSubmit)="handleSuccess($event)"
|
|
22
|
+
* (onCancel)="closeModal()"
|
|
23
|
+
* />
|
|
24
|
+
*
|
|
25
|
+
* <!-- Reportar contenido incorrecto -->
|
|
26
|
+
* <val-feedback-form
|
|
27
|
+
* [props]="{
|
|
28
|
+
* defaultType: 'poor-content',
|
|
29
|
+
* showTypeSelector: false,
|
|
30
|
+
* contentRef: { contentId: article.id, contentType: 'article' },
|
|
31
|
+
* submitButtonText: 'Reportar contenido'
|
|
32
|
+
* }"
|
|
33
|
+
* />
|
|
34
|
+
* ```
|
|
35
|
+
*/
|
|
36
|
+
export class FeedbackFormComponent {
|
|
37
|
+
constructor() {
|
|
38
|
+
/**
|
|
39
|
+
* Configuración del formulario.
|
|
40
|
+
*/
|
|
41
|
+
this.props = {};
|
|
42
|
+
/**
|
|
43
|
+
* Evento emitido cuando el feedback se envía exitosamente.
|
|
44
|
+
*/
|
|
45
|
+
this.onSubmit = new EventEmitter();
|
|
46
|
+
/**
|
|
47
|
+
* Evento emitido cuando el usuario cancela.
|
|
48
|
+
*/
|
|
49
|
+
this.onCancel = new EventEmitter();
|
|
50
|
+
this.fb = inject(FormBuilder);
|
|
51
|
+
this.feedbackService = inject(FeedbackService);
|
|
52
|
+
this.typeOptions = DEFAULT_FEEDBACK_TYPE_OPTIONS;
|
|
53
|
+
this.isSubmitting = signal(false);
|
|
54
|
+
this.isSuccess = signal(false);
|
|
55
|
+
this.error = signal(null);
|
|
56
|
+
addIcons({
|
|
57
|
+
bugOutline,
|
|
58
|
+
bulbOutline,
|
|
59
|
+
chatbubbleOutline,
|
|
60
|
+
checkmarkCircleOutline,
|
|
61
|
+
closeCircleOutline,
|
|
62
|
+
documentTextOutline,
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
ngOnInit() {
|
|
66
|
+
// Filtrar tipos habilitados si se especifica
|
|
67
|
+
if (this.props.enabledTypes?.length) {
|
|
68
|
+
this.typeOptions = this.typeOptions.filter((opt) => this.props.enabledTypes.includes(opt.value));
|
|
69
|
+
}
|
|
70
|
+
// Usar opciones personalizadas si se proporcionan
|
|
71
|
+
if (this.props.typeOptions?.length) {
|
|
72
|
+
this.typeOptions = this.props.typeOptions;
|
|
73
|
+
}
|
|
74
|
+
// Inicializar formulario
|
|
75
|
+
this.form = this.fb.group({
|
|
76
|
+
type: [this.props.defaultType || 'feedback', Validators.required],
|
|
77
|
+
title: [
|
|
78
|
+
'',
|
|
79
|
+
[Validators.required, Validators.minLength(5), Validators.maxLength(200)],
|
|
80
|
+
],
|
|
81
|
+
description: [
|
|
82
|
+
'',
|
|
83
|
+
[Validators.required, Validators.minLength(10), Validators.maxLength(5000)],
|
|
84
|
+
],
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
async handleSubmit() {
|
|
88
|
+
if (this.form.invalid || this.isSubmitting())
|
|
89
|
+
return;
|
|
90
|
+
this.isSubmitting.set(true);
|
|
91
|
+
this.error.set(null);
|
|
92
|
+
this.isSuccess.set(false);
|
|
93
|
+
try {
|
|
94
|
+
const { type, title, description } = this.form.value;
|
|
95
|
+
const response = await this.feedbackService.createAsync(type, title, description, [], // attachments (por ahora vacío)
|
|
96
|
+
this.props.contentRef);
|
|
97
|
+
this.isSuccess.set(true);
|
|
98
|
+
this.form.reset({ type: this.props.defaultType || 'feedback' });
|
|
99
|
+
this.onSubmit.emit({
|
|
100
|
+
response,
|
|
101
|
+
type: type,
|
|
102
|
+
title,
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
catch (err) {
|
|
106
|
+
this.error.set(err.error?.message || err.message || 'Error al enviar el feedback');
|
|
107
|
+
}
|
|
108
|
+
finally {
|
|
109
|
+
this.isSubmitting.set(false);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
onCancelClick() {
|
|
113
|
+
this.onCancel.emit();
|
|
114
|
+
}
|
|
115
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
116
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: FeedbackFormComponent, isStandalone: true, selector: "val-feedback-form", inputs: { props: "props" }, outputs: { onSubmit: "onSubmit", onCancel: "onCancel" }, ngImport: i0, template: `
|
|
117
|
+
<form
|
|
118
|
+
[formGroup]="form"
|
|
119
|
+
(ngSubmit)="handleSubmit()"
|
|
120
|
+
class="feedback-form"
|
|
121
|
+
[class.compact]="props.compact"
|
|
122
|
+
[ngClass]="props.cssClass"
|
|
123
|
+
>
|
|
124
|
+
<!-- Type selector -->
|
|
125
|
+
@if (props.showTypeSelector !== false) {
|
|
126
|
+
<ion-item>
|
|
127
|
+
<ion-select
|
|
128
|
+
formControlName="type"
|
|
129
|
+
label="Tipo de feedback"
|
|
130
|
+
labelPlacement="floating"
|
|
131
|
+
interface="popover"
|
|
132
|
+
>
|
|
133
|
+
@for (option of typeOptions; track option.value) {
|
|
134
|
+
<ion-select-option [value]="option.value">
|
|
135
|
+
{{ option.label }}
|
|
136
|
+
</ion-select-option>
|
|
137
|
+
}
|
|
138
|
+
</ion-select>
|
|
139
|
+
</ion-item>
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
<!-- Title -->
|
|
143
|
+
<ion-item>
|
|
144
|
+
<ion-textarea
|
|
145
|
+
formControlName="title"
|
|
146
|
+
[label]="props.titleLabel || 'Título'"
|
|
147
|
+
labelPlacement="floating"
|
|
148
|
+
[placeholder]="props.titlePlaceholder || 'Describe brevemente...'"
|
|
149
|
+
[maxlength]="200"
|
|
150
|
+
[counter]="true"
|
|
151
|
+
[autoGrow]="false"
|
|
152
|
+
rows="1"
|
|
153
|
+
></ion-textarea>
|
|
154
|
+
</ion-item>
|
|
155
|
+
@if (form.get('title')?.invalid && form.get('title')?.touched) {
|
|
156
|
+
<ion-note color="danger" class="ion-padding-start">
|
|
157
|
+
El título debe tener entre 5 y 200 caracteres
|
|
158
|
+
</ion-note>
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
<!-- Description -->
|
|
162
|
+
<ion-item>
|
|
163
|
+
<ion-textarea
|
|
164
|
+
formControlName="description"
|
|
165
|
+
[label]="props.descriptionLabel || 'Descripción'"
|
|
166
|
+
labelPlacement="floating"
|
|
167
|
+
[placeholder]="props.descriptionPlaceholder || 'Proporciona más detalles...'"
|
|
168
|
+
[maxlength]="5000"
|
|
169
|
+
[counter]="true"
|
|
170
|
+
[autoGrow]="true"
|
|
171
|
+
rows="4"
|
|
172
|
+
></ion-textarea>
|
|
173
|
+
</ion-item>
|
|
174
|
+
@if (form.get('description')?.invalid && form.get('description')?.touched) {
|
|
175
|
+
<ion-note color="danger" class="ion-padding-start">
|
|
176
|
+
La descripción debe tener entre 10 y 5000 caracteres
|
|
177
|
+
</ion-note>
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
<!-- Error message -->
|
|
181
|
+
@if (error()) {
|
|
182
|
+
<div class="feedback-alert error">
|
|
183
|
+
<ion-icon name="close-circle-outline"></ion-icon>
|
|
184
|
+
<span>{{ error() }}</span>
|
|
185
|
+
</div>
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
<!-- Success message -->
|
|
189
|
+
@if (isSuccess()) {
|
|
190
|
+
<div class="feedback-alert success">
|
|
191
|
+
<ion-icon name="checkmark-circle-outline"></ion-icon>
|
|
192
|
+
<span>{{ props.successMessage || 'Feedback enviado exitosamente' }}</span>
|
|
193
|
+
</div>
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
<!-- Actions -->
|
|
197
|
+
<div class="form-actions">
|
|
198
|
+
@if (props.cancelButtonText) {
|
|
199
|
+
<ion-button
|
|
200
|
+
fill="outline"
|
|
201
|
+
color="medium"
|
|
202
|
+
type="button"
|
|
203
|
+
(click)="onCancelClick()"
|
|
204
|
+
>
|
|
205
|
+
{{ props.cancelButtonText }}
|
|
206
|
+
</ion-button>
|
|
207
|
+
}
|
|
208
|
+
<ion-button
|
|
209
|
+
type="submit"
|
|
210
|
+
[disabled]="form.invalid || isSubmitting()"
|
|
211
|
+
expand="block"
|
|
212
|
+
>
|
|
213
|
+
@if (isSubmitting()) {
|
|
214
|
+
<ion-spinner name="crescent"></ion-spinner>
|
|
215
|
+
} @else {
|
|
216
|
+
{{ props.submitButtonText || 'Enviar' }}
|
|
217
|
+
}
|
|
218
|
+
</ion-button>
|
|
219
|
+
</div>
|
|
220
|
+
</form>
|
|
221
|
+
`, isInline: true, styles: [".feedback-form{display:flex;flex-direction:column;gap:8px;&.compact{gap:4px}}.form-actions{display:flex;gap:8px;margin-top:16px;justify-content:flex-end;ion-button{flex:1}}.feedback-alert{display:flex;align-items:center;gap:8px;padding:12px 16px;border-radius:8px;margin-top:8px;&.error{background-color:var(--ion-color-danger-tint);color:var(--ion-color-danger-shade)}&.success{background-color:var(--ion-color-success-tint);color:var(--ion-color-success-shade)}ion-icon{font-size:20px}}ion-note{font-size:12px;margin-top:-4px;margin-bottom:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonNote, selector: "ion-note", inputs: ["color", "mode"] }, { kind: "component", type: IonSelect, selector: "ion-select", inputs: ["cancelText", "color", "compareWith", "disabled", "errorText", "expandedIcon", "fill", "helperText", "interface", "interfaceOptions", "justify", "label", "labelPlacement", "mode", "multiple", "name", "okText", "placeholder", "selectedText", "shape", "toggleIcon", "value"] }, { kind: "component", type: IonSelectOption, selector: "ion-select-option", inputs: ["disabled", "value"] }, { kind: "component", type: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: IonTextarea, selector: "ion-textarea", inputs: ["autoGrow", "autocapitalize", "autofocus", "clearOnEdit", "color", "cols", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "maxlength", "minlength", "mode", "name", "placeholder", "readonly", "required", "rows", "shape", "spellcheck", "value", "wrap"] }] }); }
|
|
222
|
+
}
|
|
223
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackFormComponent, decorators: [{
|
|
224
|
+
type: Component,
|
|
225
|
+
args: [{ selector: 'val-feedback-form', standalone: true, imports: [
|
|
226
|
+
CommonModule,
|
|
227
|
+
ReactiveFormsModule,
|
|
228
|
+
IonButton,
|
|
229
|
+
IonIcon,
|
|
230
|
+
IonItem,
|
|
231
|
+
IonLabel,
|
|
232
|
+
IonList,
|
|
233
|
+
IonNote,
|
|
234
|
+
IonSelect,
|
|
235
|
+
IonSelectOption,
|
|
236
|
+
IonSpinner,
|
|
237
|
+
IonText,
|
|
238
|
+
IonTextarea,
|
|
239
|
+
], template: `
|
|
240
|
+
<form
|
|
241
|
+
[formGroup]="form"
|
|
242
|
+
(ngSubmit)="handleSubmit()"
|
|
243
|
+
class="feedback-form"
|
|
244
|
+
[class.compact]="props.compact"
|
|
245
|
+
[ngClass]="props.cssClass"
|
|
246
|
+
>
|
|
247
|
+
<!-- Type selector -->
|
|
248
|
+
@if (props.showTypeSelector !== false) {
|
|
249
|
+
<ion-item>
|
|
250
|
+
<ion-select
|
|
251
|
+
formControlName="type"
|
|
252
|
+
label="Tipo de feedback"
|
|
253
|
+
labelPlacement="floating"
|
|
254
|
+
interface="popover"
|
|
255
|
+
>
|
|
256
|
+
@for (option of typeOptions; track option.value) {
|
|
257
|
+
<ion-select-option [value]="option.value">
|
|
258
|
+
{{ option.label }}
|
|
259
|
+
</ion-select-option>
|
|
260
|
+
}
|
|
261
|
+
</ion-select>
|
|
262
|
+
</ion-item>
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
<!-- Title -->
|
|
266
|
+
<ion-item>
|
|
267
|
+
<ion-textarea
|
|
268
|
+
formControlName="title"
|
|
269
|
+
[label]="props.titleLabel || 'Título'"
|
|
270
|
+
labelPlacement="floating"
|
|
271
|
+
[placeholder]="props.titlePlaceholder || 'Describe brevemente...'"
|
|
272
|
+
[maxlength]="200"
|
|
273
|
+
[counter]="true"
|
|
274
|
+
[autoGrow]="false"
|
|
275
|
+
rows="1"
|
|
276
|
+
></ion-textarea>
|
|
277
|
+
</ion-item>
|
|
278
|
+
@if (form.get('title')?.invalid && form.get('title')?.touched) {
|
|
279
|
+
<ion-note color="danger" class="ion-padding-start">
|
|
280
|
+
El título debe tener entre 5 y 200 caracteres
|
|
281
|
+
</ion-note>
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
<!-- Description -->
|
|
285
|
+
<ion-item>
|
|
286
|
+
<ion-textarea
|
|
287
|
+
formControlName="description"
|
|
288
|
+
[label]="props.descriptionLabel || 'Descripción'"
|
|
289
|
+
labelPlacement="floating"
|
|
290
|
+
[placeholder]="props.descriptionPlaceholder || 'Proporciona más detalles...'"
|
|
291
|
+
[maxlength]="5000"
|
|
292
|
+
[counter]="true"
|
|
293
|
+
[autoGrow]="true"
|
|
294
|
+
rows="4"
|
|
295
|
+
></ion-textarea>
|
|
296
|
+
</ion-item>
|
|
297
|
+
@if (form.get('description')?.invalid && form.get('description')?.touched) {
|
|
298
|
+
<ion-note color="danger" class="ion-padding-start">
|
|
299
|
+
La descripción debe tener entre 10 y 5000 caracteres
|
|
300
|
+
</ion-note>
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
<!-- Error message -->
|
|
304
|
+
@if (error()) {
|
|
305
|
+
<div class="feedback-alert error">
|
|
306
|
+
<ion-icon name="close-circle-outline"></ion-icon>
|
|
307
|
+
<span>{{ error() }}</span>
|
|
308
|
+
</div>
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
<!-- Success message -->
|
|
312
|
+
@if (isSuccess()) {
|
|
313
|
+
<div class="feedback-alert success">
|
|
314
|
+
<ion-icon name="checkmark-circle-outline"></ion-icon>
|
|
315
|
+
<span>{{ props.successMessage || 'Feedback enviado exitosamente' }}</span>
|
|
316
|
+
</div>
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
<!-- Actions -->
|
|
320
|
+
<div class="form-actions">
|
|
321
|
+
@if (props.cancelButtonText) {
|
|
322
|
+
<ion-button
|
|
323
|
+
fill="outline"
|
|
324
|
+
color="medium"
|
|
325
|
+
type="button"
|
|
326
|
+
(click)="onCancelClick()"
|
|
327
|
+
>
|
|
328
|
+
{{ props.cancelButtonText }}
|
|
329
|
+
</ion-button>
|
|
330
|
+
}
|
|
331
|
+
<ion-button
|
|
332
|
+
type="submit"
|
|
333
|
+
[disabled]="form.invalid || isSubmitting()"
|
|
334
|
+
expand="block"
|
|
335
|
+
>
|
|
336
|
+
@if (isSubmitting()) {
|
|
337
|
+
<ion-spinner name="crescent"></ion-spinner>
|
|
338
|
+
} @else {
|
|
339
|
+
{{ props.submitButtonText || 'Enviar' }}
|
|
340
|
+
}
|
|
341
|
+
</ion-button>
|
|
342
|
+
</div>
|
|
343
|
+
</form>
|
|
344
|
+
`, styles: [".feedback-form{display:flex;flex-direction:column;gap:8px;&.compact{gap:4px}}.form-actions{display:flex;gap:8px;margin-top:16px;justify-content:flex-end;ion-button{flex:1}}.feedback-alert{display:flex;align-items:center;gap:8px;padding:12px 16px;border-radius:8px;margin-top:8px;&.error{background-color:var(--ion-color-danger-tint);color:var(--ion-color-danger-shade)}&.success{background-color:var(--ion-color-success-tint);color:var(--ion-color-success-shade)}ion-icon{font-size:20px}}ion-note{font-size:12px;margin-top:-4px;margin-bottom:8px}\n"] }]
|
|
345
|
+
}], ctorParameters: () => [], propDecorators: { props: [{
|
|
346
|
+
type: Input
|
|
347
|
+
}], onSubmit: [{
|
|
348
|
+
type: Output
|
|
349
|
+
}], onCancel: [{
|
|
350
|
+
type: Output
|
|
351
|
+
}] } });
|
|
352
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feedback-form.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/feedback-form/feedback-form.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  EventEmitter,\n  Input,\n  OnInit,\n  Output,\n  inject,\n  signal,\n} from '@angular/core';\nimport {\n  FormBuilder,\n  FormGroup,\n  ReactiveFormsModule,\n  Validators,\n} from '@angular/forms';\nimport {\n  IonButton,\n  IonIcon,\n  IonItem,\n  IonLabel,\n  IonList,\n  IonNote,\n  IonSelect,\n  IonSelectOption,\n  IonSpinner,\n  IonText,\n  IonTextarea,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  bugOutline,\n  bulbOutline,\n  chatbubbleOutline,\n  checkmarkCircleOutline,\n  closeCircleOutline,\n  documentTextOutline,\n} from 'ionicons/icons';\n\nimport {\n  FeedbackService,\n  FeedbackType,\n  CreateFeedbackResponse,\n  DEFAULT_FEEDBACK_TYPE_OPTIONS,\n} from '../../../services/feedback';\nimport { FeedbackFormMetadata, FeedbackSubmitEvent } from './types';\n\n/**\n * val-feedback-form\n *\n * Formulario reutilizable para enviar feedback desde cualquier parte de la aplicación.\n *\n * @example\n * ```html\n * <!-- Feedback general -->\n * <val-feedback-form\n *   [props]=\"{ defaultType: 'feedback', showTypeSelector: true }\"\n *   (onSubmit)=\"handleSuccess($event)\"\n *   (onCancel)=\"closeModal()\"\n * />\n *\n * <!-- Reportar contenido incorrecto -->\n * <val-feedback-form\n *   [props]=\"{\n *     defaultType: 'poor-content',\n *     showTypeSelector: false,\n *     contentRef: { contentId: article.id, contentType: 'article' },\n *     submitButtonText: 'Reportar contenido'\n *   }\"\n * />\n * ```\n */\n@Component({\n  selector: 'val-feedback-form',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    IonButton,\n    IonIcon,\n    IonItem,\n    IonLabel,\n    IonList,\n    IonNote,\n    IonSelect,\n    IonSelectOption,\n    IonSpinner,\n    IonText,\n    IonTextarea,\n  ],\n  template: `\n    <form\n      [formGroup]=\"form\"\n      (ngSubmit)=\"handleSubmit()\"\n      class=\"feedback-form\"\n      [class.compact]=\"props.compact\"\n      [ngClass]=\"props.cssClass\"\n    >\n      <!-- Type selector -->\n      @if (props.showTypeSelector !== false) {\n        <ion-item>\n          <ion-select\n            formControlName=\"type\"\n            label=\"Tipo de feedback\"\n            labelPlacement=\"floating\"\n            interface=\"popover\"\n          >\n            @for (option of typeOptions; track option.value) {\n              <ion-select-option [value]=\"option.value\">\n                {{ option.label }}\n              </ion-select-option>\n            }\n          </ion-select>\n        </ion-item>\n      }\n\n      <!-- Title -->\n      <ion-item>\n        <ion-textarea\n          formControlName=\"title\"\n          [label]=\"props.titleLabel || 'Título'\"\n          labelPlacement=\"floating\"\n          [placeholder]=\"props.titlePlaceholder || 'Describe brevemente...'\"\n          [maxlength]=\"200\"\n          [counter]=\"true\"\n          [autoGrow]=\"false\"\n          rows=\"1\"\n        ></ion-textarea>\n      </ion-item>\n      @if (form.get('title')?.invalid && form.get('title')?.touched) {\n        <ion-note color=\"danger\" class=\"ion-padding-start\">\n          El título debe tener entre 5 y 200 caracteres\n        </ion-note>\n      }\n\n      <!-- Description -->\n      <ion-item>\n        <ion-textarea\n          formControlName=\"description\"\n          [label]=\"props.descriptionLabel || 'Descripción'\"\n          labelPlacement=\"floating\"\n          [placeholder]=\"props.descriptionPlaceholder || 'Proporciona más detalles...'\"\n          [maxlength]=\"5000\"\n          [counter]=\"true\"\n          [autoGrow]=\"true\"\n          rows=\"4\"\n        ></ion-textarea>\n      </ion-item>\n      @if (form.get('description')?.invalid && form.get('description')?.touched) {\n        <ion-note color=\"danger\" class=\"ion-padding-start\">\n          La descripción debe tener entre 10 y 5000 caracteres\n        </ion-note>\n      }\n\n      <!-- Error message -->\n      @if (error()) {\n        <div class=\"feedback-alert error\">\n          <ion-icon name=\"close-circle-outline\"></ion-icon>\n          <span>{{ error() }}</span>\n        </div>\n      }\n\n      <!-- Success message -->\n      @if (isSuccess()) {\n        <div class=\"feedback-alert success\">\n          <ion-icon name=\"checkmark-circle-outline\"></ion-icon>\n          <span>{{ props.successMessage || 'Feedback enviado exitosamente' }}</span>\n        </div>\n      }\n\n      <!-- Actions -->\n      <div class=\"form-actions\">\n        @if (props.cancelButtonText) {\n          <ion-button\n            fill=\"outline\"\n            color=\"medium\"\n            type=\"button\"\n            (click)=\"onCancelClick()\"\n          >\n            {{ props.cancelButtonText }}\n          </ion-button>\n        }\n        <ion-button\n          type=\"submit\"\n          [disabled]=\"form.invalid || isSubmitting()\"\n          expand=\"block\"\n        >\n          @if (isSubmitting()) {\n            <ion-spinner name=\"crescent\"></ion-spinner>\n          } @else {\n            {{ props.submitButtonText || 'Enviar' }}\n          }\n        </ion-button>\n      </div>\n    </form>\n  `,\n  styles: [\n    `\n      .feedback-form {\n        display: flex;\n        flex-direction: column;\n        gap: 8px;\n\n        &.compact {\n          gap: 4px;\n        }\n      }\n\n      .form-actions {\n        display: flex;\n        gap: 8px;\n        margin-top: 16px;\n        justify-content: flex-end;\n\n        ion-button {\n          flex: 1;\n        }\n      }\n\n      .feedback-alert {\n        display: flex;\n        align-items: center;\n        gap: 8px;\n        padding: 12px 16px;\n        border-radius: 8px;\n        margin-top: 8px;\n\n        &.error {\n          background-color: var(--ion-color-danger-tint);\n          color: var(--ion-color-danger-shade);\n        }\n\n        &.success {\n          background-color: var(--ion-color-success-tint);\n          color: var(--ion-color-success-shade);\n        }\n\n        ion-icon {\n          font-size: 20px;\n        }\n      }\n\n      ion-note {\n        font-size: 12px;\n        margin-top: -4px;\n        margin-bottom: 8px;\n      }\n    `,\n  ],\n})\nexport class FeedbackFormComponent implements OnInit {\n  /**\n   * Configuración del formulario.\n   */\n  @Input() props: FeedbackFormMetadata = {};\n\n  /**\n   * Evento emitido cuando el feedback se envía exitosamente.\n   */\n  @Output() onSubmit = new EventEmitter<FeedbackSubmitEvent>();\n\n  /**\n   * Evento emitido cuando el usuario cancela.\n   */\n  @Output() onCancel = new EventEmitter<void>();\n\n  private fb = inject(FormBuilder);\n  private feedbackService = inject(FeedbackService);\n\n  form!: FormGroup;\n  typeOptions = DEFAULT_FEEDBACK_TYPE_OPTIONS;\n\n  isSubmitting = signal(false);\n  isSuccess = signal(false);\n  error = signal<string | null>(null);\n\n  constructor() {\n    addIcons({\n      bugOutline,\n      bulbOutline,\n      chatbubbleOutline,\n      checkmarkCircleOutline,\n      closeCircleOutline,\n      documentTextOutline,\n    });\n  }\n\n  ngOnInit(): void {\n    // Filtrar tipos habilitados si se especifica\n    if (this.props.enabledTypes?.length) {\n      this.typeOptions = this.typeOptions.filter((opt) =>\n        this.props.enabledTypes!.includes(opt.value)\n      );\n    }\n\n    // Usar opciones personalizadas si se proporcionan\n    if (this.props.typeOptions?.length) {\n      this.typeOptions = this.props.typeOptions;\n    }\n\n    // Inicializar formulario\n    this.form = this.fb.group({\n      type: [this.props.defaultType || 'feedback', Validators.required],\n      title: [\n        '',\n        [Validators.required, Validators.minLength(5), Validators.maxLength(200)],\n      ],\n      description: [\n        '',\n        [Validators.required, Validators.minLength(10), Validators.maxLength(5000)],\n      ],\n    });\n  }\n\n  async handleSubmit(): Promise<void> {\n    if (this.form.invalid || this.isSubmitting()) return;\n\n    this.isSubmitting.set(true);\n    this.error.set(null);\n    this.isSuccess.set(false);\n\n    try {\n      const { type, title, description } = this.form.value;\n\n      const response = await this.feedbackService.createAsync(\n        type as FeedbackType,\n        title,\n        description,\n        [], // attachments (por ahora vacío)\n        this.props.contentRef\n      );\n\n      this.isSuccess.set(true);\n      this.form.reset({ type: this.props.defaultType || 'feedback' });\n\n      this.onSubmit.emit({\n        response,\n        type: type as FeedbackType,\n        title,\n      });\n    } catch (err: any) {\n      this.error.set(\n        err.error?.message || err.message || 'Error al enviar el feedback'\n      );\n    } finally {\n      this.isSubmitting.set(false);\n    }\n  }\n\n  onCancelClick(): void {\n    this.onCancel.emit();\n  }\n}\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2ZlZWRiYWNrLWZvcm0vdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEZlZWRiYWNrVHlwZSxcbiAgQ29udGVudFJlZixcbiAgQ3JlYXRlRmVlZGJhY2tSZXNwb25zZSxcbiAgRmVlZGJhY2tUeXBlT3B0aW9uLFxufSBmcm9tICcuLi8uLi8uLi9zZXJ2aWNlcy9mZWVkYmFjay90eXBlcyc7XG5cbi8qKlxuICogTWV0YWRhdGEgcGFyYSBlbCBjb21wb25lbnRlIEZlZWRiYWNrRm9ybS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBGZWVkYmFja0Zvcm1NZXRhZGF0YSB7XG4gIC8qKiBUaXBvIGRlIGZlZWRiYWNrIHByZXNlbGVjY2lvbmFkbyAqL1xuICBkZWZhdWx0VHlwZT86IEZlZWRiYWNrVHlwZTtcblxuICAvKiogUmVmZXJlbmNpYSBhIGNvbnRlbmlkbyAoc2kgYXBsaWNhKSAqL1xuICBjb250ZW50UmVmPzogQ29udGVudFJlZjtcblxuICAvKiogTW9zdHJhciBzZWxlY3RvciBkZSB0aXBvIChkZWZhdWx0OiB0cnVlKSAqL1xuICBzaG93VHlwZVNlbGVjdG9yPzogYm9vbGVhbjtcblxuICAvKiogVGlwb3MgZGUgZmVlZGJhY2sgaGFiaWxpdGFkb3MgKHBvciBkZWZlY3RvIHRvZG9zKSAqL1xuICBlbmFibGVkVHlwZXM/OiBGZWVkYmFja1R5cGVbXTtcblxuICAvKiogT3BjaW9uZXMgcGVyc29uYWxpemFkYXMgcGFyYSB0aXBvcyBkZSBmZWVkYmFjayAqL1xuICB0eXBlT3B0aW9ucz86IEZlZWRiYWNrVHlwZU9wdGlvbltdO1xuXG4gIC8qKiBQbGFjZWhvbGRlciBwYXJhIHTDrXR1bG8gKi9cbiAgdGl0bGVQbGFjZWhvbGRlcj86IHN0cmluZztcblxuICAvKiogUGxhY2Vob2xkZXIgcGFyYSBkZXNjcmlwY2nDs24gKi9cbiAgZGVzY3JpcHRpb25QbGFjZWhvbGRlcj86IHN0cmluZztcblxuICAvKiogTGFiZWwgcGFyYSBjYW1wbyBkZSB0w610dWxvICovXG4gIHRpdGxlTGFiZWw/OiBzdHJpbmc7XG5cbiAgLyoqIExhYmVsIHBhcmEgY2FtcG8gZGUgZGVzY3JpcGNpw7NuICovXG4gIGRlc2NyaXB0aW9uTGFiZWw/OiBzdHJpbmc7XG5cbiAgLyoqIFRleHRvIGRlbCBib3TDs24gZGUgZW52w61vICovXG4gIHN1Ym1pdEJ1dHRvblRleHQ/OiBzdHJpbmc7XG5cbiAgLyoqIFRleHRvIGRlbCBib3TDs24gY2FuY2VsYXIgKHNpIG5vIHNlIHByb3BvcmNpb25hLCBubyBzZSBtdWVzdHJhKSAqL1xuICBjYW5jZWxCdXR0b25UZXh0Pzogc3RyaW5nO1xuXG4gIC8qKiBNb3N0cmFyIGNhbXBvIGRlIGFkanVudG9zIChkZWZhdWx0OiB0cnVlKSAqL1xuICBzaG93QXR0YWNobWVudHM/OiBib29sZWFuO1xuXG4gIC8qKiBMYWJlbCBwYXJhIGNhbXBvIGRlIGFkanVudG9zICovXG4gIGF0dGFjaG1lbnRzTGFiZWw/OiBzdHJpbmc7XG5cbiAgLyoqIE1lbnNhamUgZGUgw6l4aXRvIHBlcnNvbmFsaXphZG8gKi9cbiAgc3VjY2Vzc01lc3NhZ2U/OiBzdHJpbmc7XG5cbiAgLyoqIENTUyBjbGFzcyBhZGljaW9uYWwgcGFyYSBlbCBjb250ZW5lZG9yICovXG4gIGNzc0NsYXNzPzogc3RyaW5nO1xuXG4gIC8qKiBNb2RvIGNvbXBhY3RvIChtZW5vcyBlc3BhY2lhZG8pICovXG4gIGNvbXBhY3Q/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIEV2ZW50byBlbWl0aWRvIGN1YW5kbyBlbCBmZWVkYmFjayBzZSBlbnbDrWEgZXhpdG9zYW1lbnRlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEZlZWRiYWNrU3VibWl0RXZlbnQge1xuICByZXNwb25zZTogQ3JlYXRlRmVlZGJhY2tSZXNwb25zZTtcbiAgdHlwZTogRmVlZGJhY2tUeXBlO1xuICB0aXRsZTogc3RyaW5nO1xufVxuXG4vKipcbiAqIEVzdGFkbyBkZWwgZm9ybXVsYXJpby5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBGZWVkYmFja0Zvcm1TdGF0ZSB7XG4gIGlzU3VibWl0dGluZzogYm9vbGVhbjtcbiAgaXNTdWNjZXNzOiBib29sZWFuO1xuICBlcnJvcjogc3RyaW5nIHwgbnVsbDtcbn1cbiJdfQ==
|