@cqa-lib/cqa-ui 1.1.191 → 1.1.193
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/esm2020/lib/autocomplete/autocomplete.component.mjs +156 -0
- package/esm2020/lib/autocomplete/autocomplete.model.mjs +2 -0
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +610 -11
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +505 -17
- package/esm2020/lib/test-case-details/delete-steps/delete-steps.component.mjs +32 -11
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +131 -9
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-data.mjs +67 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-field.config.mjs +66 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer.component.mjs +112 -191
- package/esm2020/lib/test-case-details/step-group/step-group.component.mjs +31 -3
- package/esm2020/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.mjs +128 -6
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/public-api.mjs +3 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +2153 -582
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +2038 -451
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/autocomplete/autocomplete.component.d.ts +48 -0
- package/lib/autocomplete/autocomplete.model.d.ts +10 -0
- package/lib/test-case-details/api-edit-step/api-edit-step.component.d.ts +96 -8
- package/lib/test-case-details/condition-step/condition-step.component.d.ts +73 -4
- package/lib/test-case-details/delete-steps/delete-steps.component.d.ts +18 -2
- package/lib/test-case-details/loop-step/loop-step.component.d.ts +34 -4
- package/lib/test-case-details/step-details-drawer/step-details-drawer-data.d.ts +56 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer-field.config.d.ts +6 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer.component.d.ts +33 -27
- package/lib/test-case-details/step-group/step-group.component.d.ts +15 -1
- package/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.d.ts +31 -5
- package/lib/ui-kit.module.d.ts +127 -126
- package/package.json +1 -1
- package/public-api.d.ts +2 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
import { Component, EventEmitter, HostListener, Input, Output, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
export class AutocompleteComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
/** Placeholder text for the input */
|
|
8
|
+
this.placeholder = 'Search or select...';
|
|
9
|
+
/** Options shown in the dropdown (filtered by input when open) */
|
|
10
|
+
this.options = [];
|
|
11
|
+
/** Initial or controlled value (displayed in the input) */
|
|
12
|
+
this.value = '';
|
|
13
|
+
/** Disable the input */
|
|
14
|
+
this.disabled = false;
|
|
15
|
+
/** Whether to show the clear button when there is text */
|
|
16
|
+
this.showClear = true;
|
|
17
|
+
/** Accessible label for the input */
|
|
18
|
+
this.ariaLabel = 'Auto-complete';
|
|
19
|
+
/** Auto focus the input when rendered */
|
|
20
|
+
this.autoFocus = false;
|
|
21
|
+
/** Size variant (matches search bar widths) */
|
|
22
|
+
this.size = 'md';
|
|
23
|
+
/** Stretch to full width of container */
|
|
24
|
+
this.fullWidth = false;
|
|
25
|
+
/** Emit when the input value changes (e.g. for two-way binding) */
|
|
26
|
+
this.valueChange = new EventEmitter();
|
|
27
|
+
/** Emit when an option is selected from the list (value from option) */
|
|
28
|
+
this.optionSelect = new EventEmitter();
|
|
29
|
+
/** Emit when the value is cleared via the clear button */
|
|
30
|
+
this.cleared = new EventEmitter();
|
|
31
|
+
this.inputValue = '';
|
|
32
|
+
this.panelOpen = false;
|
|
33
|
+
this.blurTimeout = null;
|
|
34
|
+
this.widthClasses = {
|
|
35
|
+
sm: 'cqa-w-[295px]',
|
|
36
|
+
md: 'cqa-w-[395px]',
|
|
37
|
+
lg: 'cqa-w-[495px]',
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
ngOnInit() {
|
|
41
|
+
// Initialize inputValue from value input
|
|
42
|
+
if (this.value !== undefined && this.value !== null && this.value !== '') {
|
|
43
|
+
this.inputValue = String(this.value);
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
this.inputValue = '';
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
ngOnChanges(changes) {
|
|
50
|
+
if (changes['value']) {
|
|
51
|
+
const newValue = changes['value'].currentValue;
|
|
52
|
+
if (newValue !== undefined && newValue !== null && newValue !== '') {
|
|
53
|
+
const stringValue = String(newValue);
|
|
54
|
+
if (stringValue !== this.inputValue) {
|
|
55
|
+
this.inputValue = stringValue;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
// Clear input if value is empty/null/undefined
|
|
60
|
+
if (this.inputValue !== '') {
|
|
61
|
+
this.inputValue = '';
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
get displayOptions() {
|
|
67
|
+
const list = this.options ?? [];
|
|
68
|
+
const q = (this.inputValue ?? '').toLowerCase().trim();
|
|
69
|
+
if (!q) {
|
|
70
|
+
return list;
|
|
71
|
+
}
|
|
72
|
+
return list.filter((opt) => {
|
|
73
|
+
const label = (opt.label ?? opt.value ?? '').toLowerCase();
|
|
74
|
+
return label.includes(q);
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
onInput(event) {
|
|
78
|
+
const target = event.target;
|
|
79
|
+
const nextValue = target?.value ?? '';
|
|
80
|
+
this.inputValue = nextValue;
|
|
81
|
+
this.valueChange.emit(this.inputValue);
|
|
82
|
+
this.panelOpen = true;
|
|
83
|
+
}
|
|
84
|
+
onFocus() {
|
|
85
|
+
if (this.blurTimeout) {
|
|
86
|
+
clearTimeout(this.blurTimeout);
|
|
87
|
+
this.blurTimeout = null;
|
|
88
|
+
}
|
|
89
|
+
this.panelOpen = true;
|
|
90
|
+
}
|
|
91
|
+
onBlur() {
|
|
92
|
+
this.blurTimeout = setTimeout(() => {
|
|
93
|
+
this.panelOpen = false;
|
|
94
|
+
this.blurTimeout = null;
|
|
95
|
+
}, 150);
|
|
96
|
+
}
|
|
97
|
+
selectOption(option) {
|
|
98
|
+
const displayLabel = option.label ?? option.value;
|
|
99
|
+
this.inputValue = displayLabel;
|
|
100
|
+
this.valueChange.emit(this.inputValue);
|
|
101
|
+
this.optionSelect.emit(option);
|
|
102
|
+
this.panelOpen = false;
|
|
103
|
+
}
|
|
104
|
+
clear() {
|
|
105
|
+
if (this.disabled || this.inputValue === '') {
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
this.inputValue = '';
|
|
109
|
+
this.valueChange.emit('');
|
|
110
|
+
this.cleared.emit();
|
|
111
|
+
this.panelOpen = false;
|
|
112
|
+
}
|
|
113
|
+
getOptionLabel(option) {
|
|
114
|
+
return option.label ?? option.value ?? '';
|
|
115
|
+
}
|
|
116
|
+
trackByValue(_index, option) {
|
|
117
|
+
return option.value;
|
|
118
|
+
}
|
|
119
|
+
onEscape() {
|
|
120
|
+
this.panelOpen = false;
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
AutocompleteComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AutocompleteComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
124
|
+
AutocompleteComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AutocompleteComponent, selector: "cqa-autocomplete", inputs: { placeholder: "placeholder", options: "options", value: "value", disabled: "disabled", showClear: "showClear", ariaLabel: "ariaLabel", autoFocus: "autoFocus", size: "size", fullWidth: "fullWidth" }, outputs: { valueChange: "valueChange", optionSelect: "optionSelect", cleared: "cleared" }, host: { listeners: { "document:keydown.escape": "onEscape()" }, classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"cqa-autocomplete-root cqa-relative\"\n [style.display]=\"fullWidth ? 'block' : 'inline-block'\"\n [style.width]=\"fullWidth ? '100%' : 'auto'\">\n <!-- Field: same styling as search bar -->\n <div\n class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-6 cqa-py-3 cqa-text-[14px] cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-transition-colors\"\n [ngClass]=\"[fullWidth ? 'cqa-w-full' : widthClasses[size]]\">\n <input\n type=\"text\"\n class=\"cqa-flex-1 cqa-min-w-0 cqa-border-none cqa-outline-none cqa-bg-transparent placeholder:cqa-text-gray-300 disabled:cqa-text-gray-400 disabled:cqa-cursor-not-allowed cqa-font-['SF_Pro_Text'] cqa-font-normal cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-align-middle cqa-text-black-100\"\n [placeholder]=\"placeholder\"\n [value]=\"inputValue\"\n (input)=\"onInput($event)\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n [attr.aria-expanded]=\"panelOpen\"\n aria-autocomplete=\"list\"\n aria-haspopup=\"listbox\"\n autocomplete=\"off\"\n autocapitalize=\"none\"\n spellcheck=\"false\"\n [attr.autofocus]=\"autoFocus ? '' : null\"\n />\n <button\n *ngIf=\"showClear && inputValue\"\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-p-0 cqa-w-4 cqa-h-4 cqa-border-0 cqa-bg-transparent cqa-cursor-pointer cqa-text-gray-500 cqa-hover:cqa-text-gray-700 disabled:cqa-text-gray-300 cqa-transition-colors cqa-leading-none\"\n (click)=\"clear()\"\n [disabled]=\"disabled\"\n aria-label=\"Clear\">\n <mat-icon\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-leading-none cqa-p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n [ngClass]=\"{ 'cqa-opacity-[0.38]': disabled }\">\n close\n </mat-icon>\n </button>\n </div>\n <!-- List box: below field, same width as field -->\n <ul\n *ngIf=\"panelOpen\"\n role=\"listbox\"\n class=\"cqa-autocomplete-list cqa-absolute cqa-left-0 cqa-right-0 cqa-top-full cqa-mt-1 cqa-py-1 cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-max-h-[240px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-z-10 cqa-list-none cqa-m-0 cqa-p-0\"\n [class.cqa-w-full]=\"fullWidth\">\n <li\n *ngFor=\"let option of displayOptions; trackBy: trackByValue\"\n role=\"option\"\n (mousedown)=\"selectOption(option); $event.preventDefault()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-text-black-100 cqa-cursor-pointer cqa-font-['SF_Pro_Text'] hover:cqa-bg-gray-100 cqa-transition-colors\">\n {{ getOptionLabel(option) }}\n </li>\n <li\n *ngIf=\"displayOptions.length === 0\"\n class=\"cqa-px-4 cqa-py-3 cqa-text-[12.3px] cqa-text-gray-400 cqa-cursor-default\">\n No matches\n </li>\n </ul>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
125
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AutocompleteComponent, decorators: [{
|
|
126
|
+
type: Component,
|
|
127
|
+
args: [{ selector: 'cqa-autocomplete', host: { class: 'cqa-ui-root' }, template: "<div\n class=\"cqa-autocomplete-root cqa-relative\"\n [style.display]=\"fullWidth ? 'block' : 'inline-block'\"\n [style.width]=\"fullWidth ? '100%' : 'auto'\">\n <!-- Field: same styling as search bar -->\n <div\n class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-6 cqa-py-3 cqa-text-[14px] cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-transition-colors\"\n [ngClass]=\"[fullWidth ? 'cqa-w-full' : widthClasses[size]]\">\n <input\n type=\"text\"\n class=\"cqa-flex-1 cqa-min-w-0 cqa-border-none cqa-outline-none cqa-bg-transparent placeholder:cqa-text-gray-300 disabled:cqa-text-gray-400 disabled:cqa-cursor-not-allowed cqa-font-['SF_Pro_Text'] cqa-font-normal cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-align-middle cqa-text-black-100\"\n [placeholder]=\"placeholder\"\n [value]=\"inputValue\"\n (input)=\"onInput($event)\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n [attr.aria-expanded]=\"panelOpen\"\n aria-autocomplete=\"list\"\n aria-haspopup=\"listbox\"\n autocomplete=\"off\"\n autocapitalize=\"none\"\n spellcheck=\"false\"\n [attr.autofocus]=\"autoFocus ? '' : null\"\n />\n <button\n *ngIf=\"showClear && inputValue\"\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-p-0 cqa-w-4 cqa-h-4 cqa-border-0 cqa-bg-transparent cqa-cursor-pointer cqa-text-gray-500 cqa-hover:cqa-text-gray-700 disabled:cqa-text-gray-300 cqa-transition-colors cqa-leading-none\"\n (click)=\"clear()\"\n [disabled]=\"disabled\"\n aria-label=\"Clear\">\n <mat-icon\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-leading-none cqa-p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n [ngClass]=\"{ 'cqa-opacity-[0.38]': disabled }\">\n close\n </mat-icon>\n </button>\n </div>\n <!-- List box: below field, same width as field -->\n <ul\n *ngIf=\"panelOpen\"\n role=\"listbox\"\n class=\"cqa-autocomplete-list cqa-absolute cqa-left-0 cqa-right-0 cqa-top-full cqa-mt-1 cqa-py-1 cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-max-h-[240px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-z-10 cqa-list-none cqa-m-0 cqa-p-0\"\n [class.cqa-w-full]=\"fullWidth\">\n <li\n *ngFor=\"let option of displayOptions; trackBy: trackByValue\"\n role=\"option\"\n (mousedown)=\"selectOption(option); $event.preventDefault()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-text-black-100 cqa-cursor-pointer cqa-font-['SF_Pro_Text'] hover:cqa-bg-gray-100 cqa-transition-colors\">\n {{ getOptionLabel(option) }}\n </li>\n <li\n *ngIf=\"displayOptions.length === 0\"\n class=\"cqa-px-4 cqa-py-3 cqa-text-[12.3px] cqa-text-gray-400 cqa-cursor-default\">\n No matches\n </li>\n </ul>\n</div>\n" }]
|
|
128
|
+
}], propDecorators: { placeholder: [{
|
|
129
|
+
type: Input
|
|
130
|
+
}], options: [{
|
|
131
|
+
type: Input
|
|
132
|
+
}], value: [{
|
|
133
|
+
type: Input
|
|
134
|
+
}], disabled: [{
|
|
135
|
+
type: Input
|
|
136
|
+
}], showClear: [{
|
|
137
|
+
type: Input
|
|
138
|
+
}], ariaLabel: [{
|
|
139
|
+
type: Input
|
|
140
|
+
}], autoFocus: [{
|
|
141
|
+
type: Input
|
|
142
|
+
}], size: [{
|
|
143
|
+
type: Input
|
|
144
|
+
}], fullWidth: [{
|
|
145
|
+
type: Input
|
|
146
|
+
}], valueChange: [{
|
|
147
|
+
type: Output
|
|
148
|
+
}], optionSelect: [{
|
|
149
|
+
type: Output
|
|
150
|
+
}], cleared: [{
|
|
151
|
+
type: Output
|
|
152
|
+
}], onEscape: [{
|
|
153
|
+
type: HostListener,
|
|
154
|
+
args: ['document:keydown.escape']
|
|
155
|
+
}] } });
|
|
156
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.component.js","sourceRoot":"","sources":["../../../../../src/lib/autocomplete/autocomplete.component.ts","../../../../../src/lib/autocomplete/autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport { CqaAutocompleteOption } from './autocomplete.model';\n\ntype AutocompleteSize = 'sm' | 'md' | 'lg';\n\n@Component({\n  selector: 'cqa-autocomplete',\n  templateUrl: './autocomplete.component.html',\n  host: { class: 'cqa-ui-root' },\n})\nexport class AutocompleteComponent implements OnInit, OnChanges {\n  /** Placeholder text for the input */\n  @Input() placeholder = 'Search or select...';\n\n  /** Options shown in the dropdown (filtered by input when open) */\n  @Input() options: CqaAutocompleteOption[] = [];\n\n  /** Initial or controlled value (displayed in the input) */\n  @Input() value = '';\n\n  /** Disable the input */\n  @Input() disabled = false;\n\n  /** Whether to show the clear button when there is text */\n  @Input() showClear = true;\n\n  /** Accessible label for the input */\n  @Input() ariaLabel = 'Auto-complete';\n\n  /** Auto focus the input when rendered */\n  @Input() autoFocus = false;\n\n  /** Size variant (matches search bar widths) */\n  @Input() size: AutocompleteSize = 'md';\n\n  /** Stretch to full width of container */\n  @Input() fullWidth = false;\n\n  /** Emit when the input value changes (e.g. for two-way binding) */\n  @Output() valueChange = new EventEmitter<string>();\n\n  /** Emit when an option is selected from the list (value from option) */\n  @Output() optionSelect = new EventEmitter<CqaAutocompleteOption>();\n\n  /** Emit when the value is cleared via the clear button */\n  @Output() cleared = new EventEmitter<void>();\n\n  inputValue = '';\n  panelOpen = false;\n  private blurTimeout: ReturnType<typeof setTimeout> | null = null;\n\n  ngOnInit(): void {\n    // Initialize inputValue from value input\n    if (this.value !== undefined && this.value !== null && this.value !== '') {\n      this.inputValue = String(this.value);\n    } else {\n      this.inputValue = '';\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['value']) {\n      const newValue = changes['value'].currentValue;\n      if (newValue !== undefined && newValue !== null && newValue !== '') {\n        const stringValue = String(newValue);\n        if (stringValue !== this.inputValue) {\n          this.inputValue = stringValue;\n        }\n      } else {\n        // Clear input if value is empty/null/undefined\n        if (this.inputValue !== '') {\n          this.inputValue = '';\n        }\n      }\n    }\n  }\n\n  readonly widthClasses: Record<AutocompleteSize, string> = {\n    sm: 'cqa-w-[295px]',\n    md: 'cqa-w-[395px]',\n    lg: 'cqa-w-[495px]',\n  };\n\n  get displayOptions(): CqaAutocompleteOption[] {\n    const list = this.options ?? [];\n    const q = (this.inputValue ?? '').toLowerCase().trim();\n    if (!q) {\n      return list;\n    }\n    return list.filter((opt) => {\n      const label = (opt.label ?? opt.value ?? '').toLowerCase();\n      return label.includes(q);\n    });\n  }\n\n  onInput(event: Event): void {\n    const target = event.target as HTMLInputElement | null;\n    const nextValue = target?.value ?? '';\n    this.inputValue = nextValue;\n    this.valueChange.emit(this.inputValue);\n    this.panelOpen = true;\n  }\n\n  onFocus(): void {\n    if (this.blurTimeout) {\n      clearTimeout(this.blurTimeout);\n      this.blurTimeout = null;\n    }\n    this.panelOpen = true;\n  }\n\n  onBlur(): void {\n    this.blurTimeout = setTimeout(() => {\n      this.panelOpen = false;\n      this.blurTimeout = null;\n    }, 150);\n  }\n\n  selectOption(option: CqaAutocompleteOption): void {\n    const displayLabel = option.label ?? option.value;\n    this.inputValue = displayLabel;\n    this.valueChange.emit(this.inputValue);\n    this.optionSelect.emit(option);\n    this.panelOpen = false;\n  }\n\n  clear(): void {\n    if (this.disabled || this.inputValue === '') {\n      return;\n    }\n    this.inputValue = '';\n    this.valueChange.emit('');\n    this.cleared.emit();\n    this.panelOpen = false;\n  }\n\n  getOptionLabel(option: CqaAutocompleteOption): string {\n    return option.label ?? option.value ?? '';\n  }\n\n  trackByValue(_index: number, option: CqaAutocompleteOption): string {\n    return option.value;\n  }\n\n  @HostListener('document:keydown.escape')\n  onEscape(): void {\n    this.panelOpen = false;\n  }\n}\n","<div\n  class=\"cqa-autocomplete-root cqa-relative\"\n  [style.display]=\"fullWidth ? 'block' : 'inline-block'\"\n  [style.width]=\"fullWidth ? '100%' : 'auto'\">\n  <!-- Field: same styling as search bar -->\n  <div\n    class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-6 cqa-py-3 cqa-text-[14px] cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-transition-colors\"\n    [ngClass]=\"[fullWidth ? 'cqa-w-full' : widthClasses[size]]\">\n    <input\n      type=\"text\"\n      class=\"cqa-flex-1 cqa-min-w-0 cqa-border-none cqa-outline-none cqa-bg-transparent placeholder:cqa-text-gray-300 disabled:cqa-text-gray-400 disabled:cqa-cursor-not-allowed cqa-font-['SF_Pro_Text'] cqa-font-normal cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-align-middle cqa-text-black-100\"\n      [placeholder]=\"placeholder\"\n      [value]=\"inputValue\"\n      (input)=\"onInput($event)\"\n      (focus)=\"onFocus()\"\n      (blur)=\"onBlur()\"\n      [disabled]=\"disabled\"\n      [attr.aria-label]=\"ariaLabel\"\n      [attr.aria-expanded]=\"panelOpen\"\n      aria-autocomplete=\"list\"\n      aria-haspopup=\"listbox\"\n      autocomplete=\"off\"\n      autocapitalize=\"none\"\n      spellcheck=\"false\"\n      [attr.autofocus]=\"autoFocus ? '' : null\"\n    />\n    <button\n      *ngIf=\"showClear && inputValue\"\n      type=\"button\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-p-0 cqa-w-4 cqa-h-4 cqa-border-0 cqa-bg-transparent cqa-cursor-pointer cqa-text-gray-500 cqa-hover:cqa-text-gray-700 disabled:cqa-text-gray-300 cqa-transition-colors cqa-leading-none\"\n      (click)=\"clear()\"\n      [disabled]=\"disabled\"\n      aria-label=\"Clear\">\n      <mat-icon\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-leading-none cqa-p-0\"\n        [style.width.px]=\"16\"\n        [style.height.px]=\"16\"\n        [style.fontSize.px]=\"16\"\n        [ngClass]=\"{ 'cqa-opacity-[0.38]': disabled }\">\n        close\n      </mat-icon>\n    </button>\n  </div>\n  <!-- List box: below field, same width as field -->\n  <ul\n    *ngIf=\"panelOpen\"\n    role=\"listbox\"\n    class=\"cqa-autocomplete-list cqa-absolute cqa-left-0 cqa-right-0 cqa-top-full cqa-mt-1 cqa-py-1 cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-max-h-[240px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-z-10 cqa-list-none cqa-m-0 cqa-p-0\"\n    [class.cqa-w-full]=\"fullWidth\">\n    <li\n      *ngFor=\"let option of displayOptions; trackBy: trackByValue\"\n      role=\"option\"\n      (mousedown)=\"selectOption(option); $event.preventDefault()\"\n      class=\"cqa-px-4 cqa-py-2 cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-text-black-100 cqa-cursor-pointer cqa-font-['SF_Pro_Text'] hover:cqa-bg-gray-100 cqa-transition-colors\">\n      {{ getOptionLabel(option) }}\n    </li>\n    <li\n      *ngIf=\"displayOptions.length === 0\"\n      class=\"cqa-px-4 cqa-py-3 cqa-text-[12.3px] cqa-text-gray-400 cqa-cursor-default\">\n      No matches\n    </li>\n  </ul>\n</div>\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0b2NvbXBsZXRlLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9hdXRvY29tcGxldGUvYXV0b2NvbXBsZXRlLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE9wdGlvbiBmb3IgdGhlIGF1dG9jb21wbGV0ZSBkcm9wZG93bi5cbiAqIFVzZSBsYWJlbCBmb3IgZGlzcGxheSBhbmQgdmFsdWUgZm9yIHRoZSBzZWxlY3RlZCB2YWx1ZSAoZS5nLiBmb3IgSURzKS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDcWFBdXRvY29tcGxldGVPcHRpb24ge1xuICAvKiogVmFsdWUgZW1pdHRlZCBvbiBzZWxlY3Rpb24gKGUuZy4gaWQgb3IgY29kZSkgKi9cbiAgdmFsdWU6IHN0cmluZztcbiAgLyoqIFRleHQgc2hvd24gaW4gdGhlIGxpc3QgKGRlZmF1bHRzIHRvIHZhbHVlIGlmIG9taXR0ZWQpICovXG4gIGxhYmVsPzogc3RyaW5nO1xufVxuIl19
|