valtech-components 2.0.566 → 2.0.568
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/docs-nav-links/docs-nav-links.component.mjs +8 -28
- 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/esm2022/lib/services/docs/docs-navigation.service.mjs +91 -0
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/valtech-components.mjs +1353 -701
- 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/atoms/rights-footer/rights-footer.component.d.ts +1 -1
- 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/docs-nav-links/docs-nav-links.component.d.ts +1 -4
- 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/lib/components/organisms/article/article.component.d.ts +2 -2
- package/lib/services/docs/docs-navigation.service.d.ts +51 -0
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/src/lib/components/styles/_docs-page.scss +310 -0
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { Component, inject, Input } from '@angular/core';
|
|
3
3
|
import { ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { IonTextarea, IonNote } from '@ionic/angular/standalone';
|
|
5
|
+
import { PresetService } from '../../../services/presets';
|
|
5
6
|
import { ComponentStates } from '../../types';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "@angular/forms";
|
|
@@ -40,6 +41,15 @@ import * as i1 from "@angular/forms";
|
|
|
40
41
|
*/
|
|
41
42
|
export class TextareaInputComponent {
|
|
42
43
|
constructor() {
|
|
44
|
+
this.presets = inject(PresetService);
|
|
45
|
+
/**
|
|
46
|
+
* Textarea configuration object. Values here override preset values.
|
|
47
|
+
*/
|
|
48
|
+
this.props = {};
|
|
49
|
+
/**
|
|
50
|
+
* Resolved props after merging preset + explicit props.
|
|
51
|
+
*/
|
|
52
|
+
this.resolvedProps = {};
|
|
43
53
|
this.states = ComponentStates;
|
|
44
54
|
// Counter formatter for Ionic's built-in counter
|
|
45
55
|
this.counterFormatter = (inputLength, maxLength) => {
|
|
@@ -47,82 +57,104 @@ export class TextareaInputComponent {
|
|
|
47
57
|
};
|
|
48
58
|
}
|
|
49
59
|
ngOnInit() {
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
60
|
+
this.resolveProps();
|
|
61
|
+
this.applyDefaultValue();
|
|
62
|
+
}
|
|
63
|
+
ngOnChanges(changes) {
|
|
64
|
+
if (changes['preset'] || changes['props']) {
|
|
65
|
+
this.resolveProps();
|
|
66
|
+
this.applyDefaultValue();
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Merge preset configuration with explicit props.
|
|
71
|
+
* Explicit props take precedence over preset values.
|
|
72
|
+
*/
|
|
73
|
+
resolveProps() {
|
|
74
|
+
const presetProps = this.preset
|
|
75
|
+
? this.presets.get('textareaInput', this.preset)
|
|
76
|
+
: {};
|
|
77
|
+
this.resolvedProps = {
|
|
78
|
+
...presetProps,
|
|
79
|
+
...this.props,
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
applyDefaultValue() {
|
|
83
|
+
if (this.resolvedProps?.withDefault || this.resolvedProps?.value) {
|
|
84
|
+
const defaultValue = typeof this.resolvedProps.withDefault === 'string' ? this.resolvedProps.withDefault : this.resolvedProps.value || '';
|
|
85
|
+
if (this.resolvedProps.control && !this.resolvedProps.control.value) {
|
|
86
|
+
this.resolvedProps.control.setValue(defaultValue);
|
|
55
87
|
}
|
|
56
88
|
}
|
|
57
89
|
}
|
|
58
90
|
getLabel() {
|
|
59
|
-
return this.
|
|
91
|
+
return this.resolvedProps.label || '';
|
|
60
92
|
}
|
|
61
93
|
getPlaceholder() {
|
|
62
|
-
return this.
|
|
94
|
+
return this.resolvedProps.placeholder || '';
|
|
63
95
|
}
|
|
64
96
|
getHint() {
|
|
65
|
-
return this.
|
|
97
|
+
return this.resolvedProps.hint || '';
|
|
66
98
|
}
|
|
67
99
|
get hasError() {
|
|
68
|
-
return !!(this.
|
|
100
|
+
return !!(this.resolvedProps.control?.touched && this.resolvedProps.control?.invalid);
|
|
69
101
|
}
|
|
70
102
|
getErrorMessage() {
|
|
71
|
-
if (!this.
|
|
103
|
+
if (!this.resolvedProps.control?.errors || !this.resolvedProps.errors) {
|
|
72
104
|
return '';
|
|
73
105
|
}
|
|
74
|
-
const errorKeys = Object.keys(this.
|
|
106
|
+
const errorKeys = Object.keys(this.resolvedProps.control.errors);
|
|
75
107
|
for (const key of errorKeys) {
|
|
76
|
-
if (this.
|
|
77
|
-
return this.
|
|
108
|
+
if (this.resolvedProps.errors[key]) {
|
|
109
|
+
return this.resolvedProps.errors[key];
|
|
78
110
|
}
|
|
79
111
|
}
|
|
80
112
|
return '';
|
|
81
113
|
}
|
|
82
114
|
getRemainingChars() {
|
|
83
|
-
const currentLength = this.
|
|
84
|
-
return (this.
|
|
115
|
+
const currentLength = this.resolvedProps.control?.value?.length || 0;
|
|
116
|
+
return (this.resolvedProps.maxLength || 0) - currentLength;
|
|
85
117
|
}
|
|
86
118
|
getRemainingLabel() {
|
|
87
119
|
const remaining = this.getRemainingChars();
|
|
88
120
|
return remaining === 1 ? 'character remaining' : 'characters remaining';
|
|
89
121
|
}
|
|
90
122
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TextareaInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
91
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: TextareaInputComponent, isStandalone: true, selector: "val-textarea-input", inputs: { props: "props" }, ngImport: i0, template: `
|
|
92
|
-
<div class="textarea-container" [class]="
|
|
123
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: TextareaInputComponent, isStandalone: true, selector: "val-textarea-input", inputs: { preset: "preset", props: "props" }, usesOnChanges: true, ngImport: i0, template: `
|
|
124
|
+
<div class="textarea-container" [class]="resolvedProps.cssClass" [class.has-error]="hasError">
|
|
93
125
|
<ion-textarea
|
|
94
|
-
[formControl]="
|
|
126
|
+
[formControl]="resolvedProps.control"
|
|
95
127
|
[placeholder]="getPlaceholder()"
|
|
96
128
|
[label]="getLabel()"
|
|
97
|
-
[labelPlacement]="
|
|
98
|
-
[fill]="
|
|
99
|
-
[color]="hasError ? 'danger' :
|
|
100
|
-
[rows]="
|
|
101
|
-
[autoGrow]="
|
|
102
|
-
[maxlength]="
|
|
103
|
-
[minlength]="
|
|
104
|
-
[disabled]="
|
|
105
|
-
[readonly]="
|
|
106
|
-
[spellcheck]="
|
|
107
|
-
[autocapitalize]="
|
|
108
|
-
[inputmode]="
|
|
109
|
-
[wrap]="
|
|
110
|
-
[counter]="
|
|
129
|
+
[labelPlacement]="resolvedProps.labelPlacement || 'stacked'"
|
|
130
|
+
[fill]="resolvedProps.fill || 'outline'"
|
|
131
|
+
[color]="hasError ? 'danger' : resolvedProps.color"
|
|
132
|
+
[rows]="resolvedProps.rows || 4"
|
|
133
|
+
[autoGrow]="resolvedProps.autoGrow ?? false"
|
|
134
|
+
[maxlength]="resolvedProps.maxLength"
|
|
135
|
+
[minlength]="resolvedProps.minLength"
|
|
136
|
+
[disabled]="resolvedProps.state === states.DISABLED"
|
|
137
|
+
[readonly]="resolvedProps.state === states.WORKING"
|
|
138
|
+
[spellcheck]="resolvedProps.spellcheck ?? true"
|
|
139
|
+
[autocapitalize]="resolvedProps.autocapitalize || 'sentences'"
|
|
140
|
+
[inputmode]="resolvedProps.inputMode || 'text'"
|
|
141
|
+
[wrap]="resolvedProps.wrap || 'soft'"
|
|
142
|
+
[counter]="resolvedProps.showCounter && !!resolvedProps.maxLength"
|
|
111
143
|
[counterFormatter]="counterFormatter"
|
|
112
|
-
[class.no-resize]="
|
|
144
|
+
[class.no-resize]="resolvedProps.disableResize"
|
|
113
145
|
></ion-textarea>
|
|
114
146
|
|
|
115
|
-
@if (
|
|
147
|
+
@if (resolvedProps.showCounter && resolvedProps.maxLength && resolvedProps.counterFormat === 'remaining') {
|
|
116
148
|
<div class="char-counter remaining">
|
|
117
149
|
{{ getRemainingChars() }} {{ getRemainingLabel() }}
|
|
118
150
|
</div>
|
|
119
151
|
}
|
|
120
152
|
|
|
121
|
-
@if (
|
|
153
|
+
@if (resolvedProps.hint && !hasError) {
|
|
122
154
|
<ion-note class="hint">{{ getHint() }}</ion-note>
|
|
123
155
|
}
|
|
124
156
|
|
|
125
|
-
@if (hasError &&
|
|
157
|
+
@if (hasError && resolvedProps.showErrors) {
|
|
126
158
|
<ion-note class="error-message" color="danger">
|
|
127
159
|
{{ getErrorMessage() }}
|
|
128
160
|
</ion-note>
|
|
@@ -133,47 +165,49 @@ export class TextareaInputComponent {
|
|
|
133
165
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TextareaInputComponent, decorators: [{
|
|
134
166
|
type: Component,
|
|
135
167
|
args: [{ selector: 'val-textarea-input', standalone: true, imports: [CommonModule, ReactiveFormsModule, IonTextarea, IonNote], template: `
|
|
136
|
-
<div class="textarea-container" [class]="
|
|
168
|
+
<div class="textarea-container" [class]="resolvedProps.cssClass" [class.has-error]="hasError">
|
|
137
169
|
<ion-textarea
|
|
138
|
-
[formControl]="
|
|
170
|
+
[formControl]="resolvedProps.control"
|
|
139
171
|
[placeholder]="getPlaceholder()"
|
|
140
172
|
[label]="getLabel()"
|
|
141
|
-
[labelPlacement]="
|
|
142
|
-
[fill]="
|
|
143
|
-
[color]="hasError ? 'danger' :
|
|
144
|
-
[rows]="
|
|
145
|
-
[autoGrow]="
|
|
146
|
-
[maxlength]="
|
|
147
|
-
[minlength]="
|
|
148
|
-
[disabled]="
|
|
149
|
-
[readonly]="
|
|
150
|
-
[spellcheck]="
|
|
151
|
-
[autocapitalize]="
|
|
152
|
-
[inputmode]="
|
|
153
|
-
[wrap]="
|
|
154
|
-
[counter]="
|
|
173
|
+
[labelPlacement]="resolvedProps.labelPlacement || 'stacked'"
|
|
174
|
+
[fill]="resolvedProps.fill || 'outline'"
|
|
175
|
+
[color]="hasError ? 'danger' : resolvedProps.color"
|
|
176
|
+
[rows]="resolvedProps.rows || 4"
|
|
177
|
+
[autoGrow]="resolvedProps.autoGrow ?? false"
|
|
178
|
+
[maxlength]="resolvedProps.maxLength"
|
|
179
|
+
[minlength]="resolvedProps.minLength"
|
|
180
|
+
[disabled]="resolvedProps.state === states.DISABLED"
|
|
181
|
+
[readonly]="resolvedProps.state === states.WORKING"
|
|
182
|
+
[spellcheck]="resolvedProps.spellcheck ?? true"
|
|
183
|
+
[autocapitalize]="resolvedProps.autocapitalize || 'sentences'"
|
|
184
|
+
[inputmode]="resolvedProps.inputMode || 'text'"
|
|
185
|
+
[wrap]="resolvedProps.wrap || 'soft'"
|
|
186
|
+
[counter]="resolvedProps.showCounter && !!resolvedProps.maxLength"
|
|
155
187
|
[counterFormatter]="counterFormatter"
|
|
156
|
-
[class.no-resize]="
|
|
188
|
+
[class.no-resize]="resolvedProps.disableResize"
|
|
157
189
|
></ion-textarea>
|
|
158
190
|
|
|
159
|
-
@if (
|
|
191
|
+
@if (resolvedProps.showCounter && resolvedProps.maxLength && resolvedProps.counterFormat === 'remaining') {
|
|
160
192
|
<div class="char-counter remaining">
|
|
161
193
|
{{ getRemainingChars() }} {{ getRemainingLabel() }}
|
|
162
194
|
</div>
|
|
163
195
|
}
|
|
164
196
|
|
|
165
|
-
@if (
|
|
197
|
+
@if (resolvedProps.hint && !hasError) {
|
|
166
198
|
<ion-note class="hint">{{ getHint() }}</ion-note>
|
|
167
199
|
}
|
|
168
200
|
|
|
169
|
-
@if (hasError &&
|
|
201
|
+
@if (hasError && resolvedProps.showErrors) {
|
|
170
202
|
<ion-note class="error-message" color="danger">
|
|
171
203
|
{{ getErrorMessage() }}
|
|
172
204
|
</ion-note>
|
|
173
205
|
}
|
|
174
206
|
</div>
|
|
175
207
|
`, styles: [":host{display:block;width:100%}.textarea-container{position:relative}.textarea-container ion-textarea{--background: var(--ion-background-color, #fff);--padding-start: 12px;--padding-end: 12px;--padding-top: 12px;--padding-bottom: 12px}.textarea-container ion-textarea.no-resize textarea{resize:none!important}.textarea-container.has-error ion-textarea{--border-color: var(--ion-color-danger);--highlight-color: var(--ion-color-danger)}.char-counter{font-size:12px;color:var(--ion-color-medium);text-align:right;margin-top:4px;padding-right:4px}.char-counter.remaining{color:var(--ion-color-medium-shade)}.hint{display:block;font-size:12px;color:var(--ion-color-medium);margin-top:4px;padding-left:4px}.error-message{display:block;font-size:12px;margin-top:4px;padding-left:4px}ion-textarea::part(counter){font-size:12px;color:var(--ion-color-medium)}\n"] }]
|
|
176
|
-
}], propDecorators: {
|
|
208
|
+
}], propDecorators: { preset: [{
|
|
209
|
+
type: Input
|
|
210
|
+
}], props: [{
|
|
177
211
|
type: Input
|
|
178
212
|
}] } });
|
|
179
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"textarea-input.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/textarea-input/textarea-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, OnInit } from '@angular/core';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { IonTextarea, IonNote } from '@ionic/angular/standalone';\nimport { ComponentStates } from '../../types';\nimport { TextareaInputMetadata } from './types';\n\n@Component({\n  selector: 'val-textarea-input',\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, IonTextarea, IonNote],\n  template: `\n    <div class=\"textarea-container\" [class]=\"props.cssClass\" [class.has-error]=\"hasError\">\n      <ion-textarea\n        [formControl]=\"props.control\"\n        [placeholder]=\"getPlaceholder()\"\n        [label]=\"getLabel()\"\n        [labelPlacement]=\"props.labelPlacement || 'stacked'\"\n        [fill]=\"props.fill || 'outline'\"\n        [color]=\"hasError ? 'danger' : props.color\"\n        [rows]=\"props.rows || 4\"\n        [autoGrow]=\"props.autoGrow ?? false\"\n        [maxlength]=\"props.maxLength\"\n        [minlength]=\"props.minLength\"\n        [disabled]=\"props.state === states.DISABLED\"\n        [readonly]=\"props.state === states.WORKING\"\n        [spellcheck]=\"props.spellcheck ?? true\"\n        [autocapitalize]=\"props.autocapitalize || 'sentences'\"\n        [inputmode]=\"props.inputMode || 'text'\"\n        [wrap]=\"props.wrap || 'soft'\"\n        [counter]=\"props.showCounter && !!props.maxLength\"\n        [counterFormatter]=\"counterFormatter\"\n        [class.no-resize]=\"props.disableResize\"\n      ></ion-textarea>\n\n      @if (props.showCounter && props.maxLength && props.counterFormat === 'remaining') {\n        <div class=\"char-counter remaining\">\n          {{ getRemainingChars() }} {{ getRemainingLabel() }}\n        </div>\n      }\n\n      @if (props.hint && !hasError) {\n        <ion-note class=\"hint\">{{ getHint() }}</ion-note>\n      }\n\n      @if (hasError && props.showErrors) {\n        <ion-note class=\"error-message\" color=\"danger\">\n          {{ getErrorMessage() }}\n        </ion-note>\n      }\n    </div>\n  `,\n  styleUrls: ['./textarea-input.component.scss'],\n})\n/**\n * val-textarea-input\n *\n * A textarea input field with character counter, auto-grow, and validation support.\n *\n * @example Basic usage\n * ```html\n * <val-textarea-input\n *   [props]=\"{\n *     control: messageControl,\n *     placeholder: 'Enter your message',\n *     rows: 4\n *   }\"\n * ></val-textarea-input>\n * ```\n *\n * @example With counter and validation\n * ```html\n * <val-textarea-input\n *   [props]=\"{\n *     control: bioControl,\n *     label: 'Bio',\n *     placeholder: 'Tell us about yourself',\n *     maxLength: 500,\n *     showCounter: true,\n *     counterFormat: 'remaining',\n *     autoGrow: true,\n *     showErrors: true,\n *     errors: { maxlength: 'Bio is too long' }\n *   }\"\n * ></val-textarea-input>\n * ```\n */\nexport class TextareaInputComponent implements OnInit {\n  @Input() props: TextareaInputMetadata;\n\n  states = ComponentStates;\n\n  ngOnInit(): void {\n    // Apply default value if configured\n    if (this.props?.withDefault || this.props?.value) {\n      const defaultValue =\n        typeof this.props.withDefault === 'string' ? this.props.withDefault : this.props.value || '';\n      if (this.props.control && !this.props.control.value) {\n        this.props.control.setValue(defaultValue);\n      }\n    }\n  }\n\n  getLabel(): string {\n    return this.props.label || '';\n  }\n\n  getPlaceholder(): string {\n    return this.props.placeholder || '';\n  }\n\n  getHint(): string {\n    return this.props.hint || '';\n  }\n\n  get hasError(): boolean {\n    return !!(this.props.control?.touched && this.props.control?.invalid);\n  }\n\n  getErrorMessage(): string {\n    if (!this.props.control?.errors || !this.props.errors) {\n      return '';\n    }\n\n    const errorKeys = Object.keys(this.props.control.errors);\n    for (const key of errorKeys) {\n      if (this.props.errors[key]) {\n        return this.props.errors[key];\n      }\n    }\n\n    return '';\n  }\n\n  getRemainingChars(): number {\n    const currentLength = this.props.control?.value?.length || 0;\n    return (this.props.maxLength || 0) - currentLength;\n  }\n\n  getRemainingLabel(): string {\n    const remaining = this.getRemainingChars();\n    return remaining === 1 ? 'character remaining' : 'characters remaining';\n  }\n\n  // Counter formatter for Ionic's built-in counter\n  counterFormatter = (inputLength: number, maxLength: number): string => {\n    return `${inputLength}/${maxLength}`;\n  };\n}\n"]}
|
|
213
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"textarea-input.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/textarea-input/textarea-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject, Input, OnChanges, OnInit, SimpleChanges } from '@angular/core';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { IonTextarea, IonNote } from '@ionic/angular/standalone';\nimport { PresetService } from '../../../services/presets';\nimport { ComponentStates } from '../../types';\nimport { TextareaInputMetadata } from './types';\n\n@Component({\n  selector: 'val-textarea-input',\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, IonTextarea, IonNote],\n  template: `\n    <div class=\"textarea-container\" [class]=\"resolvedProps.cssClass\" [class.has-error]=\"hasError\">\n      <ion-textarea\n        [formControl]=\"resolvedProps.control\"\n        [placeholder]=\"getPlaceholder()\"\n        [label]=\"getLabel()\"\n        [labelPlacement]=\"resolvedProps.labelPlacement || 'stacked'\"\n        [fill]=\"resolvedProps.fill || 'outline'\"\n        [color]=\"hasError ? 'danger' : resolvedProps.color\"\n        [rows]=\"resolvedProps.rows || 4\"\n        [autoGrow]=\"resolvedProps.autoGrow ?? false\"\n        [maxlength]=\"resolvedProps.maxLength\"\n        [minlength]=\"resolvedProps.minLength\"\n        [disabled]=\"resolvedProps.state === states.DISABLED\"\n        [readonly]=\"resolvedProps.state === states.WORKING\"\n        [spellcheck]=\"resolvedProps.spellcheck ?? true\"\n        [autocapitalize]=\"resolvedProps.autocapitalize || 'sentences'\"\n        [inputmode]=\"resolvedProps.inputMode || 'text'\"\n        [wrap]=\"resolvedProps.wrap || 'soft'\"\n        [counter]=\"resolvedProps.showCounter && !!resolvedProps.maxLength\"\n        [counterFormatter]=\"counterFormatter\"\n        [class.no-resize]=\"resolvedProps.disableResize\"\n      ></ion-textarea>\n\n      @if (resolvedProps.showCounter && resolvedProps.maxLength && resolvedProps.counterFormat === 'remaining') {\n        <div class=\"char-counter remaining\">\n          {{ getRemainingChars() }} {{ getRemainingLabel() }}\n        </div>\n      }\n\n      @if (resolvedProps.hint && !hasError) {\n        <ion-note class=\"hint\">{{ getHint() }}</ion-note>\n      }\n\n      @if (hasError && resolvedProps.showErrors) {\n        <ion-note class=\"error-message\" color=\"danger\">\n          {{ getErrorMessage() }}\n        </ion-note>\n      }\n    </div>\n  `,\n  styleUrls: ['./textarea-input.component.scss'],\n})\n/**\n * val-textarea-input\n *\n * A textarea input field with character counter, auto-grow, and validation support.\n *\n * @example Basic usage\n * ```html\n * <val-textarea-input\n *   [props]=\"{\n *     control: messageControl,\n *     placeholder: 'Enter your message',\n *     rows: 4\n *   }\"\n * ></val-textarea-input>\n * ```\n *\n * @example With counter and validation\n * ```html\n * <val-textarea-input\n *   [props]=\"{\n *     control: bioControl,\n *     label: 'Bio',\n *     placeholder: 'Tell us about yourself',\n *     maxLength: 500,\n *     showCounter: true,\n *     counterFormat: 'remaining',\n *     autoGrow: true,\n *     showErrors: true,\n *     errors: { maxlength: 'Bio is too long' }\n *   }\"\n * ></val-textarea-input>\n * ```\n */\nexport class TextareaInputComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  /**\n   * Preset name to apply. Presets define reusable textarea configurations\n   * that can be registered at app level via provideValtechPresets().\n   *\n   * @example\n   * <val-textarea-input preset=\"form-field\" [props]=\"{ control: ctrl }\"></val-textarea-input>\n   */\n  @Input() preset?: string;\n\n  /**\n   * Textarea configuration object. Values here override preset values.\n   */\n  @Input() props: Partial<TextareaInputMetadata> = {};\n\n  /**\n   * Resolved props after merging preset + explicit props.\n   */\n  resolvedProps: TextareaInputMetadata = {} as TextareaInputMetadata;\n\n  states = ComponentStates;\n\n  ngOnInit(): void {\n    this.resolveProps();\n    this.applyDefaultValue();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['preset'] || changes['props']) {\n      this.resolveProps();\n      this.applyDefaultValue();\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('textareaInput', this.preset) as Partial<TextareaInputMetadata>)\n      : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as TextareaInputMetadata;\n  }\n\n  private applyDefaultValue(): void {\n    if (this.resolvedProps?.withDefault || this.resolvedProps?.value) {\n      const defaultValue =\n        typeof this.resolvedProps.withDefault === 'string' ? this.resolvedProps.withDefault : this.resolvedProps.value || '';\n      if (this.resolvedProps.control && !this.resolvedProps.control.value) {\n        this.resolvedProps.control.setValue(defaultValue);\n      }\n    }\n  }\n\n  getLabel(): string {\n    return this.resolvedProps.label || '';\n  }\n\n  getPlaceholder(): string {\n    return this.resolvedProps.placeholder || '';\n  }\n\n  getHint(): string {\n    return this.resolvedProps.hint || '';\n  }\n\n  get hasError(): boolean {\n    return !!(this.resolvedProps.control?.touched && this.resolvedProps.control?.invalid);\n  }\n\n  getErrorMessage(): string {\n    if (!this.resolvedProps.control?.errors || !this.resolvedProps.errors) {\n      return '';\n    }\n\n    const errorKeys = Object.keys(this.resolvedProps.control.errors);\n    for (const key of errorKeys) {\n      if (this.resolvedProps.errors[key]) {\n        return this.resolvedProps.errors[key];\n      }\n    }\n\n    return '';\n  }\n\n  getRemainingChars(): number {\n    const currentLength = this.resolvedProps.control?.value?.length || 0;\n    return (this.resolvedProps.maxLength || 0) - currentLength;\n  }\n\n  getRemainingLabel(): string {\n    const remaining = this.getRemainingChars();\n    return remaining === 1 ? 'character remaining' : 'characters remaining';\n  }\n\n  // Counter formatter for Ionic's built-in counter\n  counterFormatter = (inputLength: number, maxLength: number): string => {\n    return `${inputLength}/${maxLength}`;\n  };\n}\n"]}
|
|
@@ -1,7 +1,8 @@
|
|
|
1
|
-
import { Component, Input } from '@angular/core';
|
|
1
|
+
import { Component, inject, Input } from '@angular/core';
|
|
2
2
|
import { ReactiveFormsModule } from '@angular/forms';
|
|
3
3
|
import { IonToggle } from '@ionic/angular/standalone';
|
|
4
4
|
import { CommonModule } from '@angular/common';
|
|
5
|
+
import { PresetService } from '../../../services/presets';
|
|
5
6
|
import { ComponentStates } from '../../types';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "@angular/forms";
|
|
@@ -22,19 +23,50 @@ import * as i1 from "@angular/forms";
|
|
|
22
23
|
*/
|
|
23
24
|
export class ToggleInputComponent {
|
|
24
25
|
constructor() {
|
|
26
|
+
this.presets = inject(PresetService);
|
|
27
|
+
/**
|
|
28
|
+
* Input configuration object. Values here override preset values.
|
|
29
|
+
* @type {ToggleInputMetadata}
|
|
30
|
+
*/
|
|
31
|
+
this.props = {};
|
|
32
|
+
/**
|
|
33
|
+
* Resolved props after merging preset + explicit props.
|
|
34
|
+
*/
|
|
35
|
+
this.resolvedProps = {};
|
|
25
36
|
this.states = ComponentStates;
|
|
26
37
|
}
|
|
38
|
+
ngOnInit() {
|
|
39
|
+
this.resolveProps();
|
|
40
|
+
}
|
|
41
|
+
ngOnChanges(changes) {
|
|
42
|
+
if (changes['preset'] || changes['props']) {
|
|
43
|
+
this.resolveProps();
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Merge preset configuration with explicit props.
|
|
48
|
+
* Explicit props take precedence over preset values.
|
|
49
|
+
*/
|
|
50
|
+
resolveProps() {
|
|
51
|
+
const presetProps = this.preset
|
|
52
|
+
? this.presets.get('toggleInput', this.preset)
|
|
53
|
+
: {};
|
|
54
|
+
this.resolvedProps = {
|
|
55
|
+
...presetProps,
|
|
56
|
+
...this.props,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
27
59
|
getDisplayLabel() {
|
|
28
|
-
return this.
|
|
60
|
+
return this.resolvedProps.label || this.resolvedProps.contentFallback || '';
|
|
29
61
|
}
|
|
30
62
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ToggleInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
31
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: ToggleInputComponent, isStandalone: true, selector: "val-toggle-input", inputs: { props: "props" }, ngImport: i0, template: `
|
|
63
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: ToggleInputComponent, isStandalone: true, selector: "val-toggle-input", inputs: { preset: "preset", props: "props" }, usesOnChanges: true, ngImport: i0, template: `
|
|
32
64
|
<ion-toggle
|
|
33
|
-
[formControl]="
|
|
34
|
-
[color]="
|
|
35
|
-
[disabled]="
|
|
36
|
-
[labelPlacement]="
|
|
37
|
-
[justify]="
|
|
65
|
+
[formControl]="resolvedProps.control"
|
|
66
|
+
[color]="resolvedProps.color || 'primary'"
|
|
67
|
+
[disabled]="resolvedProps.state === states.DISABLED || resolvedProps.disabled"
|
|
68
|
+
[labelPlacement]="resolvedProps.labelPosition || 'end'"
|
|
69
|
+
[justify]="resolvedProps.justify || 'start'"
|
|
38
70
|
>
|
|
39
71
|
{{ getDisplayLabel() }}
|
|
40
72
|
</ion-toggle>
|
|
@@ -44,16 +76,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
44
76
|
type: Component,
|
|
45
77
|
args: [{ selector: 'val-toggle-input', standalone: true, imports: [CommonModule, ReactiveFormsModule, IonToggle], template: `
|
|
46
78
|
<ion-toggle
|
|
47
|
-
[formControl]="
|
|
48
|
-
[color]="
|
|
49
|
-
[disabled]="
|
|
50
|
-
[labelPlacement]="
|
|
51
|
-
[justify]="
|
|
79
|
+
[formControl]="resolvedProps.control"
|
|
80
|
+
[color]="resolvedProps.color || 'primary'"
|
|
81
|
+
[disabled]="resolvedProps.state === states.DISABLED || resolvedProps.disabled"
|
|
82
|
+
[labelPlacement]="resolvedProps.labelPosition || 'end'"
|
|
83
|
+
[justify]="resolvedProps.justify || 'start'"
|
|
52
84
|
>
|
|
53
85
|
{{ getDisplayLabel() }}
|
|
54
86
|
</ion-toggle>
|
|
55
87
|
`, styles: [":host{display:block}ion-toggle{--handle-spacing: 3px}\n"] }]
|
|
56
|
-
}], propDecorators: {
|
|
88
|
+
}], propDecorators: { preset: [{
|
|
89
|
+
type: Input
|
|
90
|
+
}], props: [{
|
|
57
91
|
type: Input
|
|
58
92
|
}] } });
|
|
59
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9nZ2xlLWlucHV0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9tb2xlY3VsZXMvdG9nZ2xlLWlucHV0L3RvZ2dsZS1pbnB1dC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFvQyxNQUFNLGVBQWUsQ0FBQztBQUMzRixPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNyRCxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDdEQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUUxRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sYUFBYSxDQUFDOzs7QUFtQjlDOzs7Ozs7Ozs7Ozs7OztHQWNHO0FBQ0gsTUFBTSxPQUFPLG9CQUFvQjtJQWhDakM7UUFpQ1UsWUFBTyxHQUFHLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztRQVd4Qzs7O1dBR0c7UUFDTSxVQUFLLEdBQWlDLEVBQUUsQ0FBQztRQUVsRDs7V0FFRztRQUNILGtCQUFhLEdBQXdCLEVBQXlCLENBQUM7UUFFL0QsV0FBTSxHQUFHLGVBQWUsQ0FBQztLQThCMUI7SUE1QkMsUUFBUTtRQUNOLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBRUQsV0FBVyxDQUFDLE9BQXNCO1FBQ2hDLElBQUksT0FBTyxDQUFDLFFBQVEsQ0FBQyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQzFDLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQztRQUN0QixDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7T0FHRztJQUNLLFlBQVk7UUFDbEIsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLE1BQU07WUFDN0IsQ0FBQyxDQUFFLElBQUksQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLGFBQWEsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFrQztZQUNoRixDQUFDLENBQUMsRUFBRSxDQUFDO1FBRVAsSUFBSSxDQUFDLGFBQWEsR0FBRztZQUNuQixHQUFHLFdBQVc7WUFDZCxHQUFHLElBQUksQ0FBQyxLQUFLO1NBQ1MsQ0FBQztJQUMzQixDQUFDO0lBRUQsZUFBZTtRQUNiLE9BQU8sSUFBSSxDQUFDLGFBQWEsQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLGFBQWEsQ0FBQyxlQUFlLElBQUksRUFBRSxDQUFDO0lBQzlFLENBQUM7K0dBcERVLG9CQUFvQjttR0FBcEIsb0JBQW9CLCtJQTVCckI7Ozs7Ozs7Ozs7R0FVVCxnSUFYUyxZQUFZLDhCQUFFLG1CQUFtQiwwVEFBRSxTQUFTOzs0RkE2QjNDLG9CQUFvQjtrQkFoQ2hDLFNBQVM7K0JBQ0Usa0JBQWtCLGNBQ2hCLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxtQkFBbUIsRUFBRSxTQUFTLENBQUMsWUFDN0M7Ozs7Ozs7Ozs7R0FVVDs4QkE0QlEsTUFBTTtzQkFBZCxLQUFLO2dCQU1HLEtBQUs7c0JBQWIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgaW5qZWN0LCBJbnB1dCwgT25DaGFuZ2VzLCBPbkluaXQsIFNpbXBsZUNoYW5nZXMgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBJb25Ub2dnbGUgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBQcmVzZXRTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvcHJlc2V0cyc7XG5pbXBvcnQgeyBUb2dnbGVJbnB1dE1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBDb21wb25lbnRTdGF0ZXMgfSBmcm9tICcuLi8uLi90eXBlcyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC10b2dnbGUtaW5wdXQnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlLCBJb25Ub2dnbGVdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxpb24tdG9nZ2xlXG4gICAgICBbZm9ybUNvbnRyb2xdPVwicmVzb2x2ZWRQcm9wcy5jb250cm9sXCJcbiAgICAgIFtjb2xvcl09XCJyZXNvbHZlZFByb3BzLmNvbG9yIHx8ICdwcmltYXJ5J1wiXG4gICAgICBbZGlzYWJsZWRdPVwicmVzb2x2ZWRQcm9wcy5zdGF0ZSA9PT0gc3RhdGVzLkRJU0FCTEVEIHx8IHJlc29sdmVkUHJvcHMuZGlzYWJsZWRcIlxuICAgICAgW2xhYmVsUGxhY2VtZW50XT1cInJlc29sdmVkUHJvcHMubGFiZWxQb3NpdGlvbiB8fCAnZW5kJ1wiXG4gICAgICBbanVzdGlmeV09XCJyZXNvbHZlZFByb3BzLmp1c3RpZnkgfHwgJ3N0YXJ0J1wiXG4gICAgPlxuICAgICAge3sgZ2V0RGlzcGxheUxhYmVsKCkgfX1cbiAgICA8L2lvbi10b2dnbGU+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL3RvZ2dsZS1pbnB1dC5jb21wb25lbnQuc2NzcyddLFxufSlcbi8qKlxuICogdmFsLXRvZ2dsZS1pbnB1dFxuICpcbiAqIEEgdG9nZ2xlL3N3aXRjaCBpbnB1dCBmb3IgYm9vbGVhbiB2YWx1ZXMsIGludGVncmF0ZWQgd2l0aCBBbmd1bGFyIGZvcm1zLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLXRvZ2dsZS1pbnB1dCBbcHJvcHNdPVwie1xuICogICBjb250cm9sOiBteUJvb2xlYW5Db250cm9sLFxuICogICBsYWJlbDogJ0FjdGl2YXIgbm90aWZpY2FjaW9uZXMnLFxuICogICBjb2xvcjogJ3ByaW1hcnknLFxuICogICBsYWJlbFBvc2l0aW9uOiAnZW5kJ1xuICogfVwiPjwvdmFsLXRvZ2dsZS1pbnB1dD5cbiAqXG4gKiBAaW5wdXQgcHJvcHM6IFRvZ2dsZUlucHV0TWV0YWRhdGEgLSBDb25maWd1cmF0aW9uIGZvciB0aGUgdG9nZ2xlIGlucHV0XG4gKi9cbmV4cG9ydCBjbGFzcyBUb2dnbGVJbnB1dENvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCwgT25DaGFuZ2VzIHtcbiAgcHJpdmF0ZSBwcmVzZXRzID0gaW5qZWN0KFByZXNldFNlcnZpY2UpO1xuXG4gIC8qKlxuICAgKiBQcmVzZXQgbmFtZSB0byBhcHBseS4gUHJlc2V0cyBkZWZpbmUgcmV1c2FibGUgdG9nZ2xlIGlucHV0IGNvbmZpZ3VyYXRpb25zXG4gICAqIHRoYXQgY2FuIGJlIHJlZ2lzdGVyZWQgYXQgYXBwIGxldmVsIHZpYSBwcm92aWRlVmFsdGVjaFByZXNldHMoKS5cbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogPHZhbC10b2dnbGUtaW5wdXQgcHJlc2V0PVwiZm9ybS1maWVsZFwiIFtwcm9wc109XCJ7IGNvbnRyb2w6IGN0cmwgfVwiPjwvdmFsLXRvZ2dsZS1pbnB1dD5cbiAgICovXG4gIEBJbnB1dCgpIHByZXNldD86IHN0cmluZztcblxuICAvKipcbiAgICogSW5wdXQgY29uZmlndXJhdGlvbiBvYmplY3QuIFZhbHVlcyBoZXJlIG92ZXJyaWRlIHByZXNldCB2YWx1ZXMuXG4gICAqIEB0eXBlIHtUb2dnbGVJbnB1dE1ldGFkYXRhfVxuICAgKi9cbiAgQElucHV0KCkgcHJvcHM6IFBhcnRpYWw8VG9nZ2xlSW5wdXRNZXRhZGF0YT4gPSB7fTtcblxuICAvKipcbiAgICogUmVzb2x2ZWQgcHJvcHMgYWZ0ZXIgbWVyZ2luZyBwcmVzZXQgKyBleHBsaWNpdCBwcm9wcy5cbiAgICovXG4gIHJlc29sdmVkUHJvcHM6IFRvZ2dsZUlucHV0TWV0YWRhdGEgPSB7fSBhcyBUb2dnbGVJbnB1dE1ldGFkYXRhO1xuXG4gIHN0YXRlcyA9IENvbXBvbmVudFN0YXRlcztcblxuICBuZ09uSW5pdCgpIHtcbiAgICB0aGlzLnJlc29sdmVQcm9wcygpO1xuICB9XG5cbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcykge1xuICAgIGlmIChjaGFuZ2VzWydwcmVzZXQnXSB8fCBjaGFuZ2VzWydwcm9wcyddKSB7XG4gICAgICB0aGlzLnJlc29sdmVQcm9wcygpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBNZXJnZSBwcmVzZXQgY29uZmlndXJhdGlvbiB3aXRoIGV4cGxpY2l0IHByb3BzLlxuICAgKiBFeHBsaWNpdCBwcm9wcyB0YWtlIHByZWNlZGVuY2Ugb3ZlciBwcmVzZXQgdmFsdWVzLlxuICAgKi9cbiAgcHJpdmF0ZSByZXNvbHZlUHJvcHMoKTogdm9pZCB7XG4gICAgY29uc3QgcHJlc2V0UHJvcHMgPSB0aGlzLnByZXNldFxuICAgICAgPyAodGhpcy5wcmVzZXRzLmdldCgndG9nZ2xlSW5wdXQnLCB0aGlzLnByZXNldCkgYXMgUGFydGlhbDxUb2dnbGVJbnB1dE1ldGFkYXRhPilcbiAgICAgIDoge307XG5cbiAgICB0aGlzLnJlc29sdmVkUHJvcHMgPSB7XG4gICAgICAuLi5wcmVzZXRQcm9wcyxcbiAgICAgIC4uLnRoaXMucHJvcHMsXG4gICAgfSBhcyBUb2dnbGVJbnB1dE1ldGFkYXRhO1xuICB9XG5cbiAgZ2V0RGlzcGxheUxhYmVsKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMucmVzb2x2ZWRQcm9wcy5sYWJlbCB8fCB0aGlzLnJlc29sdmVkUHJvcHMuY29udGVudEZhbGxiYWNrIHx8ICcnO1xuICB9XG59XG4iXX0=
|
|
@@ -242,7 +242,7 @@ export class WinnerDisplayComponent {
|
|
|
242
242
|
</div>
|
|
243
243
|
}
|
|
244
244
|
</div>
|
|
245
|
-
`, isInline: true, styles: [":host{display:block}.winner-display-container{--winner-color: var(--ion-color-warning);--animation-duration: .8s;position:relative;display:flex;flex-direction:column;align-items:center;text-align:center;padding:32px 24px;border-radius:16px;overflow:hidden}.bg-solid{background:var(--ion-color-light);border:2px solid var(--winner-color)}.bg-gradient{background:linear-gradient(135deg,var(--ion-color-light) 0%,rgba(var(--ion-color-warning-rgb),.1) 100%);border:2px solid var(--winner-color)}.bg-transparent{background:transparent}.trophy-container{margin-bottom:16px}.trophy-icon{font-size:56px;color:var(--winner-color);filter:drop-shadow(0 4px 8px rgba(0,0,0,.1))}.winner-label{font-size:14px;font-weight:600;text-transform:uppercase;letter-spacing:2px;color:var(--winner-color);margin-bottom:8px}.winner-info{display:flex;flex-direction:column;align-items:center;gap:8px;margin-bottom:16px}.winner-avatar{margin-bottom:8px}.winner-name{font-size:28px;font-weight:700;color:var(--ion-color-dark);margin:0;line-height:1.2}.winner-location{display:flex;align-items:center;gap:4px;font-size:14px;color:var(--ion-color-medium-shade)}.winner-location ion-icon{font-size:16px}.ticket-info{display:flex;align-items:center;gap:8px;padding:8px 16px;background:rgba(var(--ion-color-dark-rgb),.05);border-radius:24px;margin-top:8px}.ticket-info ion-icon{font-size:20px;color:var(--winner-color)}.ticket-label{font-size:14px;color:var(--ion-color-medium-shade)}.ticket-number{font-size:18px;font-weight:700;color:var(--ion-color-dark);font-variant-numeric:tabular-nums}.prize-info{display:flex;flex-direction:column;align-items:center;gap:12px;padding:16px;background:var(--ion-color-light-shade);border-radius:12px;width:100%;max-width:400px}.prize-image{width:120px;height:120px;border-radius:8px;overflow:hidden;background:#fff;box-shadow:0 4px 12px #0000001a}.prize-image img{width:100%;height:100%;object-fit:cover}.prize-details{display:flex;flex-direction:column;align-items:center;gap:4px}.prize-header{display:flex;align-items:center;gap:4px;color:var(--ion-color-medium-shade);font-size:12px}.prize-header ion-icon{font-size:16px;color:var(--winner-color)}.prize-label{text-transform:uppercase;letter-spacing:1px}.prize-name{font-size:20px;font-weight:600;color:var(--ion-color-dark);margin:0}.prize-description{font-size:14px;color:var(--ion-color-medium-shade);margin:4px 0 0}.prize-value{font-size:24px;font-weight:700;color:var(--winner-color);margin-top:8px}.confetti-container{position:absolute;inset:0;pointer-events:none;overflow:hidden;opacity:0;transition:opacity .3s}.confetti-container.active{opacity:1}.confetti-piece{position:absolute;width:10px;height:10px;top:-20px;left:var(--x, 50%);background:var(--winner-color);opacity:0;animation:confetti-fall 3s ease-out var(--delay, 0s) forwards}.confetti-piece:nth-child(2n){background:var(--ion-color-primary);width:8px;height:12px}.confetti-piece:nth-child(3n){background:var(--ion-color-danger);border-radius:50%}.confetti-piece:nth-child(5n){background:var(--ion-color-success);width:6px;height:14px}@keyframes confetti-fall{0%{opacity:1;transform:translateY(0) rotate(0)}to{opacity:0;transform:translateY(400px) rotate(var(--rotation, 720deg))}}.animation-fade:not(.revealed) .winner-info,.animation-fade:not(.revealed) .prize-info{opacity:0}.animation-fade.animating .winner-info,.animation-fade.animating .prize-info{animation:fade-in var(--animation-duration) ease-out forwards}.animation-zoom:not(.revealed) .winner-info,.animation-zoom:not(.revealed) .prize-info{opacity:0;transform:scale(.5)}.animation-zoom.animating .winner-info,.animation-zoom.animating .prize-info{animation:zoom-in var(--animation-duration) cubic-bezier(.34,1.56,.64,1) forwards}.animation-slide:not(.revealed) .winner-info,.animation-slide:not(.revealed) .prize-info{opacity:0;transform:translateY(40px)}.animation-slide.animating .winner-info,.animation-slide.animating .prize-info{animation:slide-up var(--animation-duration) ease-out forwards}.animation-slide.animating .prize-info{animation-delay:.2s}.animation-spotlight:not(.revealed) .winner-info,.animation-spotlight:not(.revealed) .prize-info{opacity:0}.animation-spotlight:not(.revealed):before{content:\"\";position:absolute;top:50%;left:50%;width:0;height:0;background:radial-gradient(circle,rgba(255,255,255,.8) 0%,transparent 70%);border-radius:50%;transform:translate(-50%,-50%)}.animation-spotlight.animating:before{animation:spotlight var(--animation-duration) ease-out forwards}.animation-spotlight.animating .winner-info,.animation-spotlight.animating .prize-info{animation:fade-in calc(var(--animation-duration) * .5) ease-out calc(var(--animation-duration) * .5) forwards}@keyframes fade-in{0%{opacity:0}to{opacity:1}}@keyframes zoom-in{0%{opacity:0;transform:scale(.5)}to{opacity:1;transform:scale(1)}}@keyframes slide-up{0%{opacity:0;transform:translateY(40px)}to{opacity:1;transform:translateY(0)}}@keyframes spotlight{0%{width:0;height:0}to{width:200%;height:200%}}.size-small{padding:20px 16px}.size-small .trophy-icon{font-size:40px}.size-small .winner-label{font-size:12px}.size-small .winner-name{font-size:20px}.size-small .ticket-number,.size-small .prize-name{font-size:16px}.size-small .prize-value{font-size:18px}.size-large{padding:48px 32px}.size-large .trophy-icon{font-size:80px}.size-large .winner-label{font-size:18px;letter-spacing:3px}.size-large .winner-name{font-size:40px}.size-large .ticket-number{font-size:24px}.size-large .prize-image{width:160px;height:160px}.size-large .prize-name{font-size:28px}.size-large .prize-value{font-size:32px}.mode-compact{padding:16px}.mode-compact .trophy-container,.mode-compact .prize-info{display:none}.mode-compact .winner-label{font-size:11px;margin-bottom:4px}.mode-compact .winner-name{font-size:18px}.mode-celebration .trophy-icon{animation:trophy-bounce 1s ease-in-out infinite}.mode-celebration.revealed .winner-name{animation:glow 2s ease-in-out infinite}@keyframes trophy-bounce{0%,to{transform:translateY(0)}50%{transform:translateY(-8px)}}@keyframes glow{0%,to{text-shadow:0 0 10px rgba(var(--ion-color-warning-rgb),.3)}50%{text-shadow:0 0 20px rgba(var(--ion-color-warning-rgb),.6)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: AvatarComponent, selector: "val-avatar", inputs: ["props"], outputs: ["onClick"] }] }); }
|
|
245
|
+
`, isInline: true, styles: [":host{display:block}.winner-display-container{--winner-color: var(--ion-color-warning);--animation-duration: .8s;position:relative;display:flex;flex-direction:column;align-items:center;text-align:center;padding:32px 24px;border-radius:16px;overflow:hidden}.bg-solid{background:var(--ion-color-light);border:2px solid var(--winner-color)}.bg-gradient{background:linear-gradient(135deg,var(--ion-color-light) 0%,rgba(var(--ion-color-warning-rgb),.1) 100%);border:2px solid var(--winner-color)}.bg-transparent{background:transparent}.trophy-container{margin-bottom:16px}.trophy-icon{font-size:56px;color:var(--winner-color);filter:drop-shadow(0 4px 8px rgba(0,0,0,.1))}.winner-label{font-size:14px;font-weight:600;text-transform:uppercase;letter-spacing:2px;color:var(--winner-color);margin-bottom:8px}.winner-info{display:flex;flex-direction:column;align-items:center;gap:8px;margin-bottom:16px}.winner-avatar{margin-bottom:8px}.winner-name{font-size:28px;font-weight:700;color:var(--ion-color-dark);margin:0;line-height:1.2}.winner-location{display:flex;align-items:center;gap:4px;font-size:14px;color:var(--ion-color-medium-shade)}.winner-location ion-icon{font-size:16px}.ticket-info{display:flex;align-items:center;gap:8px;padding:8px 16px;background:rgba(var(--ion-color-dark-rgb),.05);border-radius:24px;margin-top:8px}.ticket-info ion-icon{font-size:20px;color:var(--winner-color)}.ticket-label{font-size:14px;color:var(--ion-color-medium-shade)}.ticket-number{font-size:18px;font-weight:700;color:var(--ion-color-dark);font-variant-numeric:tabular-nums}.prize-info{display:flex;flex-direction:column;align-items:center;gap:12px;padding:16px;background:var(--ion-color-light-shade);border-radius:12px;width:100%;max-width:400px}.prize-image{width:120px;height:120px;border-radius:8px;overflow:hidden;background:#fff;box-shadow:0 4px 12px #0000001a}.prize-image img{width:100%;height:100%;object-fit:cover}.prize-details{display:flex;flex-direction:column;align-items:center;gap:4px}.prize-header{display:flex;align-items:center;gap:4px;color:var(--ion-color-medium-shade);font-size:12px}.prize-header ion-icon{font-size:16px;color:var(--winner-color)}.prize-label{text-transform:uppercase;letter-spacing:1px}.prize-name{font-size:20px;font-weight:600;color:var(--ion-color-dark);margin:0}.prize-description{font-size:14px;color:var(--ion-color-medium-shade);margin:4px 0 0}.prize-value{font-size:24px;font-weight:700;color:var(--winner-color);margin-top:8px}.confetti-container{position:absolute;inset:0;pointer-events:none;overflow:hidden;opacity:0;transition:opacity .3s}.confetti-container.active{opacity:1}.confetti-piece{position:absolute;width:10px;height:10px;top:-20px;left:var(--x, 50%);background:var(--winner-color);opacity:0;animation:confetti-fall 3s ease-out var(--delay, 0s) forwards}.confetti-piece:nth-child(2n){background:var(--ion-color-primary);width:8px;height:12px}.confetti-piece:nth-child(3n){background:var(--ion-color-danger);border-radius:50%}.confetti-piece:nth-child(5n){background:var(--ion-color-success);width:6px;height:14px}@keyframes confetti-fall{0%{opacity:1;transform:translateY(0) rotate(0)}to{opacity:0;transform:translateY(400px) rotate(var(--rotation, 720deg))}}.animation-fade:not(.revealed) .winner-info,.animation-fade:not(.revealed) .prize-info{opacity:0}.animation-fade.animating .winner-info,.animation-fade.animating .prize-info{animation:fade-in var(--animation-duration) ease-out forwards}.animation-zoom:not(.revealed) .winner-info,.animation-zoom:not(.revealed) .prize-info{opacity:0;transform:scale(.5)}.animation-zoom.animating .winner-info,.animation-zoom.animating .prize-info{animation:zoom-in var(--animation-duration) cubic-bezier(.34,1.56,.64,1) forwards}.animation-slide:not(.revealed) .winner-info,.animation-slide:not(.revealed) .prize-info{opacity:0;transform:translateY(40px)}.animation-slide.animating .winner-info,.animation-slide.animating .prize-info{animation:slide-up var(--animation-duration) ease-out forwards}.animation-slide.animating .prize-info{animation-delay:.2s}.animation-spotlight:not(.revealed) .winner-info,.animation-spotlight:not(.revealed) .prize-info{opacity:0}.animation-spotlight:not(.revealed):before{content:\"\";position:absolute;top:50%;left:50%;width:0;height:0;background:radial-gradient(circle,rgba(255,255,255,.8) 0%,transparent 70%);border-radius:50%;transform:translate(-50%,-50%)}.animation-spotlight.animating:before{animation:spotlight var(--animation-duration) ease-out forwards}.animation-spotlight.animating .winner-info,.animation-spotlight.animating .prize-info{animation:fade-in calc(var(--animation-duration) * .5) ease-out calc(var(--animation-duration) * .5) forwards}@keyframes fade-in{0%{opacity:0}to{opacity:1}}@keyframes zoom-in{0%{opacity:0;transform:scale(.5)}to{opacity:1;transform:scale(1)}}@keyframes slide-up{0%{opacity:0;transform:translateY(40px)}to{opacity:1;transform:translateY(0)}}@keyframes spotlight{0%{width:0;height:0}to{width:200%;height:200%}}.size-small{padding:20px 16px}.size-small .trophy-icon{font-size:40px}.size-small .winner-label{font-size:12px}.size-small .winner-name{font-size:20px}.size-small .ticket-number,.size-small .prize-name{font-size:16px}.size-small .prize-value{font-size:18px}.size-large{padding:48px 32px}.size-large .trophy-icon{font-size:80px}.size-large .winner-label{font-size:18px;letter-spacing:3px}.size-large .winner-name{font-size:40px}.size-large .ticket-number{font-size:24px}.size-large .prize-image{width:160px;height:160px}.size-large .prize-name{font-size:28px}.size-large .prize-value{font-size:32px}.mode-compact{padding:16px}.mode-compact .trophy-container,.mode-compact .prize-info{display:none}.mode-compact .winner-label{font-size:11px;margin-bottom:4px}.mode-compact .winner-name{font-size:18px}.mode-celebration .trophy-icon{animation:trophy-bounce 1s ease-in-out infinite}.mode-celebration.revealed .winner-name{animation:glow 2s ease-in-out infinite}@keyframes trophy-bounce{0%,to{transform:translateY(0)}50%{transform:translateY(-8px)}}@keyframes glow{0%,to{text-shadow:0 0 10px rgba(var(--ion-color-warning-rgb),.3)}50%{text-shadow:0 0 20px rgba(var(--ion-color-warning-rgb),.6)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: AvatarComponent, selector: "val-avatar", inputs: ["preset", "props"], outputs: ["onClick"] }] }); }
|
|
246
246
|
}
|
|
247
247
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: WinnerDisplayComponent, decorators: [{
|
|
248
248
|
type: Component,
|
|
@@ -340,7 +340,7 @@ export class CommentSectionComponent {
|
|
|
340
340
|
</div>
|
|
341
341
|
}
|
|
342
342
|
</div>
|
|
343
|
-
`, isInline: true, styles: [":host{display:block}.comment-section.loading{pointer-events:none}.section-header{display:flex;align-items:center;justify-content:space-between;flex-wrap:wrap;gap:12px;margin-bottom:20px;padding-bottom:16px;border-bottom:1px solid var(--ion-color-light-shade)}.header-title{display:flex;align-items:center;gap:8px}.title-icon{font-size:24px;color:var(--ion-color-primary)}.title{margin:0;font-size:18px;font-weight:600;color:var(--ion-text-color)}.count-badge{font-size:12px;font-weight:600;--padding-start: 8px;--padding-end: 8px}.header-actions{display:flex;align-items:center;gap:8px}.sort-select-item{--padding-start: 0;--padding-end: 0;--inner-padding-end: 0;--background: transparent;--min-height: 36px}.sort-select-item ion-select{--padding-start: 8px;--padding-end: 8px;min-width:140px;font-size:14px}.sort-icon{font-size:18px;color:var(--ion-color-medium);margin-right:4px}.new-comment-section{margin-bottom:24px;padding:16px;background-color:var(--ion-color-light);border-radius:12px}.input-wrapper{display:flex;gap:12px}.input-avatar{flex-shrink:0}.input-avatar val-avatar{--size: 40px}.input-container{flex:1;min-width:0}.comment-textarea{--background: var(--ion-background-color);--padding-start: 12px;--padding-end: 12px;--padding-top: 10px;--padding-bottom: 10px;border-radius:8px;font-size:14px;min-height:60px;border:1px solid var(--ion-color-light-shade);transition:border-color .2s}.comment-textarea:focus-within{border-color:var(--ion-color-primary)}.input-actions{display:flex;align-items:center;justify-content:flex-end;gap:12px;margin-top:10px}.char-counter{font-size:12px;color:var(--ion-color-medium)}.char-counter.warning{color:var(--ion-color-warning);font-weight:500}.loading-state{display:flex;flex-direction:column;gap:20px}.skeleton-comment{display:flex;gap:12px;padding:12px 0}.skeleton-comment .skeleton-content{flex:1;display:flex;flex-direction:column;gap:8px}.comments-list.with-dividers val-comment{padding:16px 0}.comments-list.with-dividers val-comment:first-child{padding-top:0}.comment-divider{height:1px;background-color:var(--ion-color-light-shade);margin:0}.load-more-section{display:flex;justify-content:center;padding:20px 0}.load-more-section ion-button{min-width:200px}.load-more-section ion-spinner{width:28px;height:28px}.empty-state{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:48px 24px;text-align:center}.empty-icon{font-size:64px;color:var(--ion-color-medium);opacity:.5;margin-bottom:16px}.empty-title{margin:0 0 8px;font-size:18px;font-weight:600;color:var(--ion-text-color)}.empty-message{margin:0;font-size:14px;color:var(--ion-color-medium);max-width:300px}@media (max-width: 576px){.section-header{flex-direction:column;align-items:flex-start}.header-actions,.sort-select-item{width:100%}.sort-select-item ion-select{width:100%}.input-wrapper{flex-direction:column}.input-avatar{display:none}.input-actions{flex-direction:column;align-items:stretch}.input-actions ion-button{width:100%}}@media (prefers-color-scheme: dark){.new-comment-section{background-color:var(--ion-color-step-100, #1a1a1a)}.comment-textarea{--background: var(--ion-color-step-50, #121212);border-color:var(--ion-color-step-200, #2a2a2a)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { 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.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { 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: IonBadge, selector: "ion-badge", inputs: ["color", "mode"] }, { 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"] }, { 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: 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: IonInfiniteScroll, selector: "ion-infinite-scroll", inputs: ["disabled", "position", "threshold"] }, { kind: "component", type: IonInfiniteScrollContent, selector: "ion-infinite-scroll-content", inputs: ["loadingSpinner", "loadingText"] }, { kind: "component", type: CommentComponent, selector: "val-comment", inputs: ["props"], outputs: ["authorClick", "reactionClick", "actionClick", "menuItemClick", "loadMoreClick", "collapseToggle"] }, { kind: "component", type: AvatarComponent, selector: "val-avatar", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: SkeletonComponent, selector: "val-skeleton", inputs: ["props"] }] }); }
|
|
343
|
+
`, isInline: true, styles: [":host{display:block}.comment-section.loading{pointer-events:none}.section-header{display:flex;align-items:center;justify-content:space-between;flex-wrap:wrap;gap:12px;margin-bottom:20px;padding-bottom:16px;border-bottom:1px solid var(--ion-color-light-shade)}.header-title{display:flex;align-items:center;gap:8px}.title-icon{font-size:24px;color:var(--ion-color-primary)}.title{margin:0;font-size:18px;font-weight:600;color:var(--ion-text-color)}.count-badge{font-size:12px;font-weight:600;--padding-start: 8px;--padding-end: 8px}.header-actions{display:flex;align-items:center;gap:8px}.sort-select-item{--padding-start: 0;--padding-end: 0;--inner-padding-end: 0;--background: transparent;--min-height: 36px}.sort-select-item ion-select{--padding-start: 8px;--padding-end: 8px;min-width:140px;font-size:14px}.sort-icon{font-size:18px;color:var(--ion-color-medium);margin-right:4px}.new-comment-section{margin-bottom:24px;padding:16px;background-color:var(--ion-color-light);border-radius:12px}.input-wrapper{display:flex;gap:12px}.input-avatar{flex-shrink:0}.input-avatar val-avatar{--size: 40px}.input-container{flex:1;min-width:0}.comment-textarea{--background: var(--ion-background-color);--padding-start: 12px;--padding-end: 12px;--padding-top: 10px;--padding-bottom: 10px;border-radius:8px;font-size:14px;min-height:60px;border:1px solid var(--ion-color-light-shade);transition:border-color .2s}.comment-textarea:focus-within{border-color:var(--ion-color-primary)}.input-actions{display:flex;align-items:center;justify-content:flex-end;gap:12px;margin-top:10px}.char-counter{font-size:12px;color:var(--ion-color-medium)}.char-counter.warning{color:var(--ion-color-warning);font-weight:500}.loading-state{display:flex;flex-direction:column;gap:20px}.skeleton-comment{display:flex;gap:12px;padding:12px 0}.skeleton-comment .skeleton-content{flex:1;display:flex;flex-direction:column;gap:8px}.comments-list.with-dividers val-comment{padding:16px 0}.comments-list.with-dividers val-comment:first-child{padding-top:0}.comment-divider{height:1px;background-color:var(--ion-color-light-shade);margin:0}.load-more-section{display:flex;justify-content:center;padding:20px 0}.load-more-section ion-button{min-width:200px}.load-more-section ion-spinner{width:28px;height:28px}.empty-state{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:48px 24px;text-align:center}.empty-icon{font-size:64px;color:var(--ion-color-medium);opacity:.5;margin-bottom:16px}.empty-title{margin:0 0 8px;font-size:18px;font-weight:600;color:var(--ion-text-color)}.empty-message{margin:0;font-size:14px;color:var(--ion-color-medium);max-width:300px}@media (max-width: 576px){.section-header{flex-direction:column;align-items:flex-start}.header-actions,.sort-select-item{width:100%}.sort-select-item ion-select{width:100%}.input-wrapper{flex-direction:column}.input-avatar{display:none}.input-actions{flex-direction:column;align-items:stretch}.input-actions ion-button{width:100%}}@media (prefers-color-scheme: dark){.new-comment-section{background-color:var(--ion-color-step-100, #1a1a1a)}.comment-textarea{--background: var(--ion-color-step-50, #121212);border-color:var(--ion-color-step-200, #2a2a2a)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { 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.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { 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: IonBadge, selector: "ion-badge", inputs: ["color", "mode"] }, { 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"] }, { 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: 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: IonInfiniteScroll, selector: "ion-infinite-scroll", inputs: ["disabled", "position", "threshold"] }, { kind: "component", type: IonInfiniteScrollContent, selector: "ion-infinite-scroll-content", inputs: ["loadingSpinner", "loadingText"] }, { kind: "component", type: CommentComponent, selector: "val-comment", inputs: ["props"], outputs: ["authorClick", "reactionClick", "actionClick", "menuItemClick", "loadMoreClick", "collapseToggle"] }, { kind: "component", type: AvatarComponent, selector: "val-avatar", inputs: ["preset", "props"], outputs: ["onClick"] }, { kind: "component", type: SkeletonComponent, selector: "val-skeleton", inputs: ["props"] }] }); }
|
|
344
344
|
}
|
|
345
345
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CommentSectionComponent, decorators: [{
|
|
346
346
|
type: Component,
|
|
@@ -534,4 +534,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
534
534
|
}], replyStart: [{
|
|
535
535
|
type: Output
|
|
536
536
|
}] } });
|
|
537
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment-section.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/comment-section/comment-section.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  IonIcon,\n  IonButton,\n  IonBadge,\n  IonTextarea,\n  IonItem,\n  IonSelect,\n  IonSelectOption,\n  IonSpinner,\n  IonInfiniteScroll,\n  IonInfiniteScrollContent,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n} from 'ionicons/icons';\n\nimport { CommentComponent } from '../../molecules/comment/comment.component';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { SkeletonComponent } from '../../atoms/skeleton/skeleton.component';\nimport {\n  CommentSectionMetadata,\n  CommentSortOption,\n  CommentSortChangeEvent,\n  CommentSubmitEvent,\n  CommentSectionLoadMoreEvent,\n} from './types';\nimport {\n  CommentAuthorClickEvent,\n  CommentReactionClickEvent,\n  CommentActionClickEvent,\n  CommentMenuItemClickEvent,\n  CommentLoadMoreEvent,\n} from '../../molecules/comment/types';\n\naddIcons({\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n});\n\n@Component({\n  selector: 'val-comment-section',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonIcon,\n    IonButton,\n    IonBadge,\n    IonTextarea,\n    IonItem,\n    IonSelect,\n    IonSelectOption,\n    IonSpinner,\n    IonInfiniteScroll,\n    IonInfiniteScrollContent,\n    CommentComponent,\n    AvatarComponent,\n    SkeletonComponent,\n  ],\n  template: `\n    <div class=\"comment-section\" [class.loading]=\"props.loading\">\n      <!-- Header -->\n      <div class=\"section-header\">\n        <div class=\"header-title\">\n          <ion-icon name=\"chatbubbles-outline\" class=\"title-icon\"></ion-icon>\n          <h3 class=\"title\">{{ displayTitle }}</h3>\n          @if (props.showCount !== false && props.count !== undefined) {\n            <ion-badge color=\"medium\" class=\"count-badge\">\n              {{ formatCount(props.count) }}\n            </ion-badge>\n          }\n        </div>\n\n        @if (props.sortOptions && props.sortOptions.length > 0) {\n          <div class=\"header-actions\">\n            <ion-item lines=\"none\" class=\"sort-select-item\">\n              <ion-icon name=\"swap-vertical-outline\" slot=\"start\" class=\"sort-icon\"></ion-icon>\n              <ion-select\n                [value]=\"props.selectedSort\"\n                [placeholder]=\"props.sortLabel || getSortByLabel()\"\n                interface=\"popover\"\n                (ionChange)=\"onSortChange($event)\"\n              >\n                @for (option of props.sortOptions; track option.token) {\n                  <ion-select-option [value]=\"option.token\">\n                    {{ getSortOptionLabel(option) }}\n                  </ion-select-option>\n                }\n              </ion-select>\n            </ion-item>\n          </div>\n        }\n      </div>\n\n      <!-- New Comment Input -->\n      @if (props.showInput !== false) {\n        <div class=\"new-comment-section\">\n          <div class=\"input-wrapper\">\n            @if (props.inputConfig?.currentUser?.avatar) {\n              <div class=\"input-avatar\">\n                <val-avatar [props]=\"props.inputConfig.currentUser.avatar\"></val-avatar>\n              </div>\n            }\n\n            <div class=\"input-container\">\n              <ion-textarea\n                [(ngModel)]=\"newCommentText\"\n                [placeholder]=\"getInputPlaceholder()\"\n                [maxlength]=\"props.inputConfig?.maxLength || 2000\"\n                [disabled]=\"props.inputConfig?.disabled || props.loading\"\n                [autoGrow]=\"true\"\n                rows=\"2\"\n                class=\"comment-textarea\"\n              ></ion-textarea>\n\n              <div class=\"input-actions\">\n                @if (props.inputConfig?.showCounter && props.inputConfig?.maxLength) {\n                  <span class=\"char-counter\" [class.warning]=\"isNearLimit()\">\n                    {{ newCommentText.length }} / {{ props.inputConfig.maxLength }}\n                  </span>\n                }\n\n                <ion-button\n                  [color]=\"props.inputConfig?.submitColor || props.color || 'primary'\"\n                  [disabled]=\"!canSubmit()\"\n                  size=\"small\"\n                  (click)=\"onSubmitComment()\"\n                >\n                  <ion-icon name=\"send-outline\" slot=\"start\"></ion-icon>\n                  {{ getSubmitLabel() }}\n                </ion-button>\n              </div>\n            </div>\n          </div>\n        </div>\n      }\n\n      @if (props.loading) {\n        <div class=\"loading-state\">\n          @for (i of getSkeletonArray(); track i) {\n            <div class=\"skeleton-comment\">\n              <val-skeleton [props]=\"{ type: 'avatar', width: '36px', height: '36px' }\"></val-skeleton>\n              <div class=\"skeleton-content\">\n                <val-skeleton [props]=\"{ type: 'text', width: '120px', height: '14px' }\"></val-skeleton>\n                <val-skeleton [props]=\"{ type: 'paragraph', lines: 2 }\"></val-skeleton>\n              </div>\n            </div>\n          }\n        </div>\n      } @else if (props.comments && props.comments.length > 0) {\n        <div class=\"comments-list\" [class.with-dividers]=\"props.showDividers\">\n          @for (comment of props.comments; track comment.token) {\n            <val-comment\n              [props]=\"comment\"\n              (authorClick)=\"authorClick.emit($event)\"\n              (reactionClick)=\"reactionClick.emit($event)\"\n              (actionClick)=\"onCommentAction($event)\"\n              (menuItemClick)=\"menuItemClick.emit($event)\"\n              (loadMoreClick)=\"commentLoadMore.emit($event)\"\n              (collapseToggle)=\"collapseToggle.emit($event)\"\n            ></val-comment>\n\n            @if (props.showDividers && !$last) {\n              <div class=\"comment-divider\"></div>\n            }\n          }\n\n          @if (props.hasMore && props.paginationMode !== 'infinite') {\n            <div class=\"load-more-section\">\n              @if (props.loadingMore) {\n                <ion-spinner name=\"crescent\" [color]=\"props.color || 'primary'\"></ion-spinner>\n              } @else {\n                <ion-button\n                  fill=\"outline\"\n                  [color]=\"props.color || 'primary'\"\n                  expand=\"block\"\n                  (click)=\"onLoadMore()\"\n                >\n                  {{ displayLoadMoreLabel }}\n                </ion-button>\n              }\n            </div>\n          }\n        </div>\n\n        @if (props.paginationMode === 'infinite') {\n          <ion-infinite-scroll\n            [threshold]=\"props.infiniteScrollThreshold || '100px'\"\n            [position]=\"props.infiniteScrollPosition || 'bottom'\"\n            [disabled]=\"!props.hasMore\"\n            (ionInfinite)=\"onInfiniteScroll($event)\"\n          >\n            <ion-infinite-scroll-content\n              [loadingSpinner]=\"'crescent'\"\n              [loadingText]=\"displayLoadMoreLabel\"\n            ></ion-infinite-scroll-content>\n          </ion-infinite-scroll>\n        }\n      } @else {\n        <div class=\"empty-state\">\n          <ion-icon\n            [name]=\"props.emptyState?.icon || 'chatbubble-ellipses-outline'\"\n            class=\"empty-icon\"\n          ></ion-icon>\n          <h4 class=\"empty-title\">{{ getEmptyTitle() }}</h4>\n          <p class=\"empty-message\">{{ getEmptyMessage() }}</p>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./comment-section.component.scss'],\n})\n/**\n * val-comment-section\n *\n * An organism component that provides a complete comment section with:\n * - Header with title and count\n * - Sort/filter options\n * - New comment input\n * - Comments list with val-comment\n * - Load more pagination\n * - Empty state\n *\n * @example Basic usage\n * <val-comment-section [props]=\"{\n *   title: 'Comments',\n *   count: 42,\n *   comments: commentsArray,\n *   showInput: true\n * }\"></val-comment-section>\n *\n * @example With sorting\n * <val-comment-section [props]=\"{\n *   title: 'Reviews',\n *   count: 128,\n *   comments: reviews,\n *   sortOptions: [\n *     { token: 'newest', label: 'Newest first' },\n *     { token: 'oldest', label: 'Oldest first' },\n *     { token: 'popular', label: 'Most popular' }\n *   ],\n *   selectedSort: 'newest'\n * }\" (sortChange)=\"onSort($event)\"></val-comment-section>\n *\n * @input props: CommentSectionMetadata - Configuration for the section\n * @output sortChange - Sort option changed\n * @output commentSubmit - New comment submitted\n * @output loadMore - Load more clicked\n * @output authorClick - Comment author clicked (bubbled from val-comment)\n * @output reactionClick - Reaction clicked (bubbled)\n * @output actionClick - Action clicked (bubbled)\n * @output menuItemClick - Menu item clicked (bubbled)\n * @output commentLoadMore - Load more replies clicked (bubbled)\n * @output collapseToggle - Comment collapse toggled (bubbled)\n */\nexport class CommentSectionComponent implements OnInit {\n  private i18n = inject(I18nService);\n\n  @Input() props: CommentSectionMetadata;\n\n  // Section events\n  @Output() sortChange = new EventEmitter<CommentSortChangeEvent>();\n  @Output() commentSubmit = new EventEmitter<CommentSubmitEvent>();\n  @Output() loadMore = new EventEmitter<CommentSectionLoadMoreEvent>();\n\n  // Bubbled events from val-comment\n  @Output() authorClick = new EventEmitter<CommentAuthorClickEvent>();\n  @Output() reactionClick = new EventEmitter<CommentReactionClickEvent>();\n  @Output() actionClick = new EventEmitter<CommentActionClickEvent>();\n  @Output() menuItemClick = new EventEmitter<CommentMenuItemClickEvent>();\n  @Output() commentLoadMore = new EventEmitter<CommentLoadMoreEvent>();\n  @Output() collapseToggle = new EventEmitter<{ token: string; collapsed: boolean }>();\n\n  // Reply state\n  @Output() replyStart = new EventEmitter<{ commentToken: string }>();\n\n  newCommentText = '';\n  replyingTo: string | null = null;\n\n  displayTitle = '';\n  displayLoadMoreLabel = '';\n\n  ngOnInit(): void {\n    this.updateDisplayTexts();\n  }\n\n  private updateDisplayTexts(): void {\n    this.displayTitle = this.props.title || this.i18n.t('comments');\n    this.displayLoadMoreLabel = this.props.loadMoreLabel || this.i18n.t('loadMoreComments');\n  }\n\n  getSortByLabel(): string {\n    return this.i18n.t('sortBy');\n  }\n\n  formatCount(count: number): string {\n    if (count >= 1000000) {\n      return (count / 1000000).toFixed(1).replace(/\\.0$/, '') + 'M';\n    }\n    if (count >= 1000) {\n      return (count / 1000).toFixed(1).replace(/\\.0$/, '') + 'K';\n    }\n    return count.toString();\n  }\n\n  getSortOptionLabel(option: CommentSortOption): string {\n    return option.label;\n  }\n\n  getInputPlaceholder(): string {\n    return this.props.inputConfig?.placeholder || this.i18n.t('writeComment');\n  }\n\n  getSubmitLabel(): string {\n    return this.props.inputConfig?.submitLabel || this.i18n.t('publish');\n  }\n\n  getEmptyTitle(): string {\n    return this.props.emptyState?.title || this.i18n.t('noCommentsYet');\n  }\n\n  getEmptyMessage(): string {\n    return this.props.emptyState?.message || this.i18n.t('beFirstToComment');\n  }\n\n  getSkeletonArray(): number[] {\n    const count = this.props.skeletonCount || 3;\n    return Array(count).fill(0).map((_, i) => i);\n  }\n\n  canSubmit(): boolean {\n    const minLength = this.props.inputConfig?.minLength || 1;\n    return (\n      this.newCommentText.trim().length >= minLength &&\n      !this.props.loading &&\n      !this.props.inputConfig?.disabled\n    );\n  }\n\n  isNearLimit(): boolean {\n    const maxLength = this.props.inputConfig?.maxLength || 2000;\n    return this.newCommentText.length > maxLength * 0.9;\n  }\n\n  onSortChange(event: CustomEvent): void {\n    const selectedToken = event.detail.value;\n    const option = this.props.sortOptions?.find((o) => o.token === selectedToken);\n\n    if (option) {\n      this.sortChange.emit({\n        option,\n        previousSort: this.props.selectedSort,\n      });\n    }\n  }\n\n  onSubmitComment(): void {\n    if (!this.canSubmit()) return;\n\n    this.commentSubmit.emit({\n      content: this.newCommentText.trim(),\n      parentToken: this.replyingTo || undefined,\n      sectionToken: this.props.token,\n    });\n\n    this.newCommentText = '';\n    this.replyingTo = null;\n  }\n\n  onCommentAction(event: CommentActionClickEvent): void {\n    // Check if it's a reply action\n    if (event.action.token === 'reply') {\n      this.replyingTo = event.commentToken;\n      this.replyStart.emit({ commentToken: event.commentToken });\n    }\n\n    this.actionClick.emit(event);\n  }\n\n  onLoadMore(): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n  }\n\n  onInfiniteScroll(event: CustomEvent): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n\n    // The parent component should call completeInfiniteScroll() when done loading\n    // Store reference to complete the infinite scroll\n    this.infiniteScrollEvent = event;\n  }\n\n  // Call this method from parent after loading more comments\n  completeInfiniteScroll(): void {\n    if (this.infiniteScrollEvent) {\n      (this.infiniteScrollEvent.target as HTMLIonInfiniteScrollElement).complete();\n      this.infiniteScrollEvent = null;\n    }\n  }\n\n  private infiniteScrollEvent: CustomEvent | null = null;\n\n  // Method to programmatically start a reply\n  startReply(commentToken: string): void {\n    this.replyingTo = commentToken;\n  }\n\n  // Method to cancel reply\n  cancelReply(): void {\n    this.replyingTo = null;\n  }\n}\n"]}
|
|
537
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment-section.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/comment-section/comment-section.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  IonIcon,\n  IonButton,\n  IonBadge,\n  IonTextarea,\n  IonItem,\n  IonSelect,\n  IonSelectOption,\n  IonSpinner,\n  IonInfiniteScroll,\n  IonInfiniteScrollContent,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n} from 'ionicons/icons';\n\nimport { CommentComponent } from '../../molecules/comment/comment.component';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { SkeletonComponent } from '../../atoms/skeleton/skeleton.component';\nimport {\n  CommentSectionMetadata,\n  CommentSortOption,\n  CommentSortChangeEvent,\n  CommentSubmitEvent,\n  CommentSectionLoadMoreEvent,\n} from './types';\nimport {\n  CommentAuthorClickEvent,\n  CommentReactionClickEvent,\n  CommentActionClickEvent,\n  CommentMenuItemClickEvent,\n  CommentLoadMoreEvent,\n} from '../../molecules/comment/types';\n\naddIcons({\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n});\n\n@Component({\n  selector: 'val-comment-section',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonIcon,\n    IonButton,\n    IonBadge,\n    IonTextarea,\n    IonItem,\n    IonSelect,\n    IonSelectOption,\n    IonSpinner,\n    IonInfiniteScroll,\n    IonInfiniteScrollContent,\n    CommentComponent,\n    AvatarComponent,\n    SkeletonComponent,\n  ],\n  template: `\n    <div class=\"comment-section\" [class.loading]=\"props.loading\">\n      <!-- Header -->\n      <div class=\"section-header\">\n        <div class=\"header-title\">\n          <ion-icon name=\"chatbubbles-outline\" class=\"title-icon\"></ion-icon>\n          <h3 class=\"title\">{{ displayTitle }}</h3>\n          @if (props.showCount !== false && props.count !== undefined) {\n            <ion-badge color=\"medium\" class=\"count-badge\">\n              {{ formatCount(props.count) }}\n            </ion-badge>\n          }\n        </div>\n\n        @if (props.sortOptions && props.sortOptions.length > 0) {\n          <div class=\"header-actions\">\n            <ion-item lines=\"none\" class=\"sort-select-item\">\n              <ion-icon name=\"swap-vertical-outline\" slot=\"start\" class=\"sort-icon\"></ion-icon>\n              <ion-select\n                [value]=\"props.selectedSort\"\n                [placeholder]=\"props.sortLabel || getSortByLabel()\"\n                interface=\"popover\"\n                (ionChange)=\"onSortChange($event)\"\n              >\n                @for (option of props.sortOptions; track option.token) {\n                  <ion-select-option [value]=\"option.token\">\n                    {{ getSortOptionLabel(option) }}\n                  </ion-select-option>\n                }\n              </ion-select>\n            </ion-item>\n          </div>\n        }\n      </div>\n\n      <!-- New Comment Input -->\n      @if (props.showInput !== false) {\n        <div class=\"new-comment-section\">\n          <div class=\"input-wrapper\">\n            @if (props.inputConfig?.currentUser?.avatar) {\n              <div class=\"input-avatar\">\n                <val-avatar [props]=\"props.inputConfig.currentUser.avatar\"></val-avatar>\n              </div>\n            }\n\n            <div class=\"input-container\">\n              <ion-textarea\n                [(ngModel)]=\"newCommentText\"\n                [placeholder]=\"getInputPlaceholder()\"\n                [maxlength]=\"props.inputConfig?.maxLength || 2000\"\n                [disabled]=\"props.inputConfig?.disabled || props.loading\"\n                [autoGrow]=\"true\"\n                rows=\"2\"\n                class=\"comment-textarea\"\n              ></ion-textarea>\n\n              <div class=\"input-actions\">\n                @if (props.inputConfig?.showCounter && props.inputConfig?.maxLength) {\n                  <span class=\"char-counter\" [class.warning]=\"isNearLimit()\">\n                    {{ newCommentText.length }} / {{ props.inputConfig.maxLength }}\n                  </span>\n                }\n\n                <ion-button\n                  [color]=\"props.inputConfig?.submitColor || props.color || 'primary'\"\n                  [disabled]=\"!canSubmit()\"\n                  size=\"small\"\n                  (click)=\"onSubmitComment()\"\n                >\n                  <ion-icon name=\"send-outline\" slot=\"start\"></ion-icon>\n                  {{ getSubmitLabel() }}\n                </ion-button>\n              </div>\n            </div>\n          </div>\n        </div>\n      }\n\n      @if (props.loading) {\n        <div class=\"loading-state\">\n          @for (i of getSkeletonArray(); track i) {\n            <div class=\"skeleton-comment\">\n              <val-skeleton [props]=\"{ type: 'avatar', width: '36px', height: '36px' }\"></val-skeleton>\n              <div class=\"skeleton-content\">\n                <val-skeleton [props]=\"{ type: 'text', width: '120px', height: '14px' }\"></val-skeleton>\n                <val-skeleton [props]=\"{ type: 'paragraph', lines: 2 }\"></val-skeleton>\n              </div>\n            </div>\n          }\n        </div>\n      } @else if (props.comments && props.comments.length > 0) {\n        <div class=\"comments-list\" [class.with-dividers]=\"props.showDividers\">\n          @for (comment of props.comments; track comment.token) {\n            <val-comment\n              [props]=\"comment\"\n              (authorClick)=\"authorClick.emit($event)\"\n              (reactionClick)=\"reactionClick.emit($event)\"\n              (actionClick)=\"onCommentAction($event)\"\n              (menuItemClick)=\"menuItemClick.emit($event)\"\n              (loadMoreClick)=\"commentLoadMore.emit($event)\"\n              (collapseToggle)=\"collapseToggle.emit($event)\"\n            ></val-comment>\n\n            @if (props.showDividers && !$last) {\n              <div class=\"comment-divider\"></div>\n            }\n          }\n\n          @if (props.hasMore && props.paginationMode !== 'infinite') {\n            <div class=\"load-more-section\">\n              @if (props.loadingMore) {\n                <ion-spinner name=\"crescent\" [color]=\"props.color || 'primary'\"></ion-spinner>\n              } @else {\n                <ion-button\n                  fill=\"outline\"\n                  [color]=\"props.color || 'primary'\"\n                  expand=\"block\"\n                  (click)=\"onLoadMore()\"\n                >\n                  {{ displayLoadMoreLabel }}\n                </ion-button>\n              }\n            </div>\n          }\n        </div>\n\n        @if (props.paginationMode === 'infinite') {\n          <ion-infinite-scroll\n            [threshold]=\"props.infiniteScrollThreshold || '100px'\"\n            [position]=\"props.infiniteScrollPosition || 'bottom'\"\n            [disabled]=\"!props.hasMore\"\n            (ionInfinite)=\"onInfiniteScroll($event)\"\n          >\n            <ion-infinite-scroll-content\n              [loadingSpinner]=\"'crescent'\"\n              [loadingText]=\"displayLoadMoreLabel\"\n            ></ion-infinite-scroll-content>\n          </ion-infinite-scroll>\n        }\n      } @else {\n        <div class=\"empty-state\">\n          <ion-icon\n            [name]=\"props.emptyState?.icon || 'chatbubble-ellipses-outline'\"\n            class=\"empty-icon\"\n          ></ion-icon>\n          <h4 class=\"empty-title\">{{ getEmptyTitle() }}</h4>\n          <p class=\"empty-message\">{{ getEmptyMessage() }}</p>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./comment-section.component.scss'],\n})\n/**\n * val-comment-section\n *\n * An organism component that provides a complete comment section with:\n * - Header with title and count\n * - Sort/filter options\n * - New comment input\n * - Comments list with val-comment\n * - Load more pagination\n * - Empty state\n *\n * @example Basic usage\n * <val-comment-section [props]=\"{\n *   title: 'Comments',\n *   count: 42,\n *   comments: commentsArray,\n *   showInput: true\n * }\"></val-comment-section>\n *\n * @example With sorting\n * <val-comment-section [props]=\"{\n *   title: 'Reviews',\n *   count: 128,\n *   comments: reviews,\n *   sortOptions: [\n *     { token: 'newest', label: 'Newest first' },\n *     { token: 'oldest', label: 'Oldest first' },\n *     { token: 'popular', label: 'Most popular' }\n *   ],\n *   selectedSort: 'newest'\n * }\" (sortChange)=\"onSort($event)\"></val-comment-section>\n *\n * @input props: CommentSectionMetadata - Configuration for the section\n * @output sortChange - Sort option changed\n * @output commentSubmit - New comment submitted\n * @output loadMore - Load more clicked\n * @output authorClick - Comment author clicked (bubbled from val-comment)\n * @output reactionClick - Reaction clicked (bubbled)\n * @output actionClick - Action clicked (bubbled)\n * @output menuItemClick - Menu item clicked (bubbled)\n * @output commentLoadMore - Load more replies clicked (bubbled)\n * @output collapseToggle - Comment collapse toggled (bubbled)\n */\nexport class CommentSectionComponent implements OnInit {\n  private i18n = inject(I18nService);\n\n  @Input() props: CommentSectionMetadata;\n\n  // Section events\n  @Output() sortChange = new EventEmitter<CommentSortChangeEvent>();\n  @Output() commentSubmit = new EventEmitter<CommentSubmitEvent>();\n  @Output() loadMore = new EventEmitter<CommentSectionLoadMoreEvent>();\n\n  // Bubbled events from val-comment\n  @Output() authorClick = new EventEmitter<CommentAuthorClickEvent>();\n  @Output() reactionClick = new EventEmitter<CommentReactionClickEvent>();\n  @Output() actionClick = new EventEmitter<CommentActionClickEvent>();\n  @Output() menuItemClick = new EventEmitter<CommentMenuItemClickEvent>();\n  @Output() commentLoadMore = new EventEmitter<CommentLoadMoreEvent>();\n  @Output() collapseToggle = new EventEmitter<{ token: string; collapsed: boolean }>();\n\n  // Reply state\n  @Output() replyStart = new EventEmitter<{ commentToken: string }>();\n\n  newCommentText = '';\n  replyingTo: string | null = null;\n\n  displayTitle = '';\n  displayLoadMoreLabel = '';\n\n  ngOnInit(): void {\n    this.updateDisplayTexts();\n  }\n\n  private updateDisplayTexts(): void {\n    this.displayTitle = this.props.title || this.i18n.t('comments');\n    this.displayLoadMoreLabel = this.props.loadMoreLabel || this.i18n.t('loadMoreComments');\n  }\n\n  getSortByLabel(): string {\n    return this.i18n.t('sortBy');\n  }\n\n  formatCount(count: number): string {\n    if (count >= 1000000) {\n      return (count / 1000000).toFixed(1).replace(/\\.0$/, '') + 'M';\n    }\n    if (count >= 1000) {\n      return (count / 1000).toFixed(1).replace(/\\.0$/, '') + 'K';\n    }\n    return count.toString();\n  }\n\n  getSortOptionLabel(option: CommentSortOption): string {\n    return option.label;\n  }\n\n  getInputPlaceholder(): string {\n    return this.props.inputConfig?.placeholder || this.i18n.t('writeComment');\n  }\n\n  getSubmitLabel(): string {\n    return this.props.inputConfig?.submitLabel || this.i18n.t('publish');\n  }\n\n  getEmptyTitle(): string {\n    return this.props.emptyState?.title || this.i18n.t('noCommentsYet');\n  }\n\n  getEmptyMessage(): string {\n    return this.props.emptyState?.message || this.i18n.t('beFirstToComment');\n  }\n\n  getSkeletonArray(): number[] {\n    const count = this.props.skeletonCount || 3;\n    return Array(count).fill(0).map((_, i) => i);\n  }\n\n  canSubmit(): boolean {\n    const minLength = this.props.inputConfig?.minLength || 1;\n    return (\n      this.newCommentText.trim().length >= minLength &&\n      !this.props.loading &&\n      !this.props.inputConfig?.disabled\n    );\n  }\n\n  isNearLimit(): boolean {\n    const maxLength = this.props.inputConfig?.maxLength || 2000;\n    return this.newCommentText.length > maxLength * 0.9;\n  }\n\n  onSortChange(event: CustomEvent): void {\n    const selectedToken = event.detail.value;\n    const option = this.props.sortOptions?.find((o) => o.token === selectedToken);\n\n    if (option) {\n      this.sortChange.emit({\n        option,\n        previousSort: this.props.selectedSort,\n      });\n    }\n  }\n\n  onSubmitComment(): void {\n    if (!this.canSubmit()) return;\n\n    this.commentSubmit.emit({\n      content: this.newCommentText.trim(),\n      parentToken: this.replyingTo || undefined,\n      sectionToken: this.props.token,\n    });\n\n    this.newCommentText = '';\n    this.replyingTo = null;\n  }\n\n  onCommentAction(event: CommentActionClickEvent): void {\n    // Check if it's a reply action\n    if (event.action.token === 'reply') {\n      this.replyingTo = event.commentToken;\n      this.replyStart.emit({ commentToken: event.commentToken });\n    }\n\n    this.actionClick.emit(event);\n  }\n\n  onLoadMore(): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n  }\n\n  onInfiniteScroll(event: CustomEvent): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n\n    // The parent component should call completeInfiniteScroll() when done loading\n    // Store reference to complete the infinite scroll\n    this.infiniteScrollEvent = event;\n  }\n\n  // Call this method from parent after loading more comments\n  completeInfiniteScroll(): void {\n    if (this.infiniteScrollEvent) {\n      (this.infiniteScrollEvent.target as HTMLIonInfiniteScrollElement).complete();\n      this.infiniteScrollEvent = null;\n    }\n  }\n\n  private infiniteScrollEvent: CustomEvent | null = null;\n\n  // Method to programmatically start a reply\n  startReply(commentToken: string): void {\n    this.replyingTo = commentToken;\n  }\n\n  // Method to cancel reply\n  cancelReply(): void {\n    this.replyingTo = null;\n  }\n}\n"]}
|