valtech-components 2.0.293 → 2.0.294

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.
@@ -0,0 +1,219 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
3
+ import { addIcons } from 'ionicons';
4
+ import { globe, language } from 'ionicons/icons';
5
+ import { Subscription, of } from 'rxjs';
6
+ import { map } from 'rxjs/operators';
7
+ import { PopoverSelectorComponent } from '../popover-selector/popover-selector.component';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "../../../services/lang-provider/lang-provider.service";
10
+ import * as i2 from "../../../services/content.service";
11
+ /**
12
+ * val-language-selector
13
+ *
14
+ * A specialized language selector component that integrates with the LangService.
15
+ * Uses the popover-selector component internally and provides language-specific functionality.
16
+ *
17
+ * @example
18
+ * // Basic usage (auto-detects languages from LangService)
19
+ * <val-language-selector
20
+ * [props]="{ showLabel: true, showFlags: true }">
21
+ * </val-language-selector>
22
+ *
23
+ * @example
24
+ * // Custom configuration
25
+ * const languageProps: LanguageSelectorMetadata = {
26
+ * showLabel: true,
27
+ * label: 'Choose Language',
28
+ * showFlags: true,
29
+ * color: 'primary',
30
+ * size: 'large',
31
+ * fill: 'outline',
32
+ * customLanguageNames: {
33
+ * 'es': 'Español',
34
+ * 'en': 'English',
35
+ * 'fr': 'Français'
36
+ * }
37
+ * };
38
+ *
39
+ * @example
40
+ * // With reactive content
41
+ * const languageProps: LanguageSelectorMetadata = {
42
+ * showLabel: true,
43
+ * labelConfig: {
44
+ * className: 'myComponent',
45
+ * key: 'languageLabel',
46
+ * fallback: 'Language'
47
+ * },
48
+ * color: 'tertiary'
49
+ * };
50
+ *
51
+ * @input props: LanguageSelectorMetadata - Configuration for the language selector
52
+ * @output languageChange: EventEmitter<string> - Emitted when language changes
53
+ */
54
+ export class LanguageSelectorComponent {
55
+ constructor(langService, contentService) {
56
+ this.langService = langService;
57
+ this.contentService = contentService;
58
+ /**
59
+ * Language selector configuration object.
60
+ * @type {LanguageSelectorMetadata}
61
+ */
62
+ this.props = {};
63
+ /**
64
+ * Event emitted when the language selection changes.
65
+ * Emits the selected language code.
66
+ */
67
+ this.languageChange = new EventEmitter();
68
+ this.subscriptions = new Subscription();
69
+ /** Default language display names */
70
+ this.defaultLanguageNames = {
71
+ es: 'Español',
72
+ en: 'English',
73
+ fr: 'Français',
74
+ de: 'Deutsch',
75
+ pt: 'Português',
76
+ it: 'Italiano',
77
+ zh: '中文',
78
+ ja: '日本語',
79
+ ko: '한국어',
80
+ ru: 'Русский',
81
+ ar: 'العربية',
82
+ };
83
+ /** Default flag icons for languages */
84
+ this.defaultLanguageFlags = {
85
+ es: '🇪🇸',
86
+ en: '🇺🇸',
87
+ fr: '🇫🇷',
88
+ de: '🇩🇪',
89
+ pt: '🇵🇹',
90
+ it: '🇮🇹',
91
+ zh: '🇨🇳',
92
+ ja: '🇯🇵',
93
+ ko: '🇰🇷',
94
+ ru: '🇷🇺',
95
+ ar: '🇸🇦',
96
+ };
97
+ // Register required icons
98
+ addIcons({ language, globe });
99
+ }
100
+ ngOnInit() {
101
+ this.initializeLanguageState();
102
+ this.initializeLabel();
103
+ this.initializePopoverProps();
104
+ }
105
+ ngOnDestroy() {
106
+ this.subscriptions.unsubscribe();
107
+ }
108
+ initializeLanguageState() {
109
+ // Get current language and available languages from LangService
110
+ this.currentLanguage$ = this.langService.currentLang$;
111
+ this.availableLanguages$ = this.currentLanguage$.pipe(map(() => this.props.availableLanguages || this.langService.availableLangs));
112
+ }
113
+ initializeLabel() {
114
+ // Initialize label observable based on configuration
115
+ if (this.props.label !== undefined) {
116
+ // Static label
117
+ this.label$ = of(this.props.label);
118
+ }
119
+ else if (this.props.labelConfig) {
120
+ // Reactive label
121
+ this.label$ = this.contentService.fromContent({
122
+ className: this.props.labelConfig.className || '_global',
123
+ key: this.props.labelConfig.key,
124
+ fallback: this.props.labelConfig.fallback || 'Language',
125
+ interpolation: this.props.labelConfig.interpolation,
126
+ });
127
+ }
128
+ else {
129
+ // Default label from global content
130
+ this.label$ = this.contentService.fromContent({
131
+ className: '_global',
132
+ key: 'language',
133
+ fallback: 'Idioma',
134
+ });
135
+ }
136
+ }
137
+ initializePopoverProps() {
138
+ // Subscribe to language state changes to update popover props
139
+ const languageSubscription = this.availableLanguages$.subscribe(availableLanguages => {
140
+ const currentLanguageSubscription = this.currentLanguage$.subscribe(currentLanguage => {
141
+ this.updatePopoverProps(availableLanguages, currentLanguage);
142
+ });
143
+ this.subscriptions.add(currentLanguageSubscription);
144
+ });
145
+ this.subscriptions.add(languageSubscription);
146
+ // Subscribe to label changes
147
+ const labelSubscription = this.label$.subscribe(label => {
148
+ if (this.popoverProps) {
149
+ this.popoverProps = {
150
+ ...this.popoverProps,
151
+ label: this.props.showLabel !== false ? label : undefined,
152
+ };
153
+ }
154
+ });
155
+ this.subscriptions.add(labelSubscription);
156
+ }
157
+ updatePopoverProps(availableLanguages, currentLanguage) {
158
+ // Convert language codes to popover options
159
+ const options = availableLanguages.map(lang => ({
160
+ value: lang,
161
+ label: this.getLanguageDisplayName(lang),
162
+ icon: this.props.showFlags ? undefined : undefined, // We'll use text flags instead
163
+ }));
164
+ // Create popover configuration
165
+ this.popoverProps = {
166
+ options,
167
+ selectedValue: currentLanguage,
168
+ label: this.props.showLabel !== false ? '' : undefined, // Will be set by label subscription
169
+ icon: 'language',
170
+ placeholder: 'Seleccionar idioma...',
171
+ color: this.props.color || 'medium',
172
+ size: this.props.size || 'default',
173
+ fill: this.props.fill || 'outline',
174
+ shape: this.props.shape,
175
+ expand: this.props.expand,
176
+ disabled: this.props.disabled || false,
177
+ interface: 'popover',
178
+ showCheckmark: true,
179
+ multiple: false,
180
+ cancelText: 'Cancelar',
181
+ okText: 'Aceptar',
182
+ };
183
+ }
184
+ getLanguageDisplayName(languageCode) {
185
+ // Use custom names if provided, otherwise use defaults, otherwise use the code itself
186
+ const customName = this.props.customLanguageNames?.[languageCode];
187
+ const defaultName = this.defaultLanguageNames[languageCode];
188
+ const flag = this.props.showFlags ? this.defaultLanguageFlags[languageCode] : '';
189
+ const displayName = customName || defaultName || languageCode.toUpperCase();
190
+ return flag ? `${flag} ${displayName}` : displayName;
191
+ }
192
+ /**
193
+ * Handle language selection change.
194
+ * @param selectedLanguage - The selected language code(s)
195
+ */
196
+ onLanguageChange(selectedLanguage) {
197
+ if (typeof selectedLanguage === 'string') {
198
+ // Update the language service
199
+ this.langService.setLang(selectedLanguage);
200
+ // Emit the change event
201
+ this.languageChange.emit(selectedLanguage);
202
+ }
203
+ }
204
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LanguageSelectorComponent, deps: [{ token: i1.LangService }, { token: i2.ContentService }], target: i0.ɵɵFactoryTarget.Component }); }
205
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: LanguageSelectorComponent, isStandalone: true, selector: "val-language-selector", inputs: { props: "props" }, outputs: { languageChange: "languageChange" }, ngImport: i0, template: `
206
+ <val-popover-selector [props]="popoverProps" (selectionChange)="onLanguageChange($event)"> </val-popover-selector>
207
+ `, isInline: true, styles: [":host{display:block}val-popover-selector .selector-trigger .trigger-text{display:flex;align-items:center;gap:8px}.language-flag{font-size:1.2em;margin-right:4px}@media (max-width: 768px){val-popover-selector .selector-trigger{--padding-start: 8px;--padding-end: 8px}val-popover-selector .selector-trigger .trigger-text{font-size:14px}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: PopoverSelectorComponent, selector: "val-popover-selector", inputs: ["props"], outputs: ["selectionChange"] }] }); }
208
+ }
209
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LanguageSelectorComponent, decorators: [{
210
+ type: Component,
211
+ args: [{ selector: 'val-language-selector', standalone: true, imports: [CommonModule, PopoverSelectorComponent], template: `
212
+ <val-popover-selector [props]="popoverProps" (selectionChange)="onLanguageChange($event)"> </val-popover-selector>
213
+ `, styles: [":host{display:block}val-popover-selector .selector-trigger .trigger-text{display:flex;align-items:center;gap:8px}.language-flag{font-size:1.2em;margin-right:4px}@media (max-width: 768px){val-popover-selector .selector-trigger{--padding-start: 8px;--padding-end: 8px}val-popover-selector .selector-trigger .trigger-text{font-size:14px}}\n"] }]
214
+ }], ctorParameters: () => [{ type: i1.LangService }, { type: i2.ContentService }], propDecorators: { props: [{
215
+ type: Input
216
+ }], languageChange: [{
217
+ type: Output
218
+ }] } });
219
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"language-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/language-selector/language-selector.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { addIcons } from 'ionicons';\nimport { globe, language } from 'ionicons/icons';\nimport { Observable, Subscription, of } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { ContentService } from '../../../services/content.service';\nimport { LangService } from '../../../services/lang-provider/lang-provider.service';\nimport { LangOption } from '../../../services/lang-provider/types';\nimport { PopoverSelectorComponent } from '../popover-selector/popover-selector.component';\nimport { PopoverOption, PopoverSelectorMetadata } from '../popover-selector/types';\nimport { LanguageSelectorMetadata } from './types';\n\n@Component({\n  selector: 'val-language-selector',\n  standalone: true,\n  imports: [CommonModule, PopoverSelectorComponent],\n  template: `\n    <val-popover-selector [props]=\"popoverProps\" (selectionChange)=\"onLanguageChange($event)\"> </val-popover-selector>\n  `,\n  styleUrls: ['./language-selector.component.scss'],\n})\n/**\n * val-language-selector\n *\n * A specialized language selector component that integrates with the LangService.\n * Uses the popover-selector component internally and provides language-specific functionality.\n *\n * @example\n * // Basic usage (auto-detects languages from LangService)\n * <val-language-selector\n *   [props]=\"{ showLabel: true, showFlags: true }\">\n * </val-language-selector>\n *\n * @example\n * // Custom configuration\n * const languageProps: LanguageSelectorMetadata = {\n *   showLabel: true,\n *   label: 'Choose Language',\n *   showFlags: true,\n *   color: 'primary',\n *   size: 'large',\n *   fill: 'outline',\n *   customLanguageNames: {\n *     'es': 'Español',\n *     'en': 'English',\n *     'fr': 'Français'\n *   }\n * };\n *\n * @example\n * // With reactive content\n * const languageProps: LanguageSelectorMetadata = {\n *   showLabel: true,\n *   labelConfig: {\n *     className: 'myComponent',\n *     key: 'languageLabel',\n *     fallback: 'Language'\n *   },\n *   color: 'tertiary'\n * };\n *\n * @input props: LanguageSelectorMetadata - Configuration for the language selector\n * @output languageChange: EventEmitter<string> - Emitted when language changes\n */\nexport class LanguageSelectorComponent implements OnInit, OnDestroy {\n  /**\n   * Language selector configuration object.\n   * @type {LanguageSelectorMetadata}\n   */\n  @Input()\n  props: LanguageSelectorMetadata = {};\n\n  /**\n   * Event emitted when the language selection changes.\n   * Emits the selected language code.\n   */\n  @Output()\n  languageChange = new EventEmitter<string>();\n\n  /** Popover selector configuration */\n  popoverProps: PopoverSelectorMetadata;\n\n  /** Current language observable */\n  currentLanguage$: Observable<LangOption>;\n\n  /** Available languages observable */\n  availableLanguages$: Observable<LangOption[]>;\n\n  /** Label observable for reactive content */\n  label$: Observable<string>;\n\n  private subscriptions = new Subscription();\n\n  /** Default language display names */\n  private readonly defaultLanguageNames: Record<string, string> = {\n    es: 'Español',\n    en: 'English',\n    fr: 'Français',\n    de: 'Deutsch',\n    pt: 'Português',\n    it: 'Italiano',\n    zh: '中文',\n    ja: '日本語',\n    ko: '한국어',\n    ru: 'Русский',\n    ar: 'العربية',\n  };\n\n  /** Default flag icons for languages */\n  private readonly defaultLanguageFlags: Record<string, string> = {\n    es: '🇪🇸',\n    en: '🇺🇸',\n    fr: '🇫🇷',\n    de: '🇩🇪',\n    pt: '🇵🇹',\n    it: '🇮🇹',\n    zh: '🇨🇳',\n    ja: '🇯🇵',\n    ko: '🇰🇷',\n    ru: '🇷🇺',\n    ar: '🇸🇦',\n  };\n\n  constructor(\n    private langService: LangService,\n    private contentService: ContentService\n  ) {\n    // Register required icons\n    addIcons({ language, globe });\n  }\n\n  ngOnInit() {\n    this.initializeLanguageState();\n    this.initializeLabel();\n    this.initializePopoverProps();\n  }\n\n  ngOnDestroy() {\n    this.subscriptions.unsubscribe();\n  }\n\n  private initializeLanguageState() {\n    // Get current language and available languages from LangService\n    this.currentLanguage$ = this.langService.currentLang$;\n    this.availableLanguages$ = this.currentLanguage$.pipe(\n      map(() => this.props.availableLanguages || this.langService.availableLangs)\n    );\n  }\n\n  private initializeLabel() {\n    // Initialize label observable based on configuration\n    if (this.props.label !== undefined) {\n      // Static label\n      this.label$ = of(this.props.label);\n    } else if (this.props.labelConfig) {\n      // Reactive label\n      this.label$ = this.contentService.fromContent({\n        className: this.props.labelConfig.className || '_global',\n        key: this.props.labelConfig.key,\n        fallback: this.props.labelConfig.fallback || 'Language',\n        interpolation: this.props.labelConfig.interpolation,\n      });\n    } else {\n      // Default label from global content\n      this.label$ = this.contentService.fromContent({\n        className: '_global',\n        key: 'language',\n        fallback: 'Idioma',\n      });\n    }\n  }\n\n  private initializePopoverProps() {\n    // Subscribe to language state changes to update popover props\n    const languageSubscription = this.availableLanguages$.subscribe(availableLanguages => {\n      const currentLanguageSubscription = this.currentLanguage$.subscribe(currentLanguage => {\n        this.updatePopoverProps(availableLanguages, currentLanguage);\n      });\n      this.subscriptions.add(currentLanguageSubscription);\n    });\n    this.subscriptions.add(languageSubscription);\n\n    // Subscribe to label changes\n    const labelSubscription = this.label$.subscribe(label => {\n      if (this.popoverProps) {\n        this.popoverProps = {\n          ...this.popoverProps,\n          label: this.props.showLabel !== false ? label : undefined,\n        };\n      }\n    });\n    this.subscriptions.add(labelSubscription);\n  }\n\n  private updatePopoverProps(availableLanguages: LangOption[], currentLanguage: LangOption) {\n    // Convert language codes to popover options\n    const options: PopoverOption[] = availableLanguages.map(lang => ({\n      value: lang,\n      label: this.getLanguageDisplayName(lang),\n      icon: this.props.showFlags ? undefined : undefined, // We'll use text flags instead\n    }));\n\n    // Create popover configuration\n    this.popoverProps = {\n      options,\n      selectedValue: currentLanguage,\n      label: this.props.showLabel !== false ? '' : undefined, // Will be set by label subscription\n      icon: 'language',\n      placeholder: 'Seleccionar idioma...',\n      color: this.props.color || 'medium',\n      size: this.props.size || 'default',\n      fill: this.props.fill || 'outline',\n      shape: this.props.shape,\n      expand: this.props.expand,\n      disabled: this.props.disabled || false,\n      interface: 'popover',\n      showCheckmark: true,\n      multiple: false,\n      cancelText: 'Cancelar',\n      okText: 'Aceptar',\n    };\n  }\n\n  private getLanguageDisplayName(languageCode: string): string {\n    // Use custom names if provided, otherwise use defaults, otherwise use the code itself\n    const customName = this.props.customLanguageNames?.[languageCode];\n    const defaultName = this.defaultLanguageNames[languageCode];\n    const flag = this.props.showFlags ? this.defaultLanguageFlags[languageCode] : '';\n\n    const displayName = customName || defaultName || languageCode.toUpperCase();\n    return flag ? `${flag} ${displayName}` : displayName;\n  }\n\n  /**\n   * Handle language selection change.\n   * @param selectedLanguage - The selected language code(s)\n   */\n  onLanguageChange(selectedLanguage: string | string[]) {\n    if (typeof selectedLanguage === 'string') {\n      // Update the language service\n      this.langService.setLang(selectedLanguage);\n\n      // Emit the change event\n      this.languageChange.emit(selectedLanguage);\n    }\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9jb21wb25lbnRzL21vbGVjdWxlcy9sYW5ndWFnZS1zZWxlY3Rvci90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29sb3IgfSBmcm9tICdAaW9uaWMvY29yZSc7XG5cbi8qKlxuICogUHJvcHMgZm9yIHZhbC1sYW5ndWFnZS1zZWxlY3RvciBjb21wb25lbnQuXG4gKiBBIHNwZWNpYWxpemVkIGNvbXBvbmVudCBmb3IgbGFuZ3VhZ2Ugc2VsZWN0aW9uLlxuICpcbiAqIEBwcm9wZXJ0eSBjdXJyZW50TGFuZ3VhZ2UgLSBDdXJyZW50bHkgc2VsZWN0ZWQgbGFuZ3VhZ2UgY29kZS5cbiAqIEBwcm9wZXJ0eSBhdmFpbGFibGVMYW5ndWFnZXMgLSBBcnJheSBvZiBhdmFpbGFibGUgbGFuZ3VhZ2UgY29kZXMuXG4gKiBAcHJvcGVydHkgc2hvd0xhYmVsIC0gV2hldGhlciB0byBzaG93IHRoZSBsYWJlbC5cbiAqIEBwcm9wZXJ0eSBsYWJlbCAtIEN1c3RvbSBsYWJlbCB0ZXh0IChzdGF0aWMpLlxuICogQHByb3BlcnR5IGxhYmVsQ29uZmlnIC0gUmVhY3RpdmUgY29udGVudCBjb25maWd1cmF0aW9uIGZvciBsYWJlbC5cbiAqIEBwcm9wZXJ0eSBzaG93RmxhZ3MgLSBXaGV0aGVyIHRvIHNob3cgZmxhZyBpY29ucyBmb3IgbGFuZ3VhZ2VzLlxuICogQHByb3BlcnR5IGNvbG9yIC0gQnV0dG9uIGNvbG9yIChJb25pYyBjb2xvciBzdHJpbmcpLlxuICogQHByb3BlcnR5IHNpemUgLSBCdXR0b24gc2l6ZSAoJ3NtYWxsJyB8ICdkZWZhdWx0JyB8ICdsYXJnZScpLlxuICogQHByb3BlcnR5IGZpbGwgLSBCdXR0b24gZmlsbCBzdHlsZSAoJ2NsZWFyJyB8ICdvdXRsaW5lJyB8ICdzb2xpZCcgfCAnZGVmYXVsdCcpLlxuICogQHByb3BlcnR5IHNoYXBlIC0gQnV0dG9uIHNoYXBlICgncm91bmQnIHwgdW5kZWZpbmVkKS5cbiAqIEBwcm9wZXJ0eSBleHBhbmQgLSBCdXR0b24gZXhwYW5zaW9uICgnZnVsbCcgfCAnYmxvY2snIHwgdW5kZWZpbmVkKS5cbiAqIEBwcm9wZXJ0eSBkaXNhYmxlZCAtIFdoZXRoZXIgdGhlIHNlbGVjdG9yIGlzIGRpc2FibGVkLlxuICogQHByb3BlcnR5IGN1c3RvbUxhbmd1YWdlTmFtZXMgLSBDdXN0b20gZGlzcGxheSBuYW1lcyBmb3IgbGFuZ3VhZ2VzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIExhbmd1YWdlU2VsZWN0b3JNZXRhZGF0YSB7XG4gIC8qKiBDdXJyZW50bHkgc2VsZWN0ZWQgbGFuZ3VhZ2UgY29kZSAqL1xuICBjdXJyZW50TGFuZ3VhZ2U/OiBzdHJpbmc7XG4gIC8qKiBBcnJheSBvZiBhdmFpbGFibGUgbGFuZ3VhZ2UgY29kZXMgKi9cbiAgYXZhaWxhYmxlTGFuZ3VhZ2VzPzogc3RyaW5nW107XG4gIC8qKiBXaGV0aGVyIHRvIHNob3cgdGhlIGxhYmVsICovXG4gIHNob3dMYWJlbD86IGJvb2xlYW47XG4gIC8qKiBTdGF0aWMgbGFiZWwgdGV4dCAodGFrZXMgcHJlY2VkZW5jZSBvdmVyIGxhYmVsQ29uZmlnKSAqL1xuICBsYWJlbD86IHN0cmluZztcbiAgLyoqIFJlYWN0aXZlIGNvbnRlbnQgY29uZmlndXJhdGlvbiBmb3IgbGFiZWwgKi9cbiAgbGFiZWxDb25maWc/OiB7XG4gICAgY2xhc3NOYW1lPzogc3RyaW5nO1xuICAgIGtleTogc3RyaW5nO1xuICAgIGZhbGxiYWNrPzogc3RyaW5nO1xuICAgIGludGVycG9sYXRpb24/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICB9O1xuICAvKiogV2hldGhlciB0byBzaG93IGZsYWcgaWNvbnMgZm9yIGxhbmd1YWdlcyAqL1xuICBzaG93RmxhZ3M/OiBib29sZWFuO1xuICAvKiogQnV0dG9uIGNvbG9yICovXG4gIGNvbG9yPzogQ29sb3I7XG4gIC8qKiBCdXR0b24gc2l6ZSAqL1xuICBzaXplPzogJ3NtYWxsJyB8ICdkZWZhdWx0JyB8ICdsYXJnZSc7XG4gIC8qKiBCdXR0b24gZmlsbCBzdHlsZSAqL1xuICBmaWxsPzogJ2NsZWFyJyB8ICdvdXRsaW5lJyB8ICdzb2xpZCcgfCAnZGVmYXVsdCc7XG4gIC8qKiBCdXR0b24gc2hhcGUgKi9cbiAgc2hhcGU/OiAncm91bmQnO1xuICAvKiogQnV0dG9uIGV4cGFuc2lvbiAqL1xuICBleHBhbmQ/OiAnZnVsbCcgfCAnYmxvY2snO1xuICAvKiogV2hldGhlciB0aGUgc2VsZWN0b3IgaXMgZGlzYWJsZWQgKi9cbiAgZGlzYWJsZWQ/OiBib29sZWFuO1xuICAvKiogQ3VzdG9tIGRpc3BsYXkgbmFtZXMgZm9yIGxhbmd1YWdlcyAqL1xuICBjdXN0b21MYW5ndWFnZU5hbWVzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbn1cbiJdfQ==
@@ -0,0 +1,170 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
3
+ import { IonButton, IonIcon, IonItem, IonLabel, IonSelect, IonSelectOption } from '@ionic/angular/standalone';
4
+ import { addIcons } from 'ionicons';
5
+ import { chevronDown } from 'ionicons/icons';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common";
8
+ /**
9
+ * val-popover-selector
10
+ *
11
+ * A generic reusable popover selector component that can be used for any type of selection.
12
+ * Provides a button trigger that opens a popover with selectable options.
13
+ *
14
+ * @example
15
+ * // Basic usage
16
+ * <val-popover-selector
17
+ * [props]="selectorProps"
18
+ * (selectionChange)="onSelectionChange($event)">
19
+ * </val-popover-selector>
20
+ *
21
+ * @example
22
+ * // With icon and custom styling
23
+ * const selectorProps: PopoverSelectorMetadata = {
24
+ * options: [
25
+ * { value: 'option1', label: 'Option 1', icon: 'star' },
26
+ * { value: 'option2', label: 'Option 2', icon: 'heart' }
27
+ * ],
28
+ * selectedValue: 'option1',
29
+ * label: 'Choose option',
30
+ * icon: 'settings',
31
+ * color: 'primary',
32
+ * size: 'large',
33
+ * fill: 'solid'
34
+ * };
35
+ *
36
+ * @input props: PopoverSelectorMetadata - Configuration for the selector
37
+ * @output selectionChange: EventEmitter<string | string[]> - Emitted when selection changes
38
+ */
39
+ export class PopoverSelectorComponent {
40
+ constructor() {
41
+ /**
42
+ * Event emitted when the selection changes.
43
+ * Emits the selected value(s).
44
+ */
45
+ this.selectionChange = new EventEmitter();
46
+ // Register required icons
47
+ addIcons({ chevronDown });
48
+ }
49
+ /**
50
+ * Handle selection change from the ion-select.
51
+ * @param event - The ion-select change event
52
+ */
53
+ onSelectionChange(event) {
54
+ const selectedValue = event.detail.value;
55
+ this.selectionChange.emit(selectedValue);
56
+ }
57
+ /**
58
+ * Get the display text for the trigger button.
59
+ * Shows the selected option's label or placeholder.
60
+ * @returns The display text
61
+ */
62
+ getDisplayText() {
63
+ if (!this.props.selectedValue) {
64
+ return this.props.placeholder || 'Seleccionar...';
65
+ }
66
+ if (Array.isArray(this.props.selectedValue)) {
67
+ // Multiple selection
68
+ if (this.props.selectedValue.length === 0) {
69
+ return this.props.placeholder || 'Seleccionar...';
70
+ }
71
+ if (this.props.selectedValue.length === 1) {
72
+ const option = this.props.options.find(opt => opt.value === this.props.selectedValue[0]);
73
+ return option?.label || this.props.selectedValue[0];
74
+ }
75
+ return `${this.props.selectedValue.length} seleccionados`;
76
+ }
77
+ // Single selection
78
+ const selectedOption = this.props.options.find(opt => opt.value === this.props.selectedValue);
79
+ return selectedOption?.label || this.props.selectedValue;
80
+ }
81
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PopoverSelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
82
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: PopoverSelectorComponent, isStandalone: true, selector: "val-popover-selector", inputs: { props: "props" }, outputs: { selectionChange: "selectionChange" }, ngImport: i0, template: `
83
+ <ion-item [disabled]="props.disabled" lines="none">
84
+ <ion-label *ngIf="props.label" position="stacked">{{ props.label }}</ion-label>
85
+ <ion-select
86
+ [value]="props.selectedValue"
87
+ [placeholder]="props.placeholder || 'Seleccionar...'"
88
+ [interface]="props.interface || 'popover'"
89
+ [multiple]="props.multiple || false"
90
+ [disabled]="props.disabled || false"
91
+ [cancelText]="props.cancelText || 'Cancelar'"
92
+ [okText]="props.okText || 'Aceptar'"
93
+ (ionChange)="onSelectionChange($event)"
94
+ class="popover-selector-select"
95
+ >
96
+ <ion-button
97
+ slot="trigger"
98
+ [fill]="props.fill || 'outline'"
99
+ [size]="props.size || 'default'"
100
+ [shape]="props.shape"
101
+ [expand]="props.expand"
102
+ [color]="props.color || 'medium'"
103
+ [disabled]="props.disabled || false"
104
+ class="selector-trigger"
105
+ >
106
+ <ion-icon *ngIf="props.icon" [name]="props.icon" slot="start"> </ion-icon>
107
+
108
+ <span class="trigger-text">{{ getDisplayText() }}</span>
109
+
110
+ <ion-icon name="chevron-down" slot="end" class="chevron-icon"> </ion-icon>
111
+ </ion-button>
112
+
113
+ <ion-select-option *ngFor="let option of props.options" [value]="option.value" [disabled]="option.disabled">
114
+ <div class="option-content">
115
+ <ion-icon *ngIf="option.icon" [name]="option.icon" class="option-icon"> </ion-icon>
116
+ <span>{{ option.label }}</span>
117
+ </div>
118
+ </ion-select-option>
119
+ </ion-select>
120
+ </ion-item>
121
+ `, isInline: true, styles: [".popover-selector-select{width:100%}.popover-selector-select .selector-trigger{width:100%;justify-content:space-between;text-align:left}.popover-selector-select .selector-trigger .trigger-text{flex:1;text-align:start;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.popover-selector-select .selector-trigger .chevron-icon{margin-left:auto;transition:transform .2s ease}.popover-selector-select .selector-trigger.select-expanded .chevron-icon{transform:rotate(180deg)}.option-content{display:flex;align-items:center;gap:8px}.option-content .option-icon{font-size:16px;width:16px;height:16px}.popover-selector-select.size-small .selector-trigger{--padding-start: 8px;--padding-end: 8px;font-size:14px}.popover-selector-select.size-large .selector-trigger{--padding-start: 16px;--padding-end: 16px;font-size:18px}ion-item[disabled]{opacity:.6;pointer-events:none}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonSelect, selector: "ion-select", inputs: ["cancelText", "color", "compareWith", "disabled", "expandedIcon", "fill", "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: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }] }); }
122
+ }
123
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PopoverSelectorComponent, decorators: [{
124
+ type: Component,
125
+ args: [{ selector: 'val-popover-selector', standalone: true, imports: [CommonModule, IonButton, IonIcon, IonSelect, IonSelectOption, IonItem, IonLabel], template: `
126
+ <ion-item [disabled]="props.disabled" lines="none">
127
+ <ion-label *ngIf="props.label" position="stacked">{{ props.label }}</ion-label>
128
+ <ion-select
129
+ [value]="props.selectedValue"
130
+ [placeholder]="props.placeholder || 'Seleccionar...'"
131
+ [interface]="props.interface || 'popover'"
132
+ [multiple]="props.multiple || false"
133
+ [disabled]="props.disabled || false"
134
+ [cancelText]="props.cancelText || 'Cancelar'"
135
+ [okText]="props.okText || 'Aceptar'"
136
+ (ionChange)="onSelectionChange($event)"
137
+ class="popover-selector-select"
138
+ >
139
+ <ion-button
140
+ slot="trigger"
141
+ [fill]="props.fill || 'outline'"
142
+ [size]="props.size || 'default'"
143
+ [shape]="props.shape"
144
+ [expand]="props.expand"
145
+ [color]="props.color || 'medium'"
146
+ [disabled]="props.disabled || false"
147
+ class="selector-trigger"
148
+ >
149
+ <ion-icon *ngIf="props.icon" [name]="props.icon" slot="start"> </ion-icon>
150
+
151
+ <span class="trigger-text">{{ getDisplayText() }}</span>
152
+
153
+ <ion-icon name="chevron-down" slot="end" class="chevron-icon"> </ion-icon>
154
+ </ion-button>
155
+
156
+ <ion-select-option *ngFor="let option of props.options" [value]="option.value" [disabled]="option.disabled">
157
+ <div class="option-content">
158
+ <ion-icon *ngIf="option.icon" [name]="option.icon" class="option-icon"> </ion-icon>
159
+ <span>{{ option.label }}</span>
160
+ </div>
161
+ </ion-select-option>
162
+ </ion-select>
163
+ </ion-item>
164
+ `, styles: [".popover-selector-select{width:100%}.popover-selector-select .selector-trigger{width:100%;justify-content:space-between;text-align:left}.popover-selector-select .selector-trigger .trigger-text{flex:1;text-align:start;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.popover-selector-select .selector-trigger .chevron-icon{margin-left:auto;transition:transform .2s ease}.popover-selector-select .selector-trigger.select-expanded .chevron-icon{transform:rotate(180deg)}.option-content{display:flex;align-items:center;gap:8px}.option-content .option-icon{font-size:16px;width:16px;height:16px}.popover-selector-select.size-small .selector-trigger{--padding-start: 8px;--padding-end: 8px;font-size:14px}.popover-selector-select.size-large .selector-trigger{--padding-start: 16px;--padding-end: 16px;font-size:18px}ion-item[disabled]{opacity:.6;pointer-events:none}\n"] }]
165
+ }], ctorParameters: () => [], propDecorators: { props: [{
166
+ type: Input
167
+ }], selectionChange: [{
168
+ type: Output
169
+ }] } });
170
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popover-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/popover-selector/popover-selector.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACvE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAC;AAC9G,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;;;AAiD7C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,MAAM,OAAO,wBAAwB;IAenC;QAPA;;;WAGG;QAEH,oBAAe,GAAG,IAAI,YAAY,EAAqB,CAAC;QAGtD,0BAA0B;QAC1B,QAAQ,CAAC,EAAE,WAAW,EAAE,CAAC,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,iBAAiB,CAAC,KAAU;QAC1B,MAAM,aAAa,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC;QACzC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IAC3C,CAAC;IAED;;;;OAIG;IACH,cAAc;QACZ,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,gBAAgB,CAAC;QACpD,CAAC;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,EAAE,CAAC;YAC5C,qBAAqB;YACrB,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC1C,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,gBAAgB,CAAC;YACpD,CAAC;YACD,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC1C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;gBACzF,OAAO,MAAM,EAAE,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;YACtD,CAAC;YACD,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,gBAAgB,CAAC;QAC5D,CAAC;QAED,mBAAmB;QACnB,MAAM,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAC9F,OAAO,cAAc,EAAE,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC;IAC3D,CAAC;+GAtDU,wBAAwB;mGAAxB,wBAAwB,6JAzEzB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuCT,i7BAxCS,YAAY,gQAAE,SAAS,oPAAE,OAAO,2JAAE,SAAS,uTAAE,eAAe,6FAAE,OAAO,0NAAE,QAAQ;;4FA0E9E,wBAAwB;kBA7EpC,SAAS;+BACE,sBAAsB,cACpB,IAAI,WACP,CAAC,YAAY,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,eAAe,EAAE,OAAO,EAAE,QAAQ,CAAC,YAChF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuCT;wDAwCD,KAAK;sBADJ,KAAK;gBAQN,eAAe;sBADd,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { IonButton, IonIcon, IonItem, IonLabel, IonSelect, IonSelectOption } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { chevronDown } from 'ionicons/icons';\nimport { PopoverSelectorMetadata } from './types';\n\n@Component({\n  selector: 'val-popover-selector',\n  standalone: true,\n  imports: [CommonModule, IonButton, IonIcon, IonSelect, IonSelectOption, IonItem, IonLabel],\n  template: `\n    <ion-item [disabled]=\"props.disabled\" lines=\"none\">\n      <ion-label *ngIf=\"props.label\" position=\"stacked\">{{ props.label }}</ion-label>\n      <ion-select\n        [value]=\"props.selectedValue\"\n        [placeholder]=\"props.placeholder || 'Seleccionar...'\"\n        [interface]=\"props.interface || 'popover'\"\n        [multiple]=\"props.multiple || false\"\n        [disabled]=\"props.disabled || false\"\n        [cancelText]=\"props.cancelText || 'Cancelar'\"\n        [okText]=\"props.okText || 'Aceptar'\"\n        (ionChange)=\"onSelectionChange($event)\"\n        class=\"popover-selector-select\"\n      >\n        <ion-button\n          slot=\"trigger\"\n          [fill]=\"props.fill || 'outline'\"\n          [size]=\"props.size || 'default'\"\n          [shape]=\"props.shape\"\n          [expand]=\"props.expand\"\n          [color]=\"props.color || 'medium'\"\n          [disabled]=\"props.disabled || false\"\n          class=\"selector-trigger\"\n        >\n          <ion-icon *ngIf=\"props.icon\" [name]=\"props.icon\" slot=\"start\"> </ion-icon>\n\n          <span class=\"trigger-text\">{{ getDisplayText() }}</span>\n\n          <ion-icon name=\"chevron-down\" slot=\"end\" class=\"chevron-icon\"> </ion-icon>\n        </ion-button>\n\n        <ion-select-option *ngFor=\"let option of props.options\" [value]=\"option.value\" [disabled]=\"option.disabled\">\n          <div class=\"option-content\">\n            <ion-icon *ngIf=\"option.icon\" [name]=\"option.icon\" class=\"option-icon\"> </ion-icon>\n            <span>{{ option.label }}</span>\n          </div>\n        </ion-select-option>\n      </ion-select>\n    </ion-item>\n  `,\n  styleUrls: ['./popover-selector.component.scss'],\n})\n/**\n * val-popover-selector\n *\n * A generic reusable popover selector component that can be used for any type of selection.\n * Provides a button trigger that opens a popover with selectable options.\n *\n * @example\n * // Basic usage\n * <val-popover-selector\n *   [props]=\"selectorProps\"\n *   (selectionChange)=\"onSelectionChange($event)\">\n * </val-popover-selector>\n *\n * @example\n * // With icon and custom styling\n * const selectorProps: PopoverSelectorMetadata = {\n *   options: [\n *     { value: 'option1', label: 'Option 1', icon: 'star' },\n *     { value: 'option2', label: 'Option 2', icon: 'heart' }\n *   ],\n *   selectedValue: 'option1',\n *   label: 'Choose option',\n *   icon: 'settings',\n *   color: 'primary',\n *   size: 'large',\n *   fill: 'solid'\n * };\n *\n * @input props: PopoverSelectorMetadata - Configuration for the selector\n * @output selectionChange: EventEmitter<string | string[]> - Emitted when selection changes\n */\nexport class PopoverSelectorComponent {\n  /**\n   * Popover selector configuration object.\n   * @type {PopoverSelectorMetadata}\n   */\n  @Input()\n  props: PopoverSelectorMetadata;\n\n  /**\n   * Event emitted when the selection changes.\n   * Emits the selected value(s).\n   */\n  @Output()\n  selectionChange = new EventEmitter<string | string[]>();\n\n  constructor() {\n    // Register required icons\n    addIcons({ chevronDown });\n  }\n\n  /**\n   * Handle selection change from the ion-select.\n   * @param event - The ion-select change event\n   */\n  onSelectionChange(event: any) {\n    const selectedValue = event.detail.value;\n    this.selectionChange.emit(selectedValue);\n  }\n\n  /**\n   * Get the display text for the trigger button.\n   * Shows the selected option's label or placeholder.\n   * @returns The display text\n   */\n  getDisplayText(): string {\n    if (!this.props.selectedValue) {\n      return this.props.placeholder || 'Seleccionar...';\n    }\n\n    if (Array.isArray(this.props.selectedValue)) {\n      // Multiple selection\n      if (this.props.selectedValue.length === 0) {\n        return this.props.placeholder || 'Seleccionar...';\n      }\n      if (this.props.selectedValue.length === 1) {\n        const option = this.props.options.find(opt => opt.value === this.props.selectedValue[0]);\n        return option?.label || this.props.selectedValue[0];\n      }\n      return `${this.props.selectedValue.length} seleccionados`;\n    }\n\n    // Single selection\n    const selectedOption = this.props.options.find(opt => opt.value === this.props.selectedValue);\n    return selectedOption?.label || this.props.selectedValue;\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9jb21wb25lbnRzL21vbGVjdWxlcy9wb3BvdmVyLXNlbGVjdG9yL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcblxuLyoqXG4gKiBPcHRpb24gZm9yIHRoZSBwb3BvdmVyIHNlbGVjdG9yLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFBvcG92ZXJPcHRpb24ge1xuICAvKiogT3B0aW9uIHZhbHVlICovXG4gIHZhbHVlOiBzdHJpbmc7XG4gIC8qKiBPcHRpb24gZGlzcGxheSBsYWJlbCAqL1xuICBsYWJlbDogc3RyaW5nO1xuICAvKiogT3B0aW9uIGljb24gKG9wdGlvbmFsKSAqL1xuICBpY29uPzogc3RyaW5nO1xuICAvKiogV2hldGhlciB0aGUgb3B0aW9uIGlzIGRpc2FibGVkICovXG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBQcm9wcyBmb3IgdmFsLXBvcG92ZXItc2VsZWN0b3IgY29tcG9uZW50LlxuICogQSBnZW5lcmljIHJldXNhYmxlIHBvcG92ZXIgc2VsZWN0b3IgY29tcG9uZW50LlxuICpcbiAqIEBwcm9wZXJ0eSBvcHRpb25zIC0gQXJyYXkgb2Ygc2VsZWN0YWJsZSBvcHRpb25zLlxuICogQHByb3BlcnR5IHNlbGVjdGVkVmFsdWUgLSBDdXJyZW50bHkgc2VsZWN0ZWQgdmFsdWUuXG4gKiBAcHJvcGVydHkgbGFiZWwgLSBEaXNwbGF5IGxhYmVsIGZvciB0aGUgc2VsZWN0b3IuXG4gKiBAcHJvcGVydHkgaWNvbiAtIEljb24gdG8gZGlzcGxheSBpbiB0aGUgdHJpZ2dlciBidXR0b24gKG9wdGlvbmFsKS5cbiAqIEBwcm9wZXJ0eSBwbGFjZWhvbGRlciAtIFBsYWNlaG9sZGVyIHRleHQgd2hlbiBubyBvcHRpb24gaXMgc2VsZWN0ZWQuXG4gKiBAcHJvcGVydHkgY29sb3IgLSBCdXR0b24gY29sb3IgKElvbmljIGNvbG9yIHN0cmluZykuXG4gKiBAcHJvcGVydHkgc2l6ZSAtIEJ1dHRvbiBzaXplICgnc21hbGwnIHwgJ2RlZmF1bHQnIHwgJ2xhcmdlJykuXG4gKiBAcHJvcGVydHkgZmlsbCAtIEJ1dHRvbiBmaWxsIHN0eWxlICgnY2xlYXInIHwgJ291dGxpbmUnIHwgJ3NvbGlkJyB8ICdkZWZhdWx0JykuXG4gKiBAcHJvcGVydHkgc2hhcGUgLSBCdXR0b24gc2hhcGUgKCdyb3VuZCcgfCB1bmRlZmluZWQpLlxuICogQHByb3BlcnR5IGV4cGFuZCAtIEJ1dHRvbiBleHBhbnNpb24gKCdmdWxsJyB8ICdibG9jaycgfCB1bmRlZmluZWQpLlxuICogQHByb3BlcnR5IGRpc2FibGVkIC0gV2hldGhlciB0aGUgc2VsZWN0b3IgaXMgZGlzYWJsZWQuXG4gKiBAcHJvcGVydHkgaW50ZXJmYWNlIC0gUG9wb3ZlciBpbnRlcmZhY2Ugc3R5bGUgKCdwb3BvdmVyJyB8ICdhY3Rpb24tc2hlZXQnIHwgJ2FsZXJ0JykuXG4gKiBAcHJvcGVydHkgc2hvd0NoZWNrbWFyayAtIFdoZXRoZXIgdG8gc2hvdyBjaGVja21hcmtzIGZvciBzZWxlY3RlZCBvcHRpb25zLlxuICogQHByb3BlcnR5IG11bHRpcGxlIC0gV2hldGhlciBtdWx0aXBsZSBzZWxlY3Rpb24gaXMgYWxsb3dlZC5cbiAqIEBwcm9wZXJ0eSBjYW5jZWxUZXh0IC0gVGV4dCBmb3IgY2FuY2VsIGJ1dHRvbi5cbiAqIEBwcm9wZXJ0eSBva1RleHQgLSBUZXh0IGZvciBPSyBidXR0b24uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUG9wb3ZlclNlbGVjdG9yTWV0YWRhdGEge1xuICAvKiogQXJyYXkgb2Ygc2VsZWN0YWJsZSBvcHRpb25zICovXG4gIG9wdGlvbnM6IFBvcG92ZXJPcHRpb25bXTtcbiAgLyoqIEN1cnJlbnRseSBzZWxlY3RlZCB2YWx1ZShzKSAqL1xuICBzZWxlY3RlZFZhbHVlPzogc3RyaW5nIHwgc3RyaW5nW107XG4gIC8qKiBEaXNwbGF5IGxhYmVsIGZvciB0aGUgc2VsZWN0b3IgKi9cbiAgbGFiZWw/OiBzdHJpbmc7XG4gIC8qKiBJY29uIHRvIGRpc3BsYXkgaW4gdGhlIHRyaWdnZXIgKi9cbiAgaWNvbj86IHN0cmluZztcbiAgLyoqIFBsYWNlaG9sZGVyIHRleHQgd2hlbiBubyBvcHRpb24gaXMgc2VsZWN0ZWQgKi9cbiAgcGxhY2Vob2xkZXI/OiBzdHJpbmc7XG4gIC8qKiBCdXR0b24gY29sb3IgKi9cbiAgY29sb3I/OiBDb2xvcjtcbiAgLyoqIEJ1dHRvbiBzaXplICovXG4gIHNpemU/OiAnc21hbGwnIHwgJ2RlZmF1bHQnIHwgJ2xhcmdlJztcbiAgLyoqIEJ1dHRvbiBmaWxsIHN0eWxlICovXG4gIGZpbGw/OiAnY2xlYXInIHwgJ291dGxpbmUnIHwgJ3NvbGlkJyB8ICdkZWZhdWx0JztcbiAgLyoqIEJ1dHRvbiBzaGFwZSAqL1xuICBzaGFwZT86ICdyb3VuZCc7XG4gIC8qKiBCdXR0b24gZXhwYW5zaW9uICovXG4gIGV4cGFuZD86ICdmdWxsJyB8ICdibG9jayc7XG4gIC8qKiBXaGV0aGVyIHRoZSBzZWxlY3RvciBpcyBkaXNhYmxlZCAqL1xuICBkaXNhYmxlZD86IGJvb2xlYW47XG4gIC8qKiBQb3BvdmVyIGludGVyZmFjZSBzdHlsZSAqL1xuICBpbnRlcmZhY2U/OiAncG9wb3ZlcicgfCAnYWN0aW9uLXNoZWV0JyB8ICdhbGVydCc7XG4gIC8qKiBXaGV0aGVyIHRvIHNob3cgY2hlY2ttYXJrcyBmb3Igc2VsZWN0ZWQgb3B0aW9ucyAqL1xuICBzaG93Q2hlY2ttYXJrPzogYm9vbGVhbjtcbiAgLyoqIFdoZXRoZXIgbXVsdGlwbGUgc2VsZWN0aW9uIGlzIGFsbG93ZWQgKi9cbiAgbXVsdGlwbGU/OiBib29sZWFuO1xuICAvKiogQ2FuY2VsIGJ1dHRvbiB0ZXh0ICovXG4gIGNhbmNlbFRleHQ/OiBzdHJpbmc7XG4gIC8qKiBPSyBidXR0b24gdGV4dCAqL1xuICBva1RleHQ/OiBzdHJpbmc7XG59XG4iXX0=