valtech-components 2.0.594 → 2.0.596
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/molecules/content-reaction/content-reaction.component.mjs +191 -0
- package/esm2022/lib/components/molecules/content-reaction/types.mjs +2 -0
- package/esm2022/lib/components/molecules/docs-section/docs-section.component.mjs +85 -0
- package/esm2022/lib/components/molecules/docs-section/types.mjs +2 -0
- package/esm2022/lib/components/molecules/feedback-form/feedback-form.component.mjs +354 -0
- package/esm2022/lib/components/molecules/feedback-form/types.mjs +2 -0
- package/esm2022/lib/components/organisms/terminal-404/terminal-404.component.mjs +9 -8
- package/esm2022/lib/components/templates/docs-page/docs-page.component.mjs +188 -0
- package/esm2022/lib/components/templates/docs-page/types.mjs +2 -0
- package/esm2022/lib/services/ads/types.mjs +1 -1
- package/esm2022/lib/services/feedback/config.mjs +49 -0
- package/esm2022/lib/services/feedback/feedback.service.mjs +228 -0
- package/esm2022/lib/services/feedback/index.mjs +44 -0
- package/esm2022/lib/services/feedback/types.mjs +30 -0
- package/esm2022/public-api.mjs +10 -5
- package/fesm2022/valtech-components.mjs +1143 -11
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/content-reaction/content-reaction.component.d.ts +57 -0
- package/lib/components/molecules/content-reaction/types.d.ts +61 -0
- package/lib/components/molecules/docs-section/docs-section.component.d.ts +29 -0
- package/lib/components/molecules/docs-section/types.d.ts +27 -0
- package/lib/components/molecules/feedback-form/feedback-form.component.d.ts +58 -0
- package/lib/components/molecules/feedback-form/types.d.ts +54 -0
- package/lib/components/organisms/terminal-404/terminal-404.component.d.ts +1 -0
- package/lib/components/templates/docs-page/docs-page.component.d.ts +54 -0
- package/lib/components/templates/docs-page/types.d.ts +69 -0
- package/lib/services/ads/types.d.ts +10 -0
- package/lib/services/feedback/config.d.ts +35 -0
- package/lib/services/feedback/feedback.service.d.ts +110 -0
- package/lib/services/feedback/index.d.ts +40 -0
- package/lib/services/feedback/types.d.ts +159 -0
- package/package.json +1 -1
- package/public-api.d.ts +9 -0
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
import { Component, inject, Input, Output, EventEmitter, signal, computed, } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import { IonButton, IonSpinner, IonTextarea, } from '@ionic/angular/standalone';
|
|
5
|
+
import { FeedbackService } from '../../../services/feedback/feedback.service';
|
|
6
|
+
import { ToastService } from '../../../services/toast.service';
|
|
7
|
+
import { I18nService } from '../../../services/i18n';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
/**
|
|
10
|
+
* Componente para reacciones de contenido con emojis.
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```html
|
|
14
|
+
* <val-content-reaction
|
|
15
|
+
* [props]="{
|
|
16
|
+
* entityRef: { entityType: 'article', entityId: 'art-123' },
|
|
17
|
+
* question: '¿Te fue útil este artículo?'
|
|
18
|
+
* }"
|
|
19
|
+
* (reactionSubmit)="onReactionSubmit($event)"
|
|
20
|
+
* />
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
export class ContentReactionComponent {
|
|
24
|
+
constructor() {
|
|
25
|
+
this.feedbackService = inject(FeedbackService);
|
|
26
|
+
this.toast = inject(ToastService);
|
|
27
|
+
this.i18n = inject(I18nService);
|
|
28
|
+
this.props = {};
|
|
29
|
+
this.reactionSubmit = new EventEmitter();
|
|
30
|
+
this.reactionChange = new EventEmitter();
|
|
31
|
+
// Estado reactivo
|
|
32
|
+
this.state = signal({
|
|
33
|
+
selectedValue: null,
|
|
34
|
+
comment: '',
|
|
35
|
+
isLoading: false,
|
|
36
|
+
isSubmitted: false,
|
|
37
|
+
hadPreviousReaction: false,
|
|
38
|
+
error: null,
|
|
39
|
+
});
|
|
40
|
+
// Valores por defecto
|
|
41
|
+
this.defaultEmojis = ['😞', '😐', '😊'];
|
|
42
|
+
this.defaultLabels = [
|
|
43
|
+
'No me ayudó',
|
|
44
|
+
'Regular',
|
|
45
|
+
'Muy útil',
|
|
46
|
+
];
|
|
47
|
+
this.reactionValues = ['negative', 'neutral', 'positive'];
|
|
48
|
+
// Computed properties
|
|
49
|
+
this.resolvedProps = computed(() => ({
|
|
50
|
+
entityRef: this.props.entityRef,
|
|
51
|
+
question: this.props.question || this.t('question'),
|
|
52
|
+
showComment: this.props.showComment ?? true,
|
|
53
|
+
commentPlaceholder: this.props.commentPlaceholder || this.t('commentPlaceholder'),
|
|
54
|
+
maxCommentLength: this.props.maxCommentLength ?? 500,
|
|
55
|
+
emojis: this.props.emojis || this.defaultEmojis,
|
|
56
|
+
emojiLabels: this.props.emojiLabels || this.defaultLabels,
|
|
57
|
+
showThankYou: this.props.showThankYou ?? true,
|
|
58
|
+
thankYouMessage: this.props.thankYouMessage || this.t('thankYou'),
|
|
59
|
+
disabled: this.props.disabled ?? false,
|
|
60
|
+
readonly: this.props.readonly ?? false,
|
|
61
|
+
}));
|
|
62
|
+
this.showCommentField = computed(() => this.state().selectedValue !== null && this.resolvedProps().showComment);
|
|
63
|
+
this.canSubmit = computed(() => this.state().selectedValue !== null && !this.state().isLoading);
|
|
64
|
+
}
|
|
65
|
+
ngOnInit() {
|
|
66
|
+
this.loadPreviousReaction();
|
|
67
|
+
}
|
|
68
|
+
ngOnChanges(changes) {
|
|
69
|
+
if (changes['props'] && !changes['props'].firstChange) {
|
|
70
|
+
this.loadPreviousReaction();
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
async loadPreviousReaction() {
|
|
74
|
+
if (!this.props.entityRef)
|
|
75
|
+
return;
|
|
76
|
+
this.state.update((s) => ({ ...s, isLoading: true, error: null }));
|
|
77
|
+
try {
|
|
78
|
+
const check = await this.feedbackService.checkFeedback(this.props.entityRef.entityType, this.props.entityRef.entityId);
|
|
79
|
+
if (check.hasFeedback && check.reactionValue) {
|
|
80
|
+
this.state.update((s) => ({
|
|
81
|
+
...s,
|
|
82
|
+
selectedValue: check.reactionValue,
|
|
83
|
+
hadPreviousReaction: true,
|
|
84
|
+
isLoading: false,
|
|
85
|
+
isSubmitted: true,
|
|
86
|
+
}));
|
|
87
|
+
}
|
|
88
|
+
else {
|
|
89
|
+
this.state.update((s) => ({ ...s, isLoading: false }));
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
catch (error) {
|
|
93
|
+
console.error('Error loading previous reaction:', error);
|
|
94
|
+
this.state.update((s) => ({ ...s, isLoading: false }));
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
selectReaction(value) {
|
|
98
|
+
if (this.resolvedProps().disabled || this.resolvedProps().readonly)
|
|
99
|
+
return;
|
|
100
|
+
const previousValue = this.state().selectedValue;
|
|
101
|
+
this.state.update((s) => ({
|
|
102
|
+
...s,
|
|
103
|
+
selectedValue: value,
|
|
104
|
+
isSubmitted: false,
|
|
105
|
+
error: null,
|
|
106
|
+
}));
|
|
107
|
+
this.reactionChange.emit({ value, previousValue });
|
|
108
|
+
}
|
|
109
|
+
async submitReaction() {
|
|
110
|
+
const currentState = this.state();
|
|
111
|
+
const props = this.resolvedProps();
|
|
112
|
+
if (!currentState.selectedValue || props.disabled)
|
|
113
|
+
return;
|
|
114
|
+
this.state.update((s) => ({ ...s, isLoading: true, error: null }));
|
|
115
|
+
try {
|
|
116
|
+
await this.feedbackService.createReaction(props.entityRef, currentState.selectedValue, currentState.comment || undefined);
|
|
117
|
+
this.state.update((s) => ({
|
|
118
|
+
...s,
|
|
119
|
+
isLoading: false,
|
|
120
|
+
isSubmitted: true,
|
|
121
|
+
hadPreviousReaction: true,
|
|
122
|
+
}));
|
|
123
|
+
this.reactionSubmit.emit({
|
|
124
|
+
value: currentState.selectedValue,
|
|
125
|
+
comment: currentState.comment || undefined,
|
|
126
|
+
entityRef: props.entityRef,
|
|
127
|
+
isUpdate: currentState.hadPreviousReaction,
|
|
128
|
+
});
|
|
129
|
+
if (props.showThankYou) {
|
|
130
|
+
this.toast.show({
|
|
131
|
+
message: props.thankYouMessage,
|
|
132
|
+
duration: 2000,
|
|
133
|
+
position: 'bottom',
|
|
134
|
+
color: 'success',
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
catch (error) {
|
|
139
|
+
console.error('Error submitting reaction:', error);
|
|
140
|
+
this.state.update((s) => ({
|
|
141
|
+
...s,
|
|
142
|
+
isLoading: false,
|
|
143
|
+
error: this.t('errorSubmitting'),
|
|
144
|
+
}));
|
|
145
|
+
this.toast.show({
|
|
146
|
+
message: this.t('errorSubmitting'),
|
|
147
|
+
duration: 3000,
|
|
148
|
+
position: 'bottom',
|
|
149
|
+
color: 'danger',
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
updateComment(event) {
|
|
154
|
+
const value = event.detail.value || '';
|
|
155
|
+
this.state.update((s) => ({ ...s, comment: value }));
|
|
156
|
+
}
|
|
157
|
+
getEmoji(index) {
|
|
158
|
+
return this.resolvedProps().emojis[index];
|
|
159
|
+
}
|
|
160
|
+
getEmojiLabel(index) {
|
|
161
|
+
return this.resolvedProps().emojiLabels[index];
|
|
162
|
+
}
|
|
163
|
+
isSelected(value) {
|
|
164
|
+
return this.state().selectedValue === value;
|
|
165
|
+
}
|
|
166
|
+
t(key) {
|
|
167
|
+
const translations = {
|
|
168
|
+
question: '¿Te resultó útil este contenido?',
|
|
169
|
+
commentPlaceholder: 'Cuéntanos más (opcional)...',
|
|
170
|
+
submit: 'Enviar opinión',
|
|
171
|
+
update: 'Actualizar opinión',
|
|
172
|
+
thankYou: '¡Gracias por tu opinión!',
|
|
173
|
+
submitted: 'Tu opinión ha sido registrada',
|
|
174
|
+
errorSubmitting: 'Error al enviar. Intenta de nuevo.',
|
|
175
|
+
};
|
|
176
|
+
return this.i18n.t(key, 'ContentReaction') || translations[key] || key;
|
|
177
|
+
}
|
|
178
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ContentReactionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
179
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ContentReactionComponent, isStandalone: true, selector: "val-content-reaction", inputs: { props: "props" }, outputs: { reactionSubmit: "reactionSubmit", reactionChange: "reactionChange" }, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"content-reaction\"\n [class.disabled]=\"resolvedProps().disabled\"\n [class.readonly]=\"resolvedProps().readonly\"\n>\n <!-- Loading inicial -->\n @if (state().isLoading && !state().selectedValue) {\n <div class=\"loading-container\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n </div>\n } @else {\n <!-- Pregunta -->\n <p class=\"question\">{{ resolvedProps().question }}</p>\n\n <!-- Emojis -->\n <div class=\"emoji-container\">\n @for (value of reactionValues; track value; let i = $index) {\n <button\n type=\"button\"\n class=\"emoji-button\"\n [class.selected]=\"isSelected(value)\"\n [class.negative]=\"value === 'negative' && isSelected(value)\"\n [class.neutral]=\"value === 'neutral' && isSelected(value)\"\n [class.positive]=\"value === 'positive' && isSelected(value)\"\n [attr.aria-label]=\"getEmojiLabel(i)\"\n [attr.aria-pressed]=\"isSelected(value)\"\n [disabled]=\"resolvedProps().disabled || resolvedProps().readonly\"\n (click)=\"selectReaction(value)\"\n >\n <span class=\"emoji\">{{ getEmoji(i) }}</span>\n </button>\n }\n </div>\n\n <!-- Campo de comentario (solo si hay selecci\u00F3n) -->\n @if (showCommentField()) {\n <div class=\"comment-section\">\n <ion-textarea\n [value]=\"state().comment\"\n [placeholder]=\"resolvedProps().commentPlaceholder\"\n [maxlength]=\"resolvedProps().maxCommentLength\"\n [disabled]=\"resolvedProps().disabled\"\n [rows]=\"3\"\n class=\"comment-textarea\"\n (ionInput)=\"updateComment($event)\"\n ></ion-textarea>\n <span class=\"char-count\">\n {{ state().comment.length }}/{{ resolvedProps().maxCommentLength }}\n </span>\n </div>\n }\n\n <!-- Bot\u00F3n de env\u00EDo -->\n @if (state().selectedValue && !state().isSubmitted) {\n <ion-button\n expand=\"block\"\n [disabled]=\"!canSubmit()\"\n (click)=\"submitReaction()\"\n class=\"submit-button\"\n >\n @if (state().isLoading) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else {\n {{ state().hadPreviousReaction ? t('update') : t('submit') }}\n }\n </ion-button>\n }\n\n <!-- Mensaje de confirmaci\u00F3n -->\n @if (state().isSubmitted) {\n <p class=\"submitted-message\">\n {{ t('submitted') }}\n </p>\n }\n\n <!-- Error -->\n @if (state().error) {\n <p class=\"error-message\">{{ state().error }}</p>\n }\n }\n</div>\n", styles: [":host{display:block}.content-reaction{padding:16px;text-align:center}.content-reaction.disabled{opacity:.6;pointer-events:none}.content-reaction.readonly{pointer-events:none}.question{font-size:16px;font-weight:500;color:var(--ion-color-dark);margin:0 0 16px}.emoji-container{display:flex;justify-content:center;gap:24px;margin-bottom:20px}.emoji-button{background:transparent;border:2px solid var(--ion-color-light-shade);border-radius:50%;width:64px;height:64px;cursor:pointer;transition:all .3s cubic-bezier(.4,0,.2,1);display:flex;align-items:center;justify-content:center;padding:0}.emoji-button .emoji{font-size:32px;transition:transform .3s cubic-bezier(.4,0,.2,1);line-height:1}.emoji-button:hover:not(:disabled){transform:scale(1.1);border-color:var(--ion-color-medium)}.emoji-button:focus{outline:2px solid var(--ion-color-primary);outline-offset:2px}.emoji-button.selected{transform:scale(1.2)}.emoji-button.selected .emoji{transform:scale(1.1)}.emoji-button.selected.negative{border-color:var(--ion-color-danger);background:var(--ion-color-danger-tint);box-shadow:0 4px 12px rgba(var(--ion-color-danger-rgb),.3)}.emoji-button.selected.neutral{border-color:var(--ion-color-warning);background:var(--ion-color-warning-tint);box-shadow:0 4px 12px rgba(var(--ion-color-warning-rgb),.3)}.emoji-button.selected.positive{border-color:var(--ion-color-success);background:var(--ion-color-success-tint);box-shadow:0 4px 12px rgba(var(--ion-color-success-rgb),.3)}.emoji-button:disabled{cursor:not-allowed;opacity:.5}.comment-section{margin-top:16px;animation:slideIn .3s ease-out}.comment-section .comment-textarea{--background: var(--ion-color-light);--border-radius: 8px;--padding-start: 12px;--padding-end: 12px;width:100%}.comment-section .char-count{display:block;text-align:right;font-size:12px;color:var(--ion-color-medium);margin-top:4px}.submit-button{margin-top:16px;--border-radius: 8px}.submitted-message{margin-top:16px;color:var(--ion-color-success);font-weight:500;animation:fadeIn .3s ease-out}.error-message{margin-top:8px;color:var(--ion-color-danger);font-size:14px}.loading-container{display:flex;justify-content:center;padding:20px}@keyframes slideIn{0%{opacity:0;transform:translateY(-10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { 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: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: IonTextarea, selector: "ion-textarea", inputs: ["autoGrow", "autocapitalize", "autofocus", "clearOnEdit", "color", "cols", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "maxlength", "minlength", "mode", "name", "placeholder", "readonly", "required", "rows", "shape", "spellcheck", "value", "wrap"] }] }); }
|
|
180
|
+
}
|
|
181
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ContentReactionComponent, decorators: [{
|
|
182
|
+
type: Component,
|
|
183
|
+
args: [{ selector: 'val-content-reaction', standalone: true, imports: [CommonModule, FormsModule, IonButton, IonSpinner, IonTextarea], template: "<div\n class=\"content-reaction\"\n [class.disabled]=\"resolvedProps().disabled\"\n [class.readonly]=\"resolvedProps().readonly\"\n>\n <!-- Loading inicial -->\n @if (state().isLoading && !state().selectedValue) {\n <div class=\"loading-container\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n </div>\n } @else {\n <!-- Pregunta -->\n <p class=\"question\">{{ resolvedProps().question }}</p>\n\n <!-- Emojis -->\n <div class=\"emoji-container\">\n @for (value of reactionValues; track value; let i = $index) {\n <button\n type=\"button\"\n class=\"emoji-button\"\n [class.selected]=\"isSelected(value)\"\n [class.negative]=\"value === 'negative' && isSelected(value)\"\n [class.neutral]=\"value === 'neutral' && isSelected(value)\"\n [class.positive]=\"value === 'positive' && isSelected(value)\"\n [attr.aria-label]=\"getEmojiLabel(i)\"\n [attr.aria-pressed]=\"isSelected(value)\"\n [disabled]=\"resolvedProps().disabled || resolvedProps().readonly\"\n (click)=\"selectReaction(value)\"\n >\n <span class=\"emoji\">{{ getEmoji(i) }}</span>\n </button>\n }\n </div>\n\n <!-- Campo de comentario (solo si hay selecci\u00F3n) -->\n @if (showCommentField()) {\n <div class=\"comment-section\">\n <ion-textarea\n [value]=\"state().comment\"\n [placeholder]=\"resolvedProps().commentPlaceholder\"\n [maxlength]=\"resolvedProps().maxCommentLength\"\n [disabled]=\"resolvedProps().disabled\"\n [rows]=\"3\"\n class=\"comment-textarea\"\n (ionInput)=\"updateComment($event)\"\n ></ion-textarea>\n <span class=\"char-count\">\n {{ state().comment.length }}/{{ resolvedProps().maxCommentLength }}\n </span>\n </div>\n }\n\n <!-- Bot\u00F3n de env\u00EDo -->\n @if (state().selectedValue && !state().isSubmitted) {\n <ion-button\n expand=\"block\"\n [disabled]=\"!canSubmit()\"\n (click)=\"submitReaction()\"\n class=\"submit-button\"\n >\n @if (state().isLoading) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else {\n {{ state().hadPreviousReaction ? t('update') : t('submit') }}\n }\n </ion-button>\n }\n\n <!-- Mensaje de confirmaci\u00F3n -->\n @if (state().isSubmitted) {\n <p class=\"submitted-message\">\n {{ t('submitted') }}\n </p>\n }\n\n <!-- Error -->\n @if (state().error) {\n <p class=\"error-message\">{{ state().error }}</p>\n }\n }\n</div>\n", styles: [":host{display:block}.content-reaction{padding:16px;text-align:center}.content-reaction.disabled{opacity:.6;pointer-events:none}.content-reaction.readonly{pointer-events:none}.question{font-size:16px;font-weight:500;color:var(--ion-color-dark);margin:0 0 16px}.emoji-container{display:flex;justify-content:center;gap:24px;margin-bottom:20px}.emoji-button{background:transparent;border:2px solid var(--ion-color-light-shade);border-radius:50%;width:64px;height:64px;cursor:pointer;transition:all .3s cubic-bezier(.4,0,.2,1);display:flex;align-items:center;justify-content:center;padding:0}.emoji-button .emoji{font-size:32px;transition:transform .3s cubic-bezier(.4,0,.2,1);line-height:1}.emoji-button:hover:not(:disabled){transform:scale(1.1);border-color:var(--ion-color-medium)}.emoji-button:focus{outline:2px solid var(--ion-color-primary);outline-offset:2px}.emoji-button.selected{transform:scale(1.2)}.emoji-button.selected .emoji{transform:scale(1.1)}.emoji-button.selected.negative{border-color:var(--ion-color-danger);background:var(--ion-color-danger-tint);box-shadow:0 4px 12px rgba(var(--ion-color-danger-rgb),.3)}.emoji-button.selected.neutral{border-color:var(--ion-color-warning);background:var(--ion-color-warning-tint);box-shadow:0 4px 12px rgba(var(--ion-color-warning-rgb),.3)}.emoji-button.selected.positive{border-color:var(--ion-color-success);background:var(--ion-color-success-tint);box-shadow:0 4px 12px rgba(var(--ion-color-success-rgb),.3)}.emoji-button:disabled{cursor:not-allowed;opacity:.5}.comment-section{margin-top:16px;animation:slideIn .3s ease-out}.comment-section .comment-textarea{--background: var(--ion-color-light);--border-radius: 8px;--padding-start: 12px;--padding-end: 12px;width:100%}.comment-section .char-count{display:block;text-align:right;font-size:12px;color:var(--ion-color-medium);margin-top:4px}.submit-button{margin-top:16px;--border-radius: 8px}.submitted-message{margin-top:16px;color:var(--ion-color-success);font-weight:500;animation:fadeIn .3s ease-out}.error-message{margin-top:8px;color:var(--ion-color-danger);font-size:14px}.loading-container{display:flex;justify-content:center;padding:20px}@keyframes slideIn{0%{opacity:0;transform:translateY(-10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}\n"] }]
|
|
184
|
+
}], propDecorators: { props: [{
|
|
185
|
+
type: Input
|
|
186
|
+
}], reactionSubmit: [{
|
|
187
|
+
type: Output
|
|
188
|
+
}], reactionChange: [{
|
|
189
|
+
type: Output
|
|
190
|
+
}] } });
|
|
191
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"content-reaction.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/content-reaction/content-reaction.component.ts","../../../../../../../src/lib/components/molecules/content-reaction/content-reaction.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  inject,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnChanges,\n  SimpleChanges,\n  signal,\n  computed,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport {\n  IonButton,\n  IonSpinner,\n  IonTextarea,\n} from '@ionic/angular/standalone';\nimport { FeedbackService } from '../../../services/feedback/feedback.service';\nimport { ReactionValue } from '../../../services/feedback/types';\nimport { ToastService } from '../../../services/toast.service';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  ContentReactionMetadata,\n  ContentReactionState,\n  ReactionSubmitEvent,\n  ReactionChangeEvent,\n} from './types';\n\n/**\n * Componente para reacciones de contenido con emojis.\n *\n * @example\n * ```html\n * <val-content-reaction\n *   [props]=\"{\n *     entityRef: { entityType: 'article', entityId: 'art-123' },\n *     question: '¿Te fue útil este artículo?'\n *   }\"\n *   (reactionSubmit)=\"onReactionSubmit($event)\"\n * />\n * ```\n */\n@Component({\n  selector: 'val-content-reaction',\n  standalone: true,\n  imports: [CommonModule, FormsModule, IonButton, IonSpinner, IonTextarea],\n  templateUrl: './content-reaction.component.html',\n  styleUrls: ['./content-reaction.component.scss'],\n})\nexport class ContentReactionComponent implements OnInit, OnChanges {\n  private feedbackService = inject(FeedbackService);\n  private toast = inject(ToastService);\n  private i18n = inject(I18nService);\n\n  @Input() props: Partial<ContentReactionMetadata> = {};\n  @Output() reactionSubmit = new EventEmitter<ReactionSubmitEvent>();\n  @Output() reactionChange = new EventEmitter<ReactionChangeEvent>();\n\n  // Estado reactivo\n  state = signal<ContentReactionState>({\n    selectedValue: null,\n    comment: '',\n    isLoading: false,\n    isSubmitted: false,\n    hadPreviousReaction: false,\n    error: null,\n  });\n\n  // Valores por defecto\n  readonly defaultEmojis: [string, string, string] = ['😞', '😐', '😊'];\n  readonly defaultLabels: [string, string, string] = [\n    'No me ayudó',\n    'Regular',\n    'Muy útil',\n  ];\n  readonly reactionValues: ReactionValue[] = ['negative', 'neutral', 'positive'];\n\n  // Computed properties\n  resolvedProps = computed(() => ({\n    entityRef: this.props.entityRef!,\n    question: this.props.question || this.t('question'),\n    showComment: this.props.showComment ?? true,\n    commentPlaceholder: this.props.commentPlaceholder || this.t('commentPlaceholder'),\n    maxCommentLength: this.props.maxCommentLength ?? 500,\n    emojis: this.props.emojis || this.defaultEmojis,\n    emojiLabels: this.props.emojiLabels || this.defaultLabels,\n    showThankYou: this.props.showThankYou ?? true,\n    thankYouMessage: this.props.thankYouMessage || this.t('thankYou'),\n    disabled: this.props.disabled ?? false,\n    readonly: this.props.readonly ?? false,\n  }));\n\n  showCommentField = computed(\n    () =>\n      this.state().selectedValue !== null && this.resolvedProps().showComment\n  );\n\n  canSubmit = computed(\n    () => this.state().selectedValue !== null && !this.state().isLoading\n  );\n\n  ngOnInit() {\n    this.loadPreviousReaction();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['props'] && !changes['props'].firstChange) {\n      this.loadPreviousReaction();\n    }\n  }\n\n  private async loadPreviousReaction(): Promise<void> {\n    if (!this.props.entityRef) return;\n\n    this.state.update((s) => ({ ...s, isLoading: true, error: null }));\n\n    try {\n      const check = await this.feedbackService.checkFeedback(\n        this.props.entityRef.entityType,\n        this.props.entityRef.entityId\n      );\n\n      if (check.hasFeedback && check.reactionValue) {\n        this.state.update((s) => ({\n          ...s,\n          selectedValue: check.reactionValue!,\n          hadPreviousReaction: true,\n          isLoading: false,\n          isSubmitted: true,\n        }));\n      } else {\n        this.state.update((s) => ({ ...s, isLoading: false }));\n      }\n    } catch (error) {\n      console.error('Error loading previous reaction:', error);\n      this.state.update((s) => ({ ...s, isLoading: false }));\n    }\n  }\n\n  selectReaction(value: ReactionValue): void {\n    if (this.resolvedProps().disabled || this.resolvedProps().readonly) return;\n\n    const previousValue = this.state().selectedValue;\n\n    this.state.update((s) => ({\n      ...s,\n      selectedValue: value,\n      isSubmitted: false,\n      error: null,\n    }));\n\n    this.reactionChange.emit({ value, previousValue });\n  }\n\n  async submitReaction(): Promise<void> {\n    const currentState = this.state();\n    const props = this.resolvedProps();\n\n    if (!currentState.selectedValue || props.disabled) return;\n\n    this.state.update((s) => ({ ...s, isLoading: true, error: null }));\n\n    try {\n      await this.feedbackService.createReaction(\n        props.entityRef,\n        currentState.selectedValue,\n        currentState.comment || undefined\n      );\n\n      this.state.update((s) => ({\n        ...s,\n        isLoading: false,\n        isSubmitted: true,\n        hadPreviousReaction: true,\n      }));\n\n      this.reactionSubmit.emit({\n        value: currentState.selectedValue,\n        comment: currentState.comment || undefined,\n        entityRef: props.entityRef,\n        isUpdate: currentState.hadPreviousReaction,\n      });\n\n      if (props.showThankYou) {\n        this.toast.show({\n          message: props.thankYouMessage,\n          duration: 2000,\n          position: 'bottom',\n          color: 'success',\n        });\n      }\n    } catch (error) {\n      console.error('Error submitting reaction:', error);\n      this.state.update((s) => ({\n        ...s,\n        isLoading: false,\n        error: this.t('errorSubmitting'),\n      }));\n\n      this.toast.show({\n        message: this.t('errorSubmitting'),\n        duration: 3000,\n        position: 'bottom',\n        color: 'danger',\n      });\n    }\n  }\n\n  updateComment(event: CustomEvent): void {\n    const value = event.detail.value || '';\n    this.state.update((s) => ({ ...s, comment: value }));\n  }\n\n  getEmoji(index: number): string {\n    return this.resolvedProps().emojis[index];\n  }\n\n  getEmojiLabel(index: number): string {\n    return this.resolvedProps().emojiLabels[index];\n  }\n\n  isSelected(value: ReactionValue): boolean {\n    return this.state().selectedValue === value;\n  }\n\n  t(key: string): string {\n    const translations: Record<string, string> = {\n      question: '¿Te resultó útil este contenido?',\n      commentPlaceholder: 'Cuéntanos más (opcional)...',\n      submit: 'Enviar opinión',\n      update: 'Actualizar opinión',\n      thankYou: '¡Gracias por tu opinión!',\n      submitted: 'Tu opinión ha sido registrada',\n      errorSubmitting: 'Error al enviar. Intenta de nuevo.',\n    };\n    return this.i18n.t(key, 'ContentReaction') || translations[key] || key;\n  }\n}\n","<div\n  class=\"content-reaction\"\n  [class.disabled]=\"resolvedProps().disabled\"\n  [class.readonly]=\"resolvedProps().readonly\"\n>\n  <!-- Loading inicial -->\n  @if (state().isLoading && !state().selectedValue) {\n    <div class=\"loading-container\">\n      <ion-spinner name=\"crescent\"></ion-spinner>\n    </div>\n  } @else {\n    <!-- Pregunta -->\n    <p class=\"question\">{{ resolvedProps().question }}</p>\n\n    <!-- Emojis -->\n    <div class=\"emoji-container\">\n      @for (value of reactionValues; track value; let i = $index) {\n        <button\n          type=\"button\"\n          class=\"emoji-button\"\n          [class.selected]=\"isSelected(value)\"\n          [class.negative]=\"value === 'negative' && isSelected(value)\"\n          [class.neutral]=\"value === 'neutral' && isSelected(value)\"\n          [class.positive]=\"value === 'positive' && isSelected(value)\"\n          [attr.aria-label]=\"getEmojiLabel(i)\"\n          [attr.aria-pressed]=\"isSelected(value)\"\n          [disabled]=\"resolvedProps().disabled || resolvedProps().readonly\"\n          (click)=\"selectReaction(value)\"\n        >\n          <span class=\"emoji\">{{ getEmoji(i) }}</span>\n        </button>\n      }\n    </div>\n\n    <!-- Campo de comentario (solo si hay selección) -->\n    @if (showCommentField()) {\n      <div class=\"comment-section\">\n        <ion-textarea\n          [value]=\"state().comment\"\n          [placeholder]=\"resolvedProps().commentPlaceholder\"\n          [maxlength]=\"resolvedProps().maxCommentLength\"\n          [disabled]=\"resolvedProps().disabled\"\n          [rows]=\"3\"\n          class=\"comment-textarea\"\n          (ionInput)=\"updateComment($event)\"\n        ></ion-textarea>\n        <span class=\"char-count\">\n          {{ state().comment.length }}/{{ resolvedProps().maxCommentLength }}\n        </span>\n      </div>\n    }\n\n    <!-- Botón de envío -->\n    @if (state().selectedValue && !state().isSubmitted) {\n      <ion-button\n        expand=\"block\"\n        [disabled]=\"!canSubmit()\"\n        (click)=\"submitReaction()\"\n        class=\"submit-button\"\n      >\n        @if (state().isLoading) {\n          <ion-spinner name=\"crescent\"></ion-spinner>\n        } @else {\n          {{ state().hadPreviousReaction ? t('update') : t('submit') }}\n        }\n      </ion-button>\n    }\n\n    <!-- Mensaje de confirmación -->\n    @if (state().isSubmitted) {\n      <p class=\"submitted-message\">\n        {{ t('submitted') }}\n      </p>\n    }\n\n    <!-- Error -->\n    @if (state().error) {\n      <p class=\"error-message\">{{ state().error }}</p>\n    }\n  }\n</div>\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2NvbnRlbnQtcmVhY3Rpb24vdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEVudGl0eVJlZiwgUmVhY3Rpb25WYWx1ZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2ZlZWRiYWNrL3R5cGVzJztcblxuLyoqXG4gKiBDb25maWd1cmFjacOzbiBkZWwgY29tcG9uZW50ZSBjb250ZW50LXJlYWN0aW9uLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvbnRlbnRSZWFjdGlvbk1ldGFkYXRhIHtcbiAgLyoqIFJlZmVyZW5jaWEgYSBsYSBlbnRpZGFkIChyZXF1ZXJpZG8pICovXG4gIGVudGl0eVJlZjogRW50aXR5UmVmO1xuXG4gIC8qKiBQcmVndW50YSBhIG1vc3RyYXIgKGRlZmF1bHQ6IFwiwr9UZSByZXN1bHTDsyDDunRpbCBlc3RlIGNvbnRlbmlkbz9cIikgKi9cbiAgcXVlc3Rpb24/OiBzdHJpbmc7XG5cbiAgLyoqIE1vc3RyYXIgY2FtcG8gZGUgY29tZW50YXJpbyAoZGVmYXVsdDogdHJ1ZSkgKi9cbiAgc2hvd0NvbW1lbnQ/OiBib29sZWFuO1xuXG4gIC8qKiBQbGFjZWhvbGRlciBkZWwgY29tZW50YXJpbyAqL1xuICBjb21tZW50UGxhY2Vob2xkZXI/OiBzdHJpbmc7XG5cbiAgLyoqIE3DoXhpbW8gZGUgY2FyYWN0ZXJlcyBkZWwgY29tZW50YXJpbyAoZGVmYXVsdDogNTAwKSAqL1xuICBtYXhDb21tZW50TGVuZ3RoPzogbnVtYmVyO1xuXG4gIC8qKiBFbW9qaXMgcGVyc29uYWxpemFkb3MgW25lZ2F0aXZlLCBuZXV0cmFsLCBwb3NpdGl2ZV0gKi9cbiAgZW1vamlzPzogW3N0cmluZywgc3RyaW5nLCBzdHJpbmddO1xuXG4gIC8qKiBMYWJlbHMgcGFyYSBlbW9qaXMgKGFjY2VzaWJpbGlkYWQpICovXG4gIGVtb2ppTGFiZWxzPzogW3N0cmluZywgc3RyaW5nLCBzdHJpbmddO1xuXG4gIC8qKiBNb3N0cmFyIHRvYXN0IGRlIGFncmFkZWNpbWllbnRvIChkZWZhdWx0OiB0cnVlKSAqL1xuICBzaG93VGhhbmtZb3U/OiBib29sZWFuO1xuXG4gIC8qKiBNZW5zYWplIGRlIGFncmFkZWNpbWllbnRvIHBlcnNvbmFsaXphZG8gKi9cbiAgdGhhbmtZb3VNZXNzYWdlPzogc3RyaW5nO1xuXG4gIC8qKiBEZXNoYWJpbGl0YXIgaW50ZXJhY2Npw7NuICovXG4gIGRpc2FibGVkPzogYm9vbGVhbjtcblxuICAvKiogTW9kbyByZWFkb25seSAoc29sbyBtb3N0cmFyIHNlbGVjY2nDs24gcHJldmlhKSAqL1xuICByZWFkb25seT86IGJvb2xlYW47XG59XG5cbi8qKlxuICogRXN0YWRvIGludGVybm8gZGVsIGNvbXBvbmVudGUuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29udGVudFJlYWN0aW9uU3RhdGUge1xuICAvKiogVmFsb3Igc2VsZWNjaW9uYWRvIGFjdHVhbG1lbnRlICovXG4gIHNlbGVjdGVkVmFsdWU6IFJlYWN0aW9uVmFsdWUgfCBudWxsO1xuXG4gIC8qKiBDb21lbnRhcmlvIGRlbCB1c3VhcmlvICovXG4gIGNvbW1lbnQ6IHN0cmluZztcblxuICAvKiogSW5kaWNhIHNpIGVzdMOhIGNhcmdhbmRvICovXG4gIGlzTG9hZGluZzogYm9vbGVhbjtcblxuICAvKiogSW5kaWNhIHNpIHlhIGVudmnDsyBsYSByZWFjY2nDs24gKi9cbiAgaXNTdWJtaXR0ZWQ6IGJvb2xlYW47XG5cbiAgLyoqIEluZGljYSBzaSB0ZW7DrWEgcmVhY2Npw7NuIHByZXZpYSAqL1xuICBoYWRQcmV2aW91c1JlYWN0aW9uOiBib29sZWFuO1xuXG4gIC8qKiBFcnJvciBtZXNzYWdlIGlmIGFueSAqL1xuICBlcnJvcjogc3RyaW5nIHwgbnVsbDtcbn1cblxuLyoqXG4gKiBFdmVudG8gZW1pdGlkbyBhbCBlbnZpYXIgcmVhY2Npw7NuLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJlYWN0aW9uU3VibWl0RXZlbnQge1xuICB2YWx1ZTogUmVhY3Rpb25WYWx1ZTtcbiAgY29tbWVudD86IHN0cmluZztcbiAgZW50aXR5UmVmOiBFbnRpdHlSZWY7XG4gIGlzVXBkYXRlOiBib29sZWFuO1xufVxuXG4vKipcbiAqIEV2ZW50byBlbWl0aWRvIGFsIGNhbWJpYXIgc2VsZWNjacOzbi5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSZWFjdGlvbkNoYW5nZUV2ZW50IHtcbiAgdmFsdWU6IFJlYWN0aW9uVmFsdWUgfCBudWxsO1xuICBwcmV2aW91c1ZhbHVlOiBSZWFjdGlvblZhbHVlIHwgbnVsbDtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* val-docs-section
|
|
6
|
+
*
|
|
7
|
+
* A semantic section wrapper for documentation pages.
|
|
8
|
+
* Automatically creates headings with IDs for TOC linking.
|
|
9
|
+
*
|
|
10
|
+
* @example Basic usage
|
|
11
|
+
* ```html
|
|
12
|
+
* <val-docs-section [props]="{ id: 'installation', title: 'Installation' }">
|
|
13
|
+
* <p>Install the package using npm...</p>
|
|
14
|
+
* <val-docs-code-example [props]="codeExample"></val-docs-code-example>
|
|
15
|
+
* </val-docs-section>
|
|
16
|
+
* ```
|
|
17
|
+
*
|
|
18
|
+
* @example With level 3 heading
|
|
19
|
+
* ```html
|
|
20
|
+
* <val-docs-section [props]="{ id: 'npm', title: 'Using npm', level: 3 }">
|
|
21
|
+
* <val-docs-code-example [props]="npmExample"></val-docs-code-example>
|
|
22
|
+
* </val-docs-section>
|
|
23
|
+
* ```
|
|
24
|
+
*/
|
|
25
|
+
export class DocsSectionComponent {
|
|
26
|
+
constructor() {
|
|
27
|
+
this.props = { id: '', title: '' };
|
|
28
|
+
}
|
|
29
|
+
get level() {
|
|
30
|
+
return this.props.level ?? 2;
|
|
31
|
+
}
|
|
32
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DocsSectionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
33
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: DocsSectionComponent, isStandalone: true, selector: "val-docs-section", inputs: { props: "props" }, ngImport: i0, template: `
|
|
34
|
+
<section
|
|
35
|
+
[id]="props.id"
|
|
36
|
+
class="docs-section"
|
|
37
|
+
[class]="props.cssClass"
|
|
38
|
+
[class.docs-section--level-2]="level === 2"
|
|
39
|
+
[class.docs-section--level-3]="level === 3"
|
|
40
|
+
>
|
|
41
|
+
@if (level === 2) {
|
|
42
|
+
<h2 class="docs-section__title">{{ props.title }}</h2>
|
|
43
|
+
} @else {
|
|
44
|
+
<h3 class="docs-section__title">{{ props.title }}</h3>
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
@if (props.description) {
|
|
48
|
+
<p class="docs-section__description">{{ props.description }}</p>
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
<div class="docs-section__content">
|
|
52
|
+
<ng-content></ng-content>
|
|
53
|
+
</div>
|
|
54
|
+
</section>
|
|
55
|
+
`, isInline: true, styles: [".docs-section{margin-bottom:2.5rem}.docs-section__title{margin:0 0 1rem;font-weight:600;color:var(--ion-text-color);scroll-margin-top:80px}.docs-section--level-2 .docs-section__title{font-size:1.5rem;padding-bottom:.5rem;border-bottom:1px solid var(--ion-border-color, #e0e0e0)}.docs-section--level-3 .docs-section__title{font-size:1.25rem}.docs-section__description{margin:0 0 1.25rem;color:var(--ion-color-medium);font-size:1rem;line-height:1.6}.docs-section__content>*:last-child{margin-bottom:0}:host-context(.dark) .docs-section--level-2 .docs-section__title,:host-context([color-scheme=\"dark\"]) .docs-section--level-2 .docs-section__title{border-bottom-color:var(--ion-border-color, #333)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
|
|
56
|
+
}
|
|
57
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DocsSectionComponent, decorators: [{
|
|
58
|
+
type: Component,
|
|
59
|
+
args: [{ selector: 'val-docs-section', standalone: true, imports: [CommonModule], template: `
|
|
60
|
+
<section
|
|
61
|
+
[id]="props.id"
|
|
62
|
+
class="docs-section"
|
|
63
|
+
[class]="props.cssClass"
|
|
64
|
+
[class.docs-section--level-2]="level === 2"
|
|
65
|
+
[class.docs-section--level-3]="level === 3"
|
|
66
|
+
>
|
|
67
|
+
@if (level === 2) {
|
|
68
|
+
<h2 class="docs-section__title">{{ props.title }}</h2>
|
|
69
|
+
} @else {
|
|
70
|
+
<h3 class="docs-section__title">{{ props.title }}</h3>
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
@if (props.description) {
|
|
74
|
+
<p class="docs-section__description">{{ props.description }}</p>
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
<div class="docs-section__content">
|
|
78
|
+
<ng-content></ng-content>
|
|
79
|
+
</div>
|
|
80
|
+
</section>
|
|
81
|
+
`, styles: [".docs-section{margin-bottom:2.5rem}.docs-section__title{margin:0 0 1rem;font-weight:600;color:var(--ion-text-color);scroll-margin-top:80px}.docs-section--level-2 .docs-section__title{font-size:1.5rem;padding-bottom:.5rem;border-bottom:1px solid var(--ion-border-color, #e0e0e0)}.docs-section--level-3 .docs-section__title{font-size:1.25rem}.docs-section__description{margin:0 0 1.25rem;color:var(--ion-color-medium);font-size:1rem;line-height:1.6}.docs-section__content>*:last-child{margin-bottom:0}:host-context(.dark) .docs-section--level-2 .docs-section__title,:host-context([color-scheme=\"dark\"]) .docs-section--level-2 .docs-section__title{border-bottom-color:var(--ion-border-color, #333)}\n"] }]
|
|
82
|
+
}], propDecorators: { props: [{
|
|
83
|
+
type: Input
|
|
84
|
+
}] } });
|
|
85
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9jcy1zZWN0aW9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9tb2xlY3VsZXMvZG9jcy1zZWN0aW9uL2RvY3Mtc2VjdGlvbi5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDakQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDOztBQUcvQzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FvQkc7QUFvRUgsTUFBTSxPQUFPLG9CQUFvQjtJQW5FakM7UUFvRVcsVUFBSyxHQUF3QixFQUFFLEVBQUUsRUFBRSxFQUFFLEVBQUUsS0FBSyxFQUFFLEVBQUUsRUFBRSxDQUFDO0tBSzdEO0lBSEMsSUFBSSxLQUFLO1FBQ1AsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssSUFBSSxDQUFDLENBQUM7SUFDL0IsQ0FBQzsrR0FMVSxvQkFBb0I7bUdBQXBCLG9CQUFvQix3R0EvRHJCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JULG93QkF2QlMsWUFBWTs7NEZBZ0VYLG9CQUFvQjtrQkFuRWhDLFNBQVM7K0JBQ0Usa0JBQWtCLGNBQ2hCLElBQUksV0FDUCxDQUFDLFlBQVksQ0FBQyxZQUNiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JUOzhCQTBDUSxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgRG9jc1NlY3Rpb25NZXRhZGF0YSB9IGZyb20gJy4vdHlwZXMnO1xuXG4vKipcbiAqIHZhbC1kb2NzLXNlY3Rpb25cbiAqXG4gKiBBIHNlbWFudGljIHNlY3Rpb24gd3JhcHBlciBmb3IgZG9jdW1lbnRhdGlvbiBwYWdlcy5cbiAqIEF1dG9tYXRpY2FsbHkgY3JlYXRlcyBoZWFkaW5ncyB3aXRoIElEcyBmb3IgVE9DIGxpbmtpbmcuXG4gKlxuICogQGV4YW1wbGUgQmFzaWMgdXNhZ2VcbiAqIGBgYGh0bWxcbiAqIDx2YWwtZG9jcy1zZWN0aW9uIFtwcm9wc109XCJ7IGlkOiAnaW5zdGFsbGF0aW9uJywgdGl0bGU6ICdJbnN0YWxsYXRpb24nIH1cIj5cbiAqICAgPHA+SW5zdGFsbCB0aGUgcGFja2FnZSB1c2luZyBucG0uLi48L3A+XG4gKiAgIDx2YWwtZG9jcy1jb2RlLWV4YW1wbGUgW3Byb3BzXT1cImNvZGVFeGFtcGxlXCI+PC92YWwtZG9jcy1jb2RlLWV4YW1wbGU+XG4gKiA8L3ZhbC1kb2NzLXNlY3Rpb24+XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZSBXaXRoIGxldmVsIDMgaGVhZGluZ1xuICogYGBgaHRtbFxuICogPHZhbC1kb2NzLXNlY3Rpb24gW3Byb3BzXT1cInsgaWQ6ICducG0nLCB0aXRsZTogJ1VzaW5nIG5wbScsIGxldmVsOiAzIH1cIj5cbiAqICAgPHZhbC1kb2NzLWNvZGUtZXhhbXBsZSBbcHJvcHNdPVwibnBtRXhhbXBsZVwiPjwvdmFsLWRvY3MtY29kZS1leGFtcGxlPlxuICogPC92YWwtZG9jcy1zZWN0aW9uPlxuICogYGBgXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC1kb2NzLXNlY3Rpb24nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8c2VjdGlvblxuICAgICAgW2lkXT1cInByb3BzLmlkXCJcbiAgICAgIGNsYXNzPVwiZG9jcy1zZWN0aW9uXCJcbiAgICAgIFtjbGFzc109XCJwcm9wcy5jc3NDbGFzc1wiXG4gICAgICBbY2xhc3MuZG9jcy1zZWN0aW9uLS1sZXZlbC0yXT1cImxldmVsID09PSAyXCJcbiAgICAgIFtjbGFzcy5kb2NzLXNlY3Rpb24tLWxldmVsLTNdPVwibGV2ZWwgPT09IDNcIlxuICAgID5cbiAgICAgIEBpZiAobGV2ZWwgPT09IDIpIHtcbiAgICAgICAgPGgyIGNsYXNzPVwiZG9jcy1zZWN0aW9uX190aXRsZVwiPnt7IHByb3BzLnRpdGxlIH19PC9oMj5cbiAgICAgIH0gQGVsc2Uge1xuICAgICAgICA8aDMgY2xhc3M9XCJkb2NzLXNlY3Rpb25fX3RpdGxlXCI+e3sgcHJvcHMudGl0bGUgfX08L2gzPlxuICAgICAgfVxuXG4gICAgICBAaWYgKHByb3BzLmRlc2NyaXB0aW9uKSB7XG4gICAgICAgIDxwIGNsYXNzPVwiZG9jcy1zZWN0aW9uX19kZXNjcmlwdGlvblwiPnt7IHByb3BzLmRlc2NyaXB0aW9uIH19PC9wPlxuICAgICAgfVxuXG4gICAgICA8ZGl2IGNsYXNzPVwiZG9jcy1zZWN0aW9uX19jb250ZW50XCI+XG4gICAgICAgIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbiAgICAgIDwvZGl2PlxuICAgIDwvc2VjdGlvbj5cbiAgYCxcbiAgc3R5bGVzOiBbYFxuICAgIC5kb2NzLXNlY3Rpb24ge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMi41cmVtO1xuICAgIH1cblxuICAgIC5kb2NzLXNlY3Rpb25fX3RpdGxlIHtcbiAgICAgIG1hcmdpbjogMCAwIDFyZW07XG4gICAgICBmb250LXdlaWdodDogNjAwO1xuICAgICAgY29sb3I6IHZhcigtLWlvbi10ZXh0LWNvbG9yKTtcbiAgICAgIHNjcm9sbC1tYXJnaW4tdG9wOiA4MHB4O1xuICAgIH1cblxuICAgIC5kb2NzLXNlY3Rpb24tLWxldmVsLTIgLmRvY3Mtc2VjdGlvbl9fdGl0bGUge1xuICAgICAgZm9udC1zaXplOiAxLjVyZW07XG4gICAgICBwYWRkaW5nLWJvdHRvbTogMC41cmVtO1xuICAgICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkIHZhcigtLWlvbi1ib3JkZXItY29sb3IsICNlMGUwZTApO1xuICAgIH1cblxuICAgIC5kb2NzLXNlY3Rpb24tLWxldmVsLTMgLmRvY3Mtc2VjdGlvbl9fdGl0bGUge1xuICAgICAgZm9udC1zaXplOiAxLjI1cmVtO1xuICAgIH1cblxuICAgIC5kb2NzLXNlY3Rpb25fX2Rlc2NyaXB0aW9uIHtcbiAgICAgIG1hcmdpbjogMCAwIDEuMjVyZW07XG4gICAgICBjb2xvcjogdmFyKC0taW9uLWNvbG9yLW1lZGl1bSk7XG4gICAgICBmb250LXNpemU6IDFyZW07XG4gICAgICBsaW5lLWhlaWdodDogMS42O1xuICAgIH1cblxuICAgIC5kb2NzLXNlY3Rpb25fX2NvbnRlbnQgPiAqOmxhc3QtY2hpbGQge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMDtcbiAgICB9XG5cbiAgICAvKiBEYXJrIG1vZGUgKi9cbiAgICA6aG9zdC1jb250ZXh0KC5kYXJrKSAuZG9jcy1zZWN0aW9uLS1sZXZlbC0yIC5kb2NzLXNlY3Rpb25fX3RpdGxlLFxuICAgIDpob3N0LWNvbnRleHQoW2NvbG9yLXNjaGVtZT1cImRhcmtcIl0pIC5kb2NzLXNlY3Rpb24tLWxldmVsLTIgLmRvY3Mtc2VjdGlvbl9fdGl0bGUge1xuICAgICAgYm9yZGVyLWJvdHRvbS1jb2xvcjogdmFyKC0taW9uLWJvcmRlci1jb2xvciwgIzMzMyk7XG4gICAgfVxuICBgXSxcbn0pXG5leHBvcnQgY2xhc3MgRG9jc1NlY3Rpb25Db21wb25lbnQge1xuICBASW5wdXQoKSBwcm9wczogRG9jc1NlY3Rpb25NZXRhZGF0YSA9IHsgaWQ6ICcnLCB0aXRsZTogJycgfTtcblxuICBnZXQgbGV2ZWwoKTogMiB8IDMge1xuICAgIHJldHVybiB0aGlzLnByb3BzLmxldmVsID8/IDI7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2RvY3Mtc2VjdGlvbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgZG9jcy1zZWN0aW9uIGNvbXBvbmVudC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEb2NzU2VjdGlvbk1ldGFkYXRhIHtcbiAgLyoqXG4gICAqIFVuaXF1ZSBJRCBmb3IgdGhlIHNlY3Rpb24gKHVzZWQgZm9yIFRPQyBsaW5raW5nKS5cbiAgICogV2lsbCBiZSBhcHBsaWVkIHRvIHRoZSBzZWN0aW9uIGVsZW1lbnQgZm9yIGFuY2hvciBuYXZpZ2F0aW9uLlxuICAgKi9cbiAgaWQ6IHN0cmluZztcblxuICAvKipcbiAgICogU2VjdGlvbiB0aXRsZSBkaXNwbGF5ZWQgYXMgaGVhZGluZy5cbiAgICovXG4gIHRpdGxlOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEhlYWRpbmcgbGV2ZWwgZm9yIHRoZSBzZWN0aW9uLlxuICAgKiBAZGVmYXVsdCAyXG4gICAqL1xuICBsZXZlbD86IDIgfCAzO1xuXG4gIC8qKlxuICAgKiBPcHRpb25hbCBkZXNjcmlwdGlvbiBiZWxvdyB0aGUgdGl0bGUuXG4gICAqL1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcblxuICAvKipcbiAgICogQ3VzdG9tIENTUyBjbGFzcy5cbiAgICovXG4gIGNzc0NsYXNzPzogc3RyaW5nO1xufVxuIl19
|