@byuhbll/components 0.0.8
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/README.md +25 -0
- package/esm2022/byuhbll-components.mjs +5 -0
- package/esm2022/lib/animations/animations.mjs +22 -0
- package/esm2022/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.mjs +121 -0
- package/esm2022/lib/hbll-checkbox/hbll-checkbox.component.mjs +17 -0
- package/esm2022/lib/hbll-header/hbll-header.component.mjs +60 -0
- package/esm2022/lib/hbll-multi-select/hbll-multi-select.component.mjs +114 -0
- package/esm2022/lib/ss-search-bar/advanced-search/advanced-search.component.mjs +270 -0
- package/esm2022/lib/ss-search-bar/constants.mjs +41 -0
- package/esm2022/lib/ss-search-bar/date-range/date-range.component.mjs +71 -0
- package/esm2022/lib/ss-search-bar/pipes/advanced-queries.pipe.mjs +27 -0
- package/esm2022/lib/ss-search-bar/simple-search/simple-search.component.mjs +85 -0
- package/esm2022/lib/ss-search-bar/ss-search-bar.component.mjs +199 -0
- package/esm2022/lib/ss-search-bar/utils.mjs +16 -0
- package/esm2022/public-api.mjs +6 -0
- package/fesm2022/byuhbll-components.mjs +1001 -0
- package/fesm2022/byuhbll-components.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/animations/animations.d.ts +4 -0
- package/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.d.ts +17 -0
- package/lib/hbll-checkbox/hbll-checkbox.component.d.ts +6 -0
- package/lib/hbll-header/hbll-header.component.d.ts +40 -0
- package/lib/hbll-multi-select/hbll-multi-select.component.d.ts +52 -0
- package/lib/ss-search-bar/advanced-search/advanced-search.component.d.ts +122 -0
- package/lib/ss-search-bar/constants.d.ts +40 -0
- package/lib/ss-search-bar/date-range/date-range.component.d.ts +25 -0
- package/lib/ss-search-bar/pipes/advanced-queries.pipe.d.ts +8 -0
- package/lib/ss-search-bar/simple-search/simple-search.component.d.ts +28 -0
- package/lib/ss-search-bar/ss-search-bar.component.d.ts +121 -0
- package/lib/ss-search-bar/utils.d.ts +7 -0
- package/package.json +25 -0
- package/public-api.d.ts +2 -0
|
@@ -0,0 +1,270 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output, ViewChild, inject, } from '@angular/core';
|
|
2
|
+
import { NonNullableFormBuilder, ReactiveFormsModule, Validators, } from '@angular/forms';
|
|
3
|
+
import { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';
|
|
4
|
+
import { filter, startWith, tap } from 'rxjs/operators';
|
|
5
|
+
import { ADVANCED_SEARCH_OPTIONS, ADVANCED_SEARCH_QUERIES_LIMIT, } from '../ss-search-bar.component';
|
|
6
|
+
import { Subscription } from 'rxjs';
|
|
7
|
+
import { validateDateRangeFields } from '../utils';
|
|
8
|
+
import { CommonModule } from '@angular/common';
|
|
9
|
+
import { ADVANCED_SEARCH_FIELD_MAP, ADVANCED_SEARCH_QUALIFIER_MAP } from '../constants';
|
|
10
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
11
|
+
import { HbllMultiSelectComponent } from '../../hbll-multi-select/hbll-multi-select.component';
|
|
12
|
+
import { HbllCheckboxComponent } from '../../hbll-checkbox/hbll-checkbox.component';
|
|
13
|
+
import { HbllPillBtnDirective } from '../../directives/hbll-pill-btn/hbll-pill-btn.directive';
|
|
14
|
+
import { DateRangeComponent } from '../date-range/date-range.component';
|
|
15
|
+
import * as i0 from "@angular/core";
|
|
16
|
+
import * as i1 from "@angular/forms";
|
|
17
|
+
import * as i2 from "@angular/material/tooltip";
|
|
18
|
+
const DATE_VALIDATOR_REGEX = '^[0-9]{4}$';
|
|
19
|
+
export class AdvancedSearchComponent {
|
|
20
|
+
constructor() {
|
|
21
|
+
this.fb = inject(NonNullableFormBuilder);
|
|
22
|
+
this.advancedSearch = new EventEmitter();
|
|
23
|
+
this.subscription = new Subscription();
|
|
24
|
+
/**
|
|
25
|
+
* Takes a `FormGroup` and syncs up the `field` and `qualifier` values.
|
|
26
|
+
* After syncing, when the `field` value changes, the `qualifier` value will update appropriately.
|
|
27
|
+
*/
|
|
28
|
+
this.syncUpFieldAndQualifier = (fg, scope) => {
|
|
29
|
+
this.subscription.add(fg.controls.field.valueChanges
|
|
30
|
+
.pipe(filter(() => scope === 'local'), tap((val) => fg.controls.qualifier.setValue(ADVANCED_SEARCH_OPTIONS.qualifiers[val][0])))
|
|
31
|
+
.subscribe());
|
|
32
|
+
};
|
|
33
|
+
this.advancedSearchForm = this.fb.group({
|
|
34
|
+
queries: this.fb.array([], [Validators.maxLength(ADVANCED_SEARCH_QUERIES_LIMIT)]),
|
|
35
|
+
types: [[]],
|
|
36
|
+
collections: [[]],
|
|
37
|
+
languages: [[]],
|
|
38
|
+
datePublished: this.fb.group({
|
|
39
|
+
from: ['', [Validators.maxLength(4), Validators.pattern(DATE_VALIDATOR_REGEX)]],
|
|
40
|
+
to: ['', [Validators.maxLength(4), Validators.pattern(DATE_VALIDATOR_REGEX)]],
|
|
41
|
+
}, { validators: validateDateRangeFields() }),
|
|
42
|
+
limitResults: this.fb.group({
|
|
43
|
+
peerReviewed: [false],
|
|
44
|
+
}),
|
|
45
|
+
expandResults: this.fb.group({
|
|
46
|
+
applyRelatedWords: [false],
|
|
47
|
+
applyEquivalentSubjects: [false],
|
|
48
|
+
fullText: [false],
|
|
49
|
+
}),
|
|
50
|
+
});
|
|
51
|
+
this.isSubmitted = false;
|
|
52
|
+
this.advancedSearchOptions = ADVANCED_SEARCH_OPTIONS;
|
|
53
|
+
this.advancedSearchFieldMap = ADVANCED_SEARCH_FIELD_MAP;
|
|
54
|
+
this.advancedSearchQualifierMap = ADVANCED_SEARCH_QUALIFIER_MAP;
|
|
55
|
+
/**
|
|
56
|
+
* Add a query to `queries` `FormArray`. This FormGroup has an additional `boolean` control.
|
|
57
|
+
*/
|
|
58
|
+
this.addQuery = () => {
|
|
59
|
+
this.queries.push(this.fb.group({
|
|
60
|
+
boolean: [ADVANCED_SEARCH_OPTIONS.boolean[0]],
|
|
61
|
+
field: [
|
|
62
|
+
this.config.scope === 'external'
|
|
63
|
+
? ADVANCED_SEARCH_OPTIONS.externalFields[0]
|
|
64
|
+
: ADVANCED_SEARCH_OPTIONS.localFields[0],
|
|
65
|
+
],
|
|
66
|
+
qualifier: [
|
|
67
|
+
ADVANCED_SEARCH_OPTIONS.qualifiers[ADVANCED_SEARCH_OPTIONS.localFields[0]][0],
|
|
68
|
+
],
|
|
69
|
+
query: ['', Validators.required],
|
|
70
|
+
}));
|
|
71
|
+
this.syncUpFieldAndQualifier(this.queries.at(-1), this.config.scope);
|
|
72
|
+
};
|
|
73
|
+
this.removeQuery = (i) => {
|
|
74
|
+
this.queries.removeAt(i);
|
|
75
|
+
};
|
|
76
|
+
/**
|
|
77
|
+
* Prepares appropriate query params before performing a local advanced search.
|
|
78
|
+
*/
|
|
79
|
+
this.doLocalAdvancedSearch = () => {
|
|
80
|
+
this.doAdvancedSearch({
|
|
81
|
+
localAdvancedSearch: {
|
|
82
|
+
...this.config.localAdvancedSearch,
|
|
83
|
+
selectedResourceTypes: this.types.value,
|
|
84
|
+
selectedCollections: this.collections.value,
|
|
85
|
+
},
|
|
86
|
+
});
|
|
87
|
+
};
|
|
88
|
+
/**
|
|
89
|
+
* Prepares appropriate query params before performing an external advanced search.
|
|
90
|
+
*/
|
|
91
|
+
this.doExternalAdvancedSearch = () => {
|
|
92
|
+
// Compares 'from' and 'to' to ensure 'to' is greater than / equal to 'from', swaps them if not
|
|
93
|
+
const from = this.from.value
|
|
94
|
+
? parseInt(this.from.value) <= parseInt(this.to.value)
|
|
95
|
+
? this.from.value
|
|
96
|
+
: this.to.value
|
|
97
|
+
: '';
|
|
98
|
+
// Compares 'from' and 'to' to ensure 'to' is greater than / equal to 'from', swaps them if not
|
|
99
|
+
const to = this.to.value
|
|
100
|
+
? parseInt(this.to.value) >= parseInt(this.from.value)
|
|
101
|
+
? this.to.value
|
|
102
|
+
: this.from.value
|
|
103
|
+
: '';
|
|
104
|
+
this.doAdvancedSearch({
|
|
105
|
+
externalAdvancedSearch: {
|
|
106
|
+
...this.config.externalAdvancedSearch,
|
|
107
|
+
datePublished: { from, to },
|
|
108
|
+
selectedLanguages: this.languages.value,
|
|
109
|
+
limitResults: {
|
|
110
|
+
peerReviewed: this.peerReviewed.value,
|
|
111
|
+
},
|
|
112
|
+
expandResults: {
|
|
113
|
+
applyRelatedWords: this.applyRelatedWords.value,
|
|
114
|
+
applyEquivalentSubjects: this.applyEquivalentSubjects.value,
|
|
115
|
+
fullText: this.fullText.value,
|
|
116
|
+
},
|
|
117
|
+
},
|
|
118
|
+
});
|
|
119
|
+
};
|
|
120
|
+
/**
|
|
121
|
+
* Verifies that there is a query to use, gathers all other query parameters needed, then performs an advanced search via `router.navigate()`.
|
|
122
|
+
* @param {Record<string, string>} queryParams base set of query params to navigate with
|
|
123
|
+
*/
|
|
124
|
+
this.doAdvancedSearch = (updatedProps) => {
|
|
125
|
+
this.isSubmitted = true;
|
|
126
|
+
// Don't perform a search unless there is at least one query
|
|
127
|
+
if (this.queries.at(0).controls.query.invalid &&
|
|
128
|
+
!this.queries.value.some((v) => !!v.query)) {
|
|
129
|
+
this.showSearchValidationToolTip();
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
// Don't perform a search unless the date form is valid
|
|
133
|
+
if (this.advancedSearchForm.controls.datePublished.invalid)
|
|
134
|
+
return;
|
|
135
|
+
this.advancedSearch.emit({
|
|
136
|
+
...this.config,
|
|
137
|
+
...updatedProps,
|
|
138
|
+
advancedSearchQueryRows: this.queries.value,
|
|
139
|
+
});
|
|
140
|
+
this.isSubmitted = false;
|
|
141
|
+
};
|
|
142
|
+
this.showSearchValidationToolTip = () => {
|
|
143
|
+
this.searchInput.nativeElement.focus();
|
|
144
|
+
this.inputTooltip.disabled = false;
|
|
145
|
+
this.inputTooltip.show();
|
|
146
|
+
};
|
|
147
|
+
this.setupForm = () => {
|
|
148
|
+
this.queries.clear();
|
|
149
|
+
// Always keep at least two queries available
|
|
150
|
+
this.addQuery();
|
|
151
|
+
this.addQuery();
|
|
152
|
+
// Ensure validation tooltip closes when uses enters a query
|
|
153
|
+
this.subscription.add(this.queries
|
|
154
|
+
.at(0)
|
|
155
|
+
.controls.query.valueChanges.pipe(startWith(''))
|
|
156
|
+
.pipe(filter(() => !!this.inputTooltip))
|
|
157
|
+
.subscribe(() => (this.inputTooltip.disabled = true)));
|
|
158
|
+
// If no `q{NUM}` use `q` for first query
|
|
159
|
+
if (!this.config.advancedSearchQueryRows.length)
|
|
160
|
+
this.queries.at(0).controls.query.setValue(this.config.q);
|
|
161
|
+
this.config.advancedSearchQueryRows.forEach((queryRow, i) => {
|
|
162
|
+
// Add a query form control for every q{num} key after the first.
|
|
163
|
+
let queryControl = null;
|
|
164
|
+
if (i === 0 || i === 1) {
|
|
165
|
+
queryControl = this.queries.at(i);
|
|
166
|
+
}
|
|
167
|
+
else {
|
|
168
|
+
this.addQuery();
|
|
169
|
+
queryControl = this.queries.at(-1);
|
|
170
|
+
}
|
|
171
|
+
// Add the query value.
|
|
172
|
+
queryControl.controls.query.setValue(queryRow.query);
|
|
173
|
+
// Add the field value, if available.
|
|
174
|
+
if (queryRow.field)
|
|
175
|
+
queryControl.controls.field.setValue(queryRow.field);
|
|
176
|
+
else
|
|
177
|
+
queryControl.controls.field.setValue(this.config.scope === 'external'
|
|
178
|
+
? this.advancedSearchOptions.externalFields[0]
|
|
179
|
+
: this.advancedSearchOptions.localFields[0]);
|
|
180
|
+
// Add the qualifier value, if available and a local search.
|
|
181
|
+
if (this.config.scope === 'local' && queryRow.qualifier)
|
|
182
|
+
queryControl.controls.qualifier.setValue(queryRow.qualifier);
|
|
183
|
+
// Add the boolean, if available and not the first query.
|
|
184
|
+
if (i !== 0 && queryRow.boolean)
|
|
185
|
+
queryControl.controls.boolean.setValue(queryRow.boolean);
|
|
186
|
+
});
|
|
187
|
+
if (this.config.scope === 'local') {
|
|
188
|
+
this.types.setValue(this.config.localAdvancedSearch.selectedResourceTypes);
|
|
189
|
+
this.collections.setValue(this.config.localAdvancedSearch.selectedCollections);
|
|
190
|
+
}
|
|
191
|
+
else {
|
|
192
|
+
this.languages.setValue(this.config.externalAdvancedSearch
|
|
193
|
+
.selectedLanguages);
|
|
194
|
+
this.peerReviewed.setValue(this.config.externalAdvancedSearch.limitResults.peerReviewed);
|
|
195
|
+
this.applyRelatedWords.setValue(this.config.externalAdvancedSearch.expandResults.applyRelatedWords);
|
|
196
|
+
this.applyEquivalentSubjects.setValue(this.config.externalAdvancedSearch.expandResults.applyEquivalentSubjects);
|
|
197
|
+
this.fullText.setValue(this.config.externalAdvancedSearch.expandResults.fullText);
|
|
198
|
+
this.from.setValue(this.config.externalAdvancedSearch.datePublished.from);
|
|
199
|
+
this.to.setValue(this.config.externalAdvancedSearch.datePublished.to);
|
|
200
|
+
}
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
set config(config) {
|
|
204
|
+
this._config = config;
|
|
205
|
+
this.setupForm();
|
|
206
|
+
}
|
|
207
|
+
get config() {
|
|
208
|
+
return this._config;
|
|
209
|
+
}
|
|
210
|
+
ngOnDestroy() {
|
|
211
|
+
this.subscription.unsubscribe();
|
|
212
|
+
}
|
|
213
|
+
get queries() {
|
|
214
|
+
return this.advancedSearchForm.controls.queries;
|
|
215
|
+
}
|
|
216
|
+
get types() {
|
|
217
|
+
return this.advancedSearchForm.controls.types;
|
|
218
|
+
}
|
|
219
|
+
get collections() {
|
|
220
|
+
return this.advancedSearchForm.controls.collections;
|
|
221
|
+
}
|
|
222
|
+
get languages() {
|
|
223
|
+
return this.advancedSearchForm.controls.languages;
|
|
224
|
+
}
|
|
225
|
+
get from() {
|
|
226
|
+
return this.advancedSearchForm.controls.datePublished.controls.from;
|
|
227
|
+
}
|
|
228
|
+
get to() {
|
|
229
|
+
return this.advancedSearchForm.controls.datePublished.controls.to;
|
|
230
|
+
}
|
|
231
|
+
get peerReviewed() {
|
|
232
|
+
return this.advancedSearchForm.controls.limitResults.controls.peerReviewed;
|
|
233
|
+
}
|
|
234
|
+
get fullText() {
|
|
235
|
+
return this.advancedSearchForm.controls.expandResults.controls.fullText;
|
|
236
|
+
}
|
|
237
|
+
get applyRelatedWords() {
|
|
238
|
+
return this.advancedSearchForm.controls.expandResults.controls.applyRelatedWords;
|
|
239
|
+
}
|
|
240
|
+
get applyEquivalentSubjects() {
|
|
241
|
+
return this.advancedSearchForm.controls.expandResults.controls.applyEquivalentSubjects;
|
|
242
|
+
}
|
|
243
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: AdvancedSearchComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
244
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: AdvancedSearchComponent, isStandalone: true, selector: "lib-ss-advanced-search", inputs: { config: "config" }, outputs: { advancedSearch: "advancedSearch" }, viewQueries: [{ propertyName: "searchInput", first: true, predicate: ["searchInput"], descendants: true }, { propertyName: "inputTooltip", first: true, predicate: MatTooltip, descendants: true }], ngImport: i0, template: "<form\n data-testid=\"advancedSearchForm\"\n (submit)=\"config.scope === 'local' ? doLocalAdvancedSearch() : doExternalAdvancedSearch()\"\n [formGroup]=\"advancedSearchForm\"\n class=\"adv-search-wrapper\"\n>\n <h3>Advanced Search</h3>\n <!-- Queries -->\n @for (query of queries.controls; track $index; let queryIndex = $index) {\n <div class=\"adv-search-row\" [formGroup]=\"query\">\n <!-- Boolean operator -->\n <!-- Add boolean for each row except for first -->\n @if (!$first) {\n <div class=\"select-wrapper\">\n <select\n [id]=\"'boolean' + queryIndex\"\n [formControlName]=\"'boolean'\"\n [attr.data-testid]=\"'boolean' + queryIndex\"\n >\n @for (boolean of advancedSearchOptions.boolean; track boolean) {\n <option [ngValue]=\"boolean\">\n {{ boolean }}\n </option>\n }\n </select>\n </div>\n }\n <!-- Field -->\n <label [for]=\"'field' + queryIndex\" class=\"hidden\">Field {{ queryIndex }}</label>\n <div class=\"select-wrapper\">\n <select\n [id]=\"'field' + queryIndex\"\n [formControlName]=\"'field'\"\n [attr.data-testid]=\"'field' + queryIndex\"\n >\n @for (\n field of config.scope === 'external'\n ? advancedSearchOptions.externalFields\n : advancedSearchOptions.localFields;\n track field\n ) {\n <option [ngValue]=\"field\">\n {{ advancedSearchFieldMap[field] }}\n </option>\n }\n </select>\n </div>\n <!-- Qualifiers are only available for local searches -->\n @if (config.scope === 'local') {\n <label [for]=\"'qualifier' + queryIndex\" class=\"hidden\"\n >Match Criteria {{ queryIndex }}</label\n >\n <div class=\"select-wrapper\">\n <select\n [id]=\"'qualifier' + queryIndex\"\n [formControlName]=\"'qualifier'\"\n [attr.data-testid]=\"'qualifier' + queryIndex\"\n >\n <!-- $any because a field can technically be local or external, though this block is scoped strictly to local fields -->\n @for (\n qualifier of $any(advancedSearchOptions.qualifiers)[\n query.controls.field.value\n ];\n track qualifier;\n let optionIndex = $index\n ) {\n <option\n [ngValue]=\"qualifier\"\n [attr.data-testid]=\"\n 'qualifier' + queryIndex + '-option' + optionIndex\n \"\n >\n {{ $any(advancedSearchQualifierMap)[qualifier] }}\n </option>\n }\n </select>\n </div>\n }\n <!-- Query -->\n <label [for]=\"'query' + queryIndex\" class=\"hidden\">Query {{ queryIndex }}</label>\n <div class=\"query-input\">\n <input\n #searchInput\n [id]=\"'query' + queryIndex\"\n [formControlName]=\"'query'\"\n [attr.data-testid]=\"'query' + queryIndex\"\n required\n aria-required=\"true\"\n matTooltip=\"Fill out this field\"\n [matTooltipPosition]=\"'above'\"\n [matTooltipDisabled]=\"true\"\n [attr.aria-invalid]=\"isSubmitted && queries.at(0).controls.query.invalid\"\n aria-describedby=\"invalidInputDesc\"\n />\n </div>\n @if (!$first) {\n <button\n class=\"row-cancel\"\n type=\"button\"\n (click)=\"removeQuery(queryIndex)\"\n [attr.data-testid]=\"'cancelRow' + queryIndex\"\n >\n <span class=\"material-symbols-outlined icon\"> cancel </span>\n </button>\n }\n </div>\n }\n <button\n type=\"button\"\n (click)=\"addQuery()\"\n id=\"addQueryBtn\"\n data-testid=\"addQuery\"\n [disabled]=\"queries.length > 11\"\n >\n <span class=\"material-symbols-outlined icon\"> add </span>\n Add a row\n </button>\n <!-- OTHER OPTIONS -->\n <div class=\"other-options\">\n <!-- LOCAL -->\n @if (config.scope === 'local') {\n <!-- Type -->\n <div data-testid=\"resourceType\">\n <h4>Resource Type</h4>\n <div class=\"multi-select-wrapper\">\n <lib-hbll-multi-select\n [label]=\"'Resource Type'\"\n [allOptions]=\"config.localAdvancedSearch.availableResourceTypes\"\n [selectedKeys]=\"types.value\"\n (selectedKeysChange)=\"types.setValue($event)\"\n ></lib-hbll-multi-select>\n </div>\n </div>\n <!-- Collection -->\n <div data-testid=\"collection\">\n <h4>Collection</h4>\n <div class=\"multi-select-wrapper\">\n <lib-hbll-multi-select\n [label]=\"'Collection'\"\n [allOptions]=\"config.localAdvancedSearch.availableCollections\"\n [selectedKeys]=\"collections.value\"\n (selectedKeysChange)=\"collections.setValue($event)\"\n ></lib-hbll-multi-select>\n </div>\n </div>\n }\n <!-- EXTERNAL -->\n @if (config.scope === 'external') {\n <div formGroupName=\"limitResults\" class=\"checkbox-section\">\n <h4>For Fewer Results Try</h4>\n <!-- Peer reviewed -->\n <label class=\"checkbox-label\" for=\"peerReviewed\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"peerReviewed\"\n type=\"checkbox\"\n [formControlName]=\"'peerReviewed'\"\n data-testid=\"peerReviewed\"\n />\n <lib-hbll-checkbox [isChecked]=\"peerReviewed.value\"></lib-hbll-checkbox>\n <span class=\"label-text\">Peer reviewed journal articles</span>\n </label>\n </div>\n <div formGroupName=\"expandResults\" class=\"checkbox-section\">\n <h4>For More Results Try</h4>\n <!-- Apply related words -->\n <label class=\"checkbox-label\" for=\"applyRelatedWords\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"applyRelatedWords\"\n type=\"checkbox\"\n [formControlName]=\"'applyRelatedWords'\"\n data-testid=\"applyRelatedWords\"\n />\n <lib-hbll-checkbox [isChecked]=\"applyRelatedWords.value\"></lib-hbll-checkbox>\n <span class=\"label-text\">Apply related words</span>\n </label>\n <!-- Apply equivalent subjects -->\n <label class=\"checkbox-label\" for=\"applyEquivalentSubjects\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"applyEquivalentSubjects\"\n type=\"checkbox\"\n [formControlName]=\"'applyEquivalentSubjects'\"\n data-testid=\"applyEquivalentSubjects\"\n />\n <lib-hbll-checkbox\n [isChecked]=\"applyEquivalentSubjects.value\"\n ></lib-hbll-checkbox>\n <span class=\"label-text\">Apply equivalent subjects</span>\n </label>\n <!-- Full text -->\n <label class=\"checkbox-label\" for=\"fullText\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"fullText\"\n type=\"checkbox\"\n [formControlName]=\"'fullText'\"\n data-testid=\"fullText\"\n />\n <lib-hbll-checkbox [isChecked]=\"fullText.value\"></lib-hbll-checkbox>\n <span class=\"label-text\"\n >Include results the library doesn't have access to</span\n >\n </label>\n </div>\n <!-- Date -->\n <div id=\"dateOptions\">\n <h4>Date Published</h4>\n <lib-date-range\n [from]=\"from.value\"\n [to]=\"to.value\"\n (fromChange)=\"from.setValue($event)\"\n (toChange)=\"to.setValue($event)\"\n ></lib-date-range>\n </div>\n <!-- Language -->\n <div>\n <h4>Language</h4>\n <lib-hbll-multi-select\n [label]=\"'language'\"\n [allOptions]=\"config.externalAdvancedSearch.availableLanguages\"\n [selectedKeys]=\"languages.value\"\n (selectedKeysChange)=\"languages.setValue($any($event))\"\n ></lib-hbll-multi-select>\n </div>\n }\n </div>\n <div id=\"advSearchFooter\">\n <button libHbllPillBtn [color]=\"'blue'\" type=\"submit\" data-testid=\"advSearchSubmitButton\">\n Search\n </button>\n </div>\n</form>\n", styles: ["select,textarea,input{appearance:none;font-family:inherit;padding:.38em;border:solid 1px #707070;border-radius:4px;font-size:1em;background-color:#fff}select:focus,textarea:focus,input:focus{border-color:#3a6093}.select-wrapper{position:relative}.select-wrapper select{cursor:pointer;padding-right:2em}.select-wrapper:after{font-family:Material Symbols Outlined;content:\"arrow_drop_down\";pointer-events:none;top:0;font-size:1.5em;right:.2em;height:100%;position:absolute;display:flex;align-items:center;opacity:70%}a,button{border:none;background:none;font-family:inherit;padding:0;margin:0;font-size:inherit;color:#1c7ec9;text-decoration:none;cursor:pointer}a:hover,button:hover{color:#8ab6f0}:host{box-sizing:border-box}:host *{box-sizing:inherit}.select-wrapper{margin-right:.3em}.row-cancel{color:#aaa;margin-left:.4em;margin-top:.4em}.row-cancel:hover{color:#666}.row-cancel .icon{font-size:1em}.hidden{display:none}#addQueryBtn{display:flex;align-items:center}#addQueryBtn:disabled{color:#999;pointer-events:none}#addQueryBtn .icon{font-size:1em}.adv-search-wrapper{width:100%}h3{font-size:1.13em;font-weight:300;margin-bottom:.7em}h4{margin-bottom:.3em;font-weight:600}.adv-search-row{display:flex;flex-wrap:wrap;width:100%;margin-bottom:.6em}.query-input{display:inline-block;width:100%;margin-top:.4rem}.query-input input{width:100%}.other-options{display:grid;grid-template-columns:repeat(1,1fr);gap:.6em 1.6em;grid-auto-rows:minmax(1em,auto);margin-top:1.4em}.other-options label{margin-right:1rem}.checkbox-label{display:inline-flex;align-items:flex-start;cursor:pointer}.checkbox-label:not(:last-of-type){margin-bottom:.75em}.checkbox-label .label-text{margin-top:-.1em;margin-left:.45em}.checkbox-section{margin-bottom:.8em}#advSearchFooter{margin-bottom:.5em;display:flex;justify-content:flex-end;font-size:1.2em}@media screen and (min-width: 615px){.adv-search-row{flex-wrap:nowrap}.other-options{grid-template-columns:repeat(2,1fr)}.query-input{margin-top:0}#dateOptions{grid-column:span 2}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i2.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: HbllMultiSelectComponent, selector: "lib-hbll-multi-select", inputs: ["allOptions", "label", "selectedKeys"], outputs: ["selectedKeysChange"] }, { kind: "component", type: HbllCheckboxComponent, selector: "lib-hbll-checkbox", inputs: ["isChecked"] }, { kind: "directive", type: HbllPillBtnDirective, selector: "[libHbllPillBtn]", inputs: ["isDestructiveStyle", "isDisabled", "isFullWidth", "color"] }, { kind: "component", type: DateRangeComponent, selector: "lib-date-range", inputs: ["from", "to"], outputs: ["validDateChange", "fromChange", "toChange"] }] }); }
|
|
245
|
+
}
|
|
246
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: AdvancedSearchComponent, decorators: [{
|
|
247
|
+
type: Component,
|
|
248
|
+
args: [{ selector: 'lib-ss-advanced-search', standalone: true, imports: [
|
|
249
|
+
CommonModule,
|
|
250
|
+
ReactiveFormsModule,
|
|
251
|
+
MatTooltipModule,
|
|
252
|
+
MatIconModule,
|
|
253
|
+
HbllMultiSelectComponent,
|
|
254
|
+
HbllCheckboxComponent,
|
|
255
|
+
HbllPillBtnDirective,
|
|
256
|
+
DateRangeComponent,
|
|
257
|
+
], template: "<form\n data-testid=\"advancedSearchForm\"\n (submit)=\"config.scope === 'local' ? doLocalAdvancedSearch() : doExternalAdvancedSearch()\"\n [formGroup]=\"advancedSearchForm\"\n class=\"adv-search-wrapper\"\n>\n <h3>Advanced Search</h3>\n <!-- Queries -->\n @for (query of queries.controls; track $index; let queryIndex = $index) {\n <div class=\"adv-search-row\" [formGroup]=\"query\">\n <!-- Boolean operator -->\n <!-- Add boolean for each row except for first -->\n @if (!$first) {\n <div class=\"select-wrapper\">\n <select\n [id]=\"'boolean' + queryIndex\"\n [formControlName]=\"'boolean'\"\n [attr.data-testid]=\"'boolean' + queryIndex\"\n >\n @for (boolean of advancedSearchOptions.boolean; track boolean) {\n <option [ngValue]=\"boolean\">\n {{ boolean }}\n </option>\n }\n </select>\n </div>\n }\n <!-- Field -->\n <label [for]=\"'field' + queryIndex\" class=\"hidden\">Field {{ queryIndex }}</label>\n <div class=\"select-wrapper\">\n <select\n [id]=\"'field' + queryIndex\"\n [formControlName]=\"'field'\"\n [attr.data-testid]=\"'field' + queryIndex\"\n >\n @for (\n field of config.scope === 'external'\n ? advancedSearchOptions.externalFields\n : advancedSearchOptions.localFields;\n track field\n ) {\n <option [ngValue]=\"field\">\n {{ advancedSearchFieldMap[field] }}\n </option>\n }\n </select>\n </div>\n <!-- Qualifiers are only available for local searches -->\n @if (config.scope === 'local') {\n <label [for]=\"'qualifier' + queryIndex\" class=\"hidden\"\n >Match Criteria {{ queryIndex }}</label\n >\n <div class=\"select-wrapper\">\n <select\n [id]=\"'qualifier' + queryIndex\"\n [formControlName]=\"'qualifier'\"\n [attr.data-testid]=\"'qualifier' + queryIndex\"\n >\n <!-- $any because a field can technically be local or external, though this block is scoped strictly to local fields -->\n @for (\n qualifier of $any(advancedSearchOptions.qualifiers)[\n query.controls.field.value\n ];\n track qualifier;\n let optionIndex = $index\n ) {\n <option\n [ngValue]=\"qualifier\"\n [attr.data-testid]=\"\n 'qualifier' + queryIndex + '-option' + optionIndex\n \"\n >\n {{ $any(advancedSearchQualifierMap)[qualifier] }}\n </option>\n }\n </select>\n </div>\n }\n <!-- Query -->\n <label [for]=\"'query' + queryIndex\" class=\"hidden\">Query {{ queryIndex }}</label>\n <div class=\"query-input\">\n <input\n #searchInput\n [id]=\"'query' + queryIndex\"\n [formControlName]=\"'query'\"\n [attr.data-testid]=\"'query' + queryIndex\"\n required\n aria-required=\"true\"\n matTooltip=\"Fill out this field\"\n [matTooltipPosition]=\"'above'\"\n [matTooltipDisabled]=\"true\"\n [attr.aria-invalid]=\"isSubmitted && queries.at(0).controls.query.invalid\"\n aria-describedby=\"invalidInputDesc\"\n />\n </div>\n @if (!$first) {\n <button\n class=\"row-cancel\"\n type=\"button\"\n (click)=\"removeQuery(queryIndex)\"\n [attr.data-testid]=\"'cancelRow' + queryIndex\"\n >\n <span class=\"material-symbols-outlined icon\"> cancel </span>\n </button>\n }\n </div>\n }\n <button\n type=\"button\"\n (click)=\"addQuery()\"\n id=\"addQueryBtn\"\n data-testid=\"addQuery\"\n [disabled]=\"queries.length > 11\"\n >\n <span class=\"material-symbols-outlined icon\"> add </span>\n Add a row\n </button>\n <!-- OTHER OPTIONS -->\n <div class=\"other-options\">\n <!-- LOCAL -->\n @if (config.scope === 'local') {\n <!-- Type -->\n <div data-testid=\"resourceType\">\n <h4>Resource Type</h4>\n <div class=\"multi-select-wrapper\">\n <lib-hbll-multi-select\n [label]=\"'Resource Type'\"\n [allOptions]=\"config.localAdvancedSearch.availableResourceTypes\"\n [selectedKeys]=\"types.value\"\n (selectedKeysChange)=\"types.setValue($event)\"\n ></lib-hbll-multi-select>\n </div>\n </div>\n <!-- Collection -->\n <div data-testid=\"collection\">\n <h4>Collection</h4>\n <div class=\"multi-select-wrapper\">\n <lib-hbll-multi-select\n [label]=\"'Collection'\"\n [allOptions]=\"config.localAdvancedSearch.availableCollections\"\n [selectedKeys]=\"collections.value\"\n (selectedKeysChange)=\"collections.setValue($event)\"\n ></lib-hbll-multi-select>\n </div>\n </div>\n }\n <!-- EXTERNAL -->\n @if (config.scope === 'external') {\n <div formGroupName=\"limitResults\" class=\"checkbox-section\">\n <h4>For Fewer Results Try</h4>\n <!-- Peer reviewed -->\n <label class=\"checkbox-label\" for=\"peerReviewed\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"peerReviewed\"\n type=\"checkbox\"\n [formControlName]=\"'peerReviewed'\"\n data-testid=\"peerReviewed\"\n />\n <lib-hbll-checkbox [isChecked]=\"peerReviewed.value\"></lib-hbll-checkbox>\n <span class=\"label-text\">Peer reviewed journal articles</span>\n </label>\n </div>\n <div formGroupName=\"expandResults\" class=\"checkbox-section\">\n <h4>For More Results Try</h4>\n <!-- Apply related words -->\n <label class=\"checkbox-label\" for=\"applyRelatedWords\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"applyRelatedWords\"\n type=\"checkbox\"\n [formControlName]=\"'applyRelatedWords'\"\n data-testid=\"applyRelatedWords\"\n />\n <lib-hbll-checkbox [isChecked]=\"applyRelatedWords.value\"></lib-hbll-checkbox>\n <span class=\"label-text\">Apply related words</span>\n </label>\n <!-- Apply equivalent subjects -->\n <label class=\"checkbox-label\" for=\"applyEquivalentSubjects\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"applyEquivalentSubjects\"\n type=\"checkbox\"\n [formControlName]=\"'applyEquivalentSubjects'\"\n data-testid=\"applyEquivalentSubjects\"\n />\n <lib-hbll-checkbox\n [isChecked]=\"applyEquivalentSubjects.value\"\n ></lib-hbll-checkbox>\n <span class=\"label-text\">Apply equivalent subjects</span>\n </label>\n <!-- Full text -->\n <label class=\"checkbox-label\" for=\"fullText\" tabindex=\"0\">\n <input\n class=\"hidden\"\n id=\"fullText\"\n type=\"checkbox\"\n [formControlName]=\"'fullText'\"\n data-testid=\"fullText\"\n />\n <lib-hbll-checkbox [isChecked]=\"fullText.value\"></lib-hbll-checkbox>\n <span class=\"label-text\"\n >Include results the library doesn't have access to</span\n >\n </label>\n </div>\n <!-- Date -->\n <div id=\"dateOptions\">\n <h4>Date Published</h4>\n <lib-date-range\n [from]=\"from.value\"\n [to]=\"to.value\"\n (fromChange)=\"from.setValue($event)\"\n (toChange)=\"to.setValue($event)\"\n ></lib-date-range>\n </div>\n <!-- Language -->\n <div>\n <h4>Language</h4>\n <lib-hbll-multi-select\n [label]=\"'language'\"\n [allOptions]=\"config.externalAdvancedSearch.availableLanguages\"\n [selectedKeys]=\"languages.value\"\n (selectedKeysChange)=\"languages.setValue($any($event))\"\n ></lib-hbll-multi-select>\n </div>\n }\n </div>\n <div id=\"advSearchFooter\">\n <button libHbllPillBtn [color]=\"'blue'\" type=\"submit\" data-testid=\"advSearchSubmitButton\">\n Search\n </button>\n </div>\n</form>\n", styles: ["select,textarea,input{appearance:none;font-family:inherit;padding:.38em;border:solid 1px #707070;border-radius:4px;font-size:1em;background-color:#fff}select:focus,textarea:focus,input:focus{border-color:#3a6093}.select-wrapper{position:relative}.select-wrapper select{cursor:pointer;padding-right:2em}.select-wrapper:after{font-family:Material Symbols Outlined;content:\"arrow_drop_down\";pointer-events:none;top:0;font-size:1.5em;right:.2em;height:100%;position:absolute;display:flex;align-items:center;opacity:70%}a,button{border:none;background:none;font-family:inherit;padding:0;margin:0;font-size:inherit;color:#1c7ec9;text-decoration:none;cursor:pointer}a:hover,button:hover{color:#8ab6f0}:host{box-sizing:border-box}:host *{box-sizing:inherit}.select-wrapper{margin-right:.3em}.row-cancel{color:#aaa;margin-left:.4em;margin-top:.4em}.row-cancel:hover{color:#666}.row-cancel .icon{font-size:1em}.hidden{display:none}#addQueryBtn{display:flex;align-items:center}#addQueryBtn:disabled{color:#999;pointer-events:none}#addQueryBtn .icon{font-size:1em}.adv-search-wrapper{width:100%}h3{font-size:1.13em;font-weight:300;margin-bottom:.7em}h4{margin-bottom:.3em;font-weight:600}.adv-search-row{display:flex;flex-wrap:wrap;width:100%;margin-bottom:.6em}.query-input{display:inline-block;width:100%;margin-top:.4rem}.query-input input{width:100%}.other-options{display:grid;grid-template-columns:repeat(1,1fr);gap:.6em 1.6em;grid-auto-rows:minmax(1em,auto);margin-top:1.4em}.other-options label{margin-right:1rem}.checkbox-label{display:inline-flex;align-items:flex-start;cursor:pointer}.checkbox-label:not(:last-of-type){margin-bottom:.75em}.checkbox-label .label-text{margin-top:-.1em;margin-left:.45em}.checkbox-section{margin-bottom:.8em}#advSearchFooter{margin-bottom:.5em;display:flex;justify-content:flex-end;font-size:1.2em}@media screen and (min-width: 615px){.adv-search-row{flex-wrap:nowrap}.other-options{grid-template-columns:repeat(2,1fr)}.query-input{margin-top:0}#dateOptions{grid-column:span 2}}\n"] }]
|
|
258
|
+
}], propDecorators: { config: [{
|
|
259
|
+
type: Input,
|
|
260
|
+
args: [{ required: true }]
|
|
261
|
+
}], advancedSearch: [{
|
|
262
|
+
type: Output
|
|
263
|
+
}], searchInput: [{
|
|
264
|
+
type: ViewChild,
|
|
265
|
+
args: ['searchInput']
|
|
266
|
+
}], inputTooltip: [{
|
|
267
|
+
type: ViewChild,
|
|
268
|
+
args: [MatTooltip]
|
|
269
|
+
}] } });
|
|
270
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"advanced-search.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/ss-search-bar/advanced-search/advanced-search.component.ts","../../../../../../projects/components/src/lib/ss-search-bar/advanced-search/advanced-search.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    Output,\n    ViewChild,\n    inject,\n} from '@angular/core';\nimport {\n    FormControl,\n    FormGroup,\n    NonNullableFormBuilder,\n    ReactiveFormsModule,\n    Validators,\n} from '@angular/forms';\nimport { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';\nimport { filter, startWith, tap } from 'rxjs/operators';\nimport {\n    AdvancedSearchBooleanOption,\n    AdvancedSearchLocalFieldOption,\n    AdvancedSearchLanguageOption,\n    AdvancedSearchQualifierOption,\n    ADVANCED_SEARCH_OPTIONS,\n    ADVANCED_SEARCH_QUERIES_LIMIT,\n    AdvancedSearchQualifier,\n    AdvancedSearchExternalFieldOption,\n    AdvancedSearchQueryRow,\n} from '../ss-search-bar.component';\nimport { Subscription } from 'rxjs';\nimport { SearchConfig } from '../ss-search-bar.component';\nimport { validateDateRangeFields } from '../utils';\nimport { CommonModule } from '@angular/common';\nimport { ADVANCED_SEARCH_FIELD_MAP, ADVANCED_SEARCH_QUALIFIER_MAP } from '../constants';\nimport { MatIconModule } from '@angular/material/icon';\nimport { HbllMultiSelectComponent } from '../../hbll-multi-select/hbll-multi-select.component';\nimport { HbllCheckboxComponent } from '../../hbll-checkbox/hbll-checkbox.component';\nimport { HbllPillBtnDirective } from '../../directives/hbll-pill-btn/hbll-pill-btn.directive';\nimport { DateRangeComponent } from '../date-range/date-range.component';\n\ntype QueryRow = {\n    boolean: FormControl<AdvancedSearchBooleanOption>;\n    field: FormControl<AdvancedSearchLocalFieldOption | AdvancedSearchExternalFieldOption>;\n    qualifier: FormControl<AdvancedSearchQualifierOption>;\n    query: FormControl<string>;\n};\n\nconst DATE_VALIDATOR_REGEX = '^[0-9]{4}$';\n\n@Component({\n    selector: 'lib-ss-advanced-search',\n    standalone: true,\n    templateUrl: './advanced-search.component.html',\n    styleUrls: ['./advanced-search.component.scss'],\n    imports: [\n        CommonModule,\n        ReactiveFormsModule,\n        MatTooltipModule,\n        MatIconModule,\n        HbllMultiSelectComponent,\n        HbllCheckboxComponent,\n        HbllPillBtnDirective,\n        DateRangeComponent,\n    ],\n})\nexport class AdvancedSearchComponent implements OnDestroy {\n    private readonly fb = inject(NonNullableFormBuilder);\n    private _config!: SearchConfig;\n    @Input({ required: true }) set config(config: SearchConfig) {\n        this._config = config;\n        this.setupForm();\n    }\n    get config() {\n        return this._config;\n    }\n    @Output() advancedSearch = new EventEmitter<SearchConfig>();\n\n    // Though multiple instances exists, this represent query 1.\n    @ViewChild('searchInput') private searchInput!: ElementRef;\n    // The tooltip is disabled in the template.\n    // It is enabled and displayed if a user searches against an empty query (simple or q1).\n    // Any change to the search field disables the tooltip.\n    @ViewChild(MatTooltip) private inputTooltip!: MatTooltip;\n\n    private subscription = new Subscription();\n\n    ngOnDestroy(): void {\n        this.subscription.unsubscribe();\n    }\n\n    /**\n     * Takes a `FormGroup` and syncs up the `field` and `qualifier` values.\n     * After syncing, when the `field` value changes, the `qualifier` value will update appropriately.\n     */\n    private syncUpFieldAndQualifier = (\n        fg: FormGroup<{\n            boolean: FormControl;\n            field: FormControl;\n            qualifier: FormControl;\n            query: FormControl;\n        }>,\n        scope: SearchConfig['scope'],\n    ) => {\n        this.subscription.add(\n            fg.controls.field.valueChanges\n                .pipe(\n                    filter(() => scope === 'local'),\n                    tap((val: AdvancedSearchQualifier) =>\n                        fg.controls.qualifier.setValue(ADVANCED_SEARCH_OPTIONS.qualifiers[val][0]),\n                    ),\n                )\n                .subscribe(),\n        );\n    };\n\n    protected advancedSearchForm = this.fb.group({\n        queries: this.fb.array<FormGroup<QueryRow>>(\n            [],\n            [Validators.maxLength(ADVANCED_SEARCH_QUERIES_LIMIT)],\n        ),\n        types: [[] as string[]],\n        collections: [[] as string[]],\n        languages: [[] as AdvancedSearchLanguageOption[]],\n        datePublished: this.fb.group(\n            {\n                from: ['', [Validators.maxLength(4), Validators.pattern(DATE_VALIDATOR_REGEX)]],\n                to: ['', [Validators.maxLength(4), Validators.pattern(DATE_VALIDATOR_REGEX)]],\n            },\n            { validators: validateDateRangeFields() },\n        ),\n        limitResults: this.fb.group({\n            peerReviewed: [false],\n        }),\n        expandResults: this.fb.group({\n            applyRelatedWords: [false],\n            applyEquivalentSubjects: [false],\n            fullText: [false],\n        }),\n    });\n\n    protected isSubmitted = false;\n    protected advancedSearchOptions = ADVANCED_SEARCH_OPTIONS;\n    protected advancedSearchFieldMap = ADVANCED_SEARCH_FIELD_MAP;\n    protected advancedSearchQualifierMap = ADVANCED_SEARCH_QUALIFIER_MAP;\n\n    get queries() {\n        return this.advancedSearchForm.controls.queries;\n    }\n    get types() {\n        return this.advancedSearchForm.controls.types;\n    }\n    get collections() {\n        return this.advancedSearchForm.controls.collections;\n    }\n    get languages() {\n        return this.advancedSearchForm.controls.languages;\n    }\n    get from() {\n        return this.advancedSearchForm.controls.datePublished.controls.from;\n    }\n    get to() {\n        return this.advancedSearchForm.controls.datePublished.controls.to;\n    }\n    get peerReviewed() {\n        return this.advancedSearchForm.controls.limitResults.controls.peerReviewed;\n    }\n    get fullText() {\n        return this.advancedSearchForm.controls.expandResults.controls.fullText;\n    }\n    get applyRelatedWords() {\n        return this.advancedSearchForm.controls.expandResults.controls.applyRelatedWords;\n    }\n    get applyEquivalentSubjects() {\n        return this.advancedSearchForm.controls.expandResults.controls.applyEquivalentSubjects;\n    }\n\n    /**\n     * Add a query to `queries` `FormArray`. This FormGroup has an additional `boolean` control.\n     */\n    protected addQuery = () => {\n        this.queries.push(\n            this.fb.group({\n                boolean: [ADVANCED_SEARCH_OPTIONS.boolean[0] as AdvancedSearchBooleanOption],\n                field: [\n                    this.config.scope === 'external'\n                        ? ADVANCED_SEARCH_OPTIONS.externalFields[0]\n                        : (ADVANCED_SEARCH_OPTIONS.localFields[0] as\n                              | AdvancedSearchLocalFieldOption\n                              | AdvancedSearchExternalFieldOption),\n                ],\n                qualifier: [\n                    ADVANCED_SEARCH_OPTIONS.qualifiers[\n                        ADVANCED_SEARCH_OPTIONS.localFields[0]\n                    ][0] as AdvancedSearchQualifierOption,\n                ],\n                query: ['', Validators.required],\n            }),\n        );\n        this.syncUpFieldAndQualifier(this.queries.at(-1), this.config.scope);\n    };\n\n    protected removeQuery = (i: number) => {\n        this.queries.removeAt(i);\n    };\n\n    /**\n     * Prepares appropriate query params before performing a local advanced search.\n     */\n    protected doLocalAdvancedSearch = () => {\n        this.doAdvancedSearch({\n            localAdvancedSearch: {\n                ...this.config.localAdvancedSearch,\n                selectedResourceTypes: this.types.value,\n                selectedCollections: this.collections.value,\n            },\n        });\n    };\n\n    /**\n     * Prepares appropriate query params before performing an external advanced search.\n     */\n    protected doExternalAdvancedSearch = () => {\n        // Compares 'from' and 'to' to ensure 'to' is greater than / equal to 'from', swaps them if not\n        const from = this.from.value\n            ? parseInt(this.from.value) <= parseInt(this.to.value)\n                ? this.from.value\n                : this.to.value\n            : '';\n        // Compares 'from' and 'to' to ensure 'to' is greater than / equal to 'from', swaps them if not\n        const to = this.to.value\n            ? parseInt(this.to.value) >= parseInt(this.from.value)\n                ? this.to.value\n                : this.from.value\n            : '';\n        this.doAdvancedSearch({\n            externalAdvancedSearch: {\n                ...this.config.externalAdvancedSearch,\n                datePublished: { from, to },\n                selectedLanguages: this.languages.value,\n                limitResults: {\n                    peerReviewed: this.peerReviewed.value,\n                },\n                expandResults: {\n                    applyRelatedWords: this.applyRelatedWords.value,\n                    applyEquivalentSubjects: this.applyEquivalentSubjects.value,\n                    fullText: this.fullText.value,\n                },\n            },\n        });\n    };\n\n    /**\n     * Verifies that there is a query to use, gathers all other query parameters needed, then performs an advanced search via `router.navigate()`.\n     * @param {Record<string, string>} queryParams base set of query params to navigate with\n     */\n    private doAdvancedSearch = (updatedProps: Partial<SearchConfig>): void => {\n        this.isSubmitted = true;\n        // Don't perform a search unless there is at least one query\n        if (\n            this.queries.at(0).controls.query.invalid &&\n            !this.queries.value.some((v) => !!v.query)\n        ) {\n            this.showSearchValidationToolTip();\n            return;\n        }\n        // Don't perform a search unless the date form is valid\n        if (this.advancedSearchForm.controls.datePublished.invalid) return;\n\n        this.advancedSearch.emit({\n            ...this.config,\n            ...updatedProps,\n            advancedSearchQueryRows: this.queries.value as AdvancedSearchQueryRow[],\n        });\n        this.isSubmitted = false;\n    };\n\n    private showSearchValidationToolTip = () => {\n        this.searchInput.nativeElement.focus();\n        this.inputTooltip.disabled = false;\n        this.inputTooltip.show();\n    };\n\n    private setupForm = () => {\n        this.queries.clear();\n        // Always keep at least two queries available\n        this.addQuery();\n        this.addQuery();\n\n        // Ensure validation tooltip closes when uses enters a query\n        this.subscription.add(\n            this.queries\n                .at(0)\n                .controls.query.valueChanges.pipe(startWith(''))\n                .pipe(filter(() => !!this.inputTooltip))\n                .subscribe(() => (this.inputTooltip.disabled = true)),\n        );\n\n        // If no `q{NUM}` use `q` for first query\n        if (!this.config.advancedSearchQueryRows.length)\n            this.queries.at(0).controls.query.setValue(this.config.q);\n\n        this.config.advancedSearchQueryRows.forEach((queryRow, i) => {\n            // Add a query form control for every q{num} key after the first.\n            let queryControl = null;\n            if (i === 0 || i === 1) {\n                queryControl = this.queries.at(i);\n            } else {\n                this.addQuery();\n                queryControl = this.queries.at(-1);\n            }\n\n            // Add the query value.\n            queryControl.controls.query.setValue(queryRow.query);\n\n            // Add the field value, if available.\n            if (queryRow.field) queryControl.controls.field.setValue(queryRow.field);\n            else\n                queryControl.controls.field.setValue(\n                    this.config.scope === 'external'\n                        ? this.advancedSearchOptions.externalFields[0]\n                        : this.advancedSearchOptions.localFields[0],\n                );\n\n            // Add the qualifier value, if available and a local search.\n            if (this.config.scope === 'local' && queryRow.qualifier)\n                queryControl.controls.qualifier.setValue(queryRow.qualifier);\n\n            // Add the boolean, if available and not the first query.\n            if (i !== 0 && queryRow.boolean)\n                queryControl.controls.boolean.setValue(queryRow.boolean);\n        });\n        if (this.config.scope === 'local') {\n            this.types.setValue(this.config.localAdvancedSearch.selectedResourceTypes);\n            this.collections.setValue(this.config.localAdvancedSearch.selectedCollections);\n        } else {\n            this.languages.setValue(\n                this.config.externalAdvancedSearch\n                    .selectedLanguages as AdvancedSearchLanguageOption[],\n            );\n            this.peerReviewed.setValue(\n                this.config.externalAdvancedSearch.limitResults.peerReviewed,\n            );\n            this.applyRelatedWords.setValue(\n                this.config.externalAdvancedSearch.expandResults.applyRelatedWords,\n            );\n            this.applyEquivalentSubjects.setValue(\n                this.config.externalAdvancedSearch.expandResults.applyEquivalentSubjects,\n            );\n            this.fullText.setValue(this.config.externalAdvancedSearch.expandResults.fullText);\n            this.from.setValue(this.config.externalAdvancedSearch.datePublished.from);\n            this.to.setValue(this.config.externalAdvancedSearch.datePublished.to);\n        }\n    };\n}\n","<form\n    data-testid=\"advancedSearchForm\"\n    (submit)=\"config.scope === 'local' ? doLocalAdvancedSearch() : doExternalAdvancedSearch()\"\n    [formGroup]=\"advancedSearchForm\"\n    class=\"adv-search-wrapper\"\n>\n    <h3>Advanced Search</h3>\n    <!-- Queries -->\n    @for (query of queries.controls; track $index; let queryIndex = $index) {\n        <div class=\"adv-search-row\" [formGroup]=\"query\">\n            <!-- Boolean operator -->\n            <!-- Add boolean for each row except for first -->\n            @if (!$first) {\n                <div class=\"select-wrapper\">\n                    <select\n                        [id]=\"'boolean' + queryIndex\"\n                        [formControlName]=\"'boolean'\"\n                        [attr.data-testid]=\"'boolean' + queryIndex\"\n                    >\n                        @for (boolean of advancedSearchOptions.boolean; track boolean) {\n                            <option [ngValue]=\"boolean\">\n                                {{ boolean }}\n                            </option>\n                        }\n                    </select>\n                </div>\n            }\n            <!-- Field -->\n            <label [for]=\"'field' + queryIndex\" class=\"hidden\">Field {{ queryIndex }}</label>\n            <div class=\"select-wrapper\">\n                <select\n                    [id]=\"'field' + queryIndex\"\n                    [formControlName]=\"'field'\"\n                    [attr.data-testid]=\"'field' + queryIndex\"\n                >\n                    @for (\n                        field of config.scope === 'external'\n                            ? advancedSearchOptions.externalFields\n                            : advancedSearchOptions.localFields;\n                        track field\n                    ) {\n                        <option [ngValue]=\"field\">\n                            {{ advancedSearchFieldMap[field] }}\n                        </option>\n                    }\n                </select>\n            </div>\n            <!-- Qualifiers are only available for local searches -->\n            @if (config.scope === 'local') {\n                <label [for]=\"'qualifier' + queryIndex\" class=\"hidden\"\n                    >Match Criteria {{ queryIndex }}</label\n                >\n                <div class=\"select-wrapper\">\n                    <select\n                        [id]=\"'qualifier' + queryIndex\"\n                        [formControlName]=\"'qualifier'\"\n                        [attr.data-testid]=\"'qualifier' + queryIndex\"\n                    >\n                        <!-- $any because a field can technically be local or external, though this block is scoped strictly to local fields -->\n                        @for (\n                            qualifier of $any(advancedSearchOptions.qualifiers)[\n                                query.controls.field.value\n                            ];\n                            track qualifier;\n                            let optionIndex = $index\n                        ) {\n                            <option\n                                [ngValue]=\"qualifier\"\n                                [attr.data-testid]=\"\n                                    'qualifier' + queryIndex + '-option' + optionIndex\n                                \"\n                            >\n                                {{ $any(advancedSearchQualifierMap)[qualifier] }}\n                            </option>\n                        }\n                    </select>\n                </div>\n            }\n            <!-- Query -->\n            <label [for]=\"'query' + queryIndex\" class=\"hidden\">Query {{ queryIndex }}</label>\n            <div class=\"query-input\">\n                <input\n                    #searchInput\n                    [id]=\"'query' + queryIndex\"\n                    [formControlName]=\"'query'\"\n                    [attr.data-testid]=\"'query' + queryIndex\"\n                    required\n                    aria-required=\"true\"\n                    matTooltip=\"Fill out this field\"\n                    [matTooltipPosition]=\"'above'\"\n                    [matTooltipDisabled]=\"true\"\n                    [attr.aria-invalid]=\"isSubmitted && queries.at(0).controls.query.invalid\"\n                    aria-describedby=\"invalidInputDesc\"\n                />\n            </div>\n            @if (!$first) {\n                <button\n                    class=\"row-cancel\"\n                    type=\"button\"\n                    (click)=\"removeQuery(queryIndex)\"\n                    [attr.data-testid]=\"'cancelRow' + queryIndex\"\n                >\n                    <span class=\"material-symbols-outlined icon\"> cancel </span>\n                </button>\n            }\n        </div>\n    }\n    <button\n        type=\"button\"\n        (click)=\"addQuery()\"\n        id=\"addQueryBtn\"\n        data-testid=\"addQuery\"\n        [disabled]=\"queries.length > 11\"\n    >\n        <span class=\"material-symbols-outlined icon\"> add </span>\n        Add a row\n    </button>\n    <!-- OTHER OPTIONS -->\n    <div class=\"other-options\">\n        <!-- LOCAL -->\n        @if (config.scope === 'local') {\n            <!-- Type -->\n            <div data-testid=\"resourceType\">\n                <h4>Resource Type</h4>\n                <div class=\"multi-select-wrapper\">\n                    <lib-hbll-multi-select\n                        [label]=\"'Resource Type'\"\n                        [allOptions]=\"config.localAdvancedSearch.availableResourceTypes\"\n                        [selectedKeys]=\"types.value\"\n                        (selectedKeysChange)=\"types.setValue($event)\"\n                    ></lib-hbll-multi-select>\n                </div>\n            </div>\n            <!-- Collection -->\n            <div data-testid=\"collection\">\n                <h4>Collection</h4>\n                <div class=\"multi-select-wrapper\">\n                    <lib-hbll-multi-select\n                        [label]=\"'Collection'\"\n                        [allOptions]=\"config.localAdvancedSearch.availableCollections\"\n                        [selectedKeys]=\"collections.value\"\n                        (selectedKeysChange)=\"collections.setValue($event)\"\n                    ></lib-hbll-multi-select>\n                </div>\n            </div>\n        }\n        <!-- EXTERNAL -->\n        @if (config.scope === 'external') {\n            <div formGroupName=\"limitResults\" class=\"checkbox-section\">\n                <h4>For Fewer Results Try</h4>\n                <!-- Peer reviewed -->\n                <label class=\"checkbox-label\" for=\"peerReviewed\" tabindex=\"0\">\n                    <input\n                        class=\"hidden\"\n                        id=\"peerReviewed\"\n                        type=\"checkbox\"\n                        [formControlName]=\"'peerReviewed'\"\n                        data-testid=\"peerReviewed\"\n                    />\n                    <lib-hbll-checkbox [isChecked]=\"peerReviewed.value\"></lib-hbll-checkbox>\n                    <span class=\"label-text\">Peer reviewed journal articles</span>\n                </label>\n            </div>\n            <div formGroupName=\"expandResults\" class=\"checkbox-section\">\n                <h4>For More Results Try</h4>\n                <!-- Apply related words -->\n                <label class=\"checkbox-label\" for=\"applyRelatedWords\" tabindex=\"0\">\n                    <input\n                        class=\"hidden\"\n                        id=\"applyRelatedWords\"\n                        type=\"checkbox\"\n                        [formControlName]=\"'applyRelatedWords'\"\n                        data-testid=\"applyRelatedWords\"\n                    />\n                    <lib-hbll-checkbox [isChecked]=\"applyRelatedWords.value\"></lib-hbll-checkbox>\n                    <span class=\"label-text\">Apply related words</span>\n                </label>\n                <!-- Apply equivalent subjects -->\n                <label class=\"checkbox-label\" for=\"applyEquivalentSubjects\" tabindex=\"0\">\n                    <input\n                        class=\"hidden\"\n                        id=\"applyEquivalentSubjects\"\n                        type=\"checkbox\"\n                        [formControlName]=\"'applyEquivalentSubjects'\"\n                        data-testid=\"applyEquivalentSubjects\"\n                    />\n                    <lib-hbll-checkbox\n                        [isChecked]=\"applyEquivalentSubjects.value\"\n                    ></lib-hbll-checkbox>\n                    <span class=\"label-text\">Apply equivalent subjects</span>\n                </label>\n                <!-- Full text -->\n                <label class=\"checkbox-label\" for=\"fullText\" tabindex=\"0\">\n                    <input\n                        class=\"hidden\"\n                        id=\"fullText\"\n                        type=\"checkbox\"\n                        [formControlName]=\"'fullText'\"\n                        data-testid=\"fullText\"\n                    />\n                    <lib-hbll-checkbox [isChecked]=\"fullText.value\"></lib-hbll-checkbox>\n                    <span class=\"label-text\"\n                        >Include results the library doesn't have access to</span\n                    >\n                </label>\n            </div>\n            <!-- Date -->\n            <div id=\"dateOptions\">\n                <h4>Date Published</h4>\n                <lib-date-range\n                    [from]=\"from.value\"\n                    [to]=\"to.value\"\n                    (fromChange)=\"from.setValue($event)\"\n                    (toChange)=\"to.setValue($event)\"\n                ></lib-date-range>\n            </div>\n            <!-- Language -->\n            <div>\n                <h4>Language</h4>\n                <lib-hbll-multi-select\n                    [label]=\"'language'\"\n                    [allOptions]=\"config.externalAdvancedSearch.availableLanguages\"\n                    [selectedKeys]=\"languages.value\"\n                    (selectedKeysChange)=\"languages.setValue($any($event))\"\n                ></lib-hbll-multi-select>\n            </div>\n        }\n    </div>\n    <div id=\"advSearchFooter\">\n        <button libHbllPillBtn [color]=\"'blue'\" type=\"submit\" data-testid=\"advSearchSubmitButton\">\n            Search\n        </button>\n    </div>\n</form>\n"]}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
export const ADVANCED_SEARCH_QUALIFIER_MAP = {
|
|
2
|
+
contains: 'contains',
|
|
3
|
+
begins_with: 'begins with',
|
|
4
|
+
exact: 'is (exact)',
|
|
5
|
+
index: 'browse value',
|
|
6
|
+
subject_lc: 'LC browse',
|
|
7
|
+
subject_expanded: 'expanded browse',
|
|
8
|
+
};
|
|
9
|
+
export const ADVANCED_SEARCH_FIELD_MAP = {
|
|
10
|
+
abstract: 'Abstract',
|
|
11
|
+
allText: 'All Text',
|
|
12
|
+
any: 'Any Field',
|
|
13
|
+
callNumber: 'Call Number',
|
|
14
|
+
contributor: 'Contributor',
|
|
15
|
+
creator: 'Author/Creator',
|
|
16
|
+
genre: 'Genre',
|
|
17
|
+
isbn: 'ISBN',
|
|
18
|
+
issn: 'ISSN',
|
|
19
|
+
series: 'Series',
|
|
20
|
+
source: 'Source',
|
|
21
|
+
subject: 'Subject',
|
|
22
|
+
title: 'Title',
|
|
23
|
+
};
|
|
24
|
+
export const PRIMARY_BLUE = 'rgba(63, 112, 176, 1)';
|
|
25
|
+
export const PRIMARY_BLUE_HOVER = 'hsl(214, 48%, 60%, 1)';
|
|
26
|
+
export const PRIMARY_LIGHT_BLUE = 'rgba(223, 233, 247, 1)';
|
|
27
|
+
export const PRIMARY_PURPLE = '#8772A6';
|
|
28
|
+
export const PRIMARY_PURPLE_HOVER = '#7C659F';
|
|
29
|
+
export const DESTRUCTIVE_RED = 'hsl(5, 47%, 47%, 1)';
|
|
30
|
+
export const DESTRUCTIVE_RED_HOVER = 'hsl(5, 47%, 50%, 1)';
|
|
31
|
+
export const SECONDARY_TEXT_GRAY = 'rgba(112, 112, 112, 1)';
|
|
32
|
+
export const LIGHT_GRAY = 'rgba(230, 230, 230, 1)';
|
|
33
|
+
export const DROP_SHADOW_DARK_BLUE = 'rgba(51, 62, 77, 0.16)';
|
|
34
|
+
export const TEXT_SEMIBOLD = '600';
|
|
35
|
+
export const TEXT_BOLD = '700';
|
|
36
|
+
export const ANIMATION_LENGTH_STD = '0.15s';
|
|
37
|
+
export const LINE_HEIGHT_STD = '1.4';
|
|
38
|
+
export const CONTEXTUAL_SELECTOR_HEIGHT = '4.5rem';
|
|
39
|
+
export const CONTENT_MAX_WIDTH = '75rem';
|
|
40
|
+
export const CONTENT_PADDING = '5.4rem';
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50cy9zcmMvbGliL3NzLXNlYXJjaC1iYXIvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLDZCQUE2QixHQUFHO0lBQ3pDLFFBQVEsRUFBRSxVQUFVO0lBQ3BCLFdBQVcsRUFBRSxhQUFhO0lBQzFCLEtBQUssRUFBRSxZQUFZO0lBQ25CLEtBQUssRUFBRSxjQUFjO0lBQ3JCLFVBQVUsRUFBRSxXQUFXO0lBQ3ZCLGdCQUFnQixFQUFFLGlCQUFpQjtDQUN0QyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0seUJBQXlCLEdBQUc7SUFDckMsUUFBUSxFQUFFLFVBQVU7SUFDcEIsT0FBTyxFQUFFLFVBQVU7SUFDbkIsR0FBRyxFQUFFLFdBQVc7SUFDaEIsVUFBVSxFQUFFLGFBQWE7SUFDekIsV0FBVyxFQUFFLGFBQWE7SUFDMUIsT0FBTyxFQUFFLGdCQUFnQjtJQUN6QixLQUFLLEVBQUUsT0FBTztJQUNkLElBQUksRUFBRSxNQUFNO0lBQ1osSUFBSSxFQUFFLE1BQU07SUFDWixNQUFNLEVBQUUsUUFBUTtJQUNoQixNQUFNLEVBQUUsUUFBUTtJQUNoQixPQUFPLEVBQUUsU0FBUztJQUNsQixLQUFLLEVBQUUsT0FBTztDQUNqQixDQUFDO0FBQ0YsTUFBTSxDQUFDLE1BQU0sWUFBWSxHQUFHLHVCQUF1QixDQUFDO0FBQ3BELE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLHVCQUF1QixDQUFDO0FBQzFELE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLHdCQUF3QixDQUFDO0FBQzNELE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBRyxTQUFTLENBQUM7QUFDeEMsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQUcsU0FBUyxDQUFDO0FBQzlDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRyxxQkFBcUIsQ0FBQztBQUNyRCxNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBRyxxQkFBcUIsQ0FBQztBQUMzRCxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyx3QkFBd0IsQ0FBQztBQUM1RCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUcsd0JBQXdCLENBQUM7QUFDbkQsTUFBTSxDQUFDLE1BQU0scUJBQXFCLEdBQUcsd0JBQXdCLENBQUM7QUFDOUQsTUFBTSxDQUFDLE1BQU0sYUFBYSxHQUFHLEtBQUssQ0FBQztBQUNuQyxNQUFNLENBQUMsTUFBTSxTQUFTLEdBQUcsS0FBSyxDQUFDO0FBQy9CLE1BQU0sQ0FBQyxNQUFNLG9CQUFvQixHQUFHLE9BQU8sQ0FBQztBQUM1QyxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsS0FBSyxDQUFDO0FBQ3JDLE1BQU0sQ0FBQyxNQUFNLDBCQUEwQixHQUFHLFFBQVEsQ0FBQztBQUNuRCxNQUFNLENBQUMsTUFBTSxpQkFBaUIsR0FBRyxPQUFPLENBQUM7QUFDekMsTUFBTSxDQUFDLE1BQU0sZUFBZSxHQUFHLFFBQVEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBBRFZBTkNFRF9TRUFSQ0hfUVVBTElGSUVSX01BUCA9IHtcbiAgICBjb250YWluczogJ2NvbnRhaW5zJyxcbiAgICBiZWdpbnNfd2l0aDogJ2JlZ2lucyB3aXRoJyxcbiAgICBleGFjdDogJ2lzIChleGFjdCknLFxuICAgIGluZGV4OiAnYnJvd3NlIHZhbHVlJyxcbiAgICBzdWJqZWN0X2xjOiAnTEMgYnJvd3NlJyxcbiAgICBzdWJqZWN0X2V4cGFuZGVkOiAnZXhwYW5kZWQgYnJvd3NlJyxcbn07XG5cbmV4cG9ydCBjb25zdCBBRFZBTkNFRF9TRUFSQ0hfRklFTERfTUFQID0ge1xuICAgIGFic3RyYWN0OiAnQWJzdHJhY3QnLFxuICAgIGFsbFRleHQ6ICdBbGwgVGV4dCcsXG4gICAgYW55OiAnQW55IEZpZWxkJyxcbiAgICBjYWxsTnVtYmVyOiAnQ2FsbCBOdW1iZXInLFxuICAgIGNvbnRyaWJ1dG9yOiAnQ29udHJpYnV0b3InLFxuICAgIGNyZWF0b3I6ICdBdXRob3IvQ3JlYXRvcicsXG4gICAgZ2VucmU6ICdHZW5yZScsXG4gICAgaXNibjogJ0lTQk4nLFxuICAgIGlzc246ICdJU1NOJyxcbiAgICBzZXJpZXM6ICdTZXJpZXMnLFxuICAgIHNvdXJjZTogJ1NvdXJjZScsXG4gICAgc3ViamVjdDogJ1N1YmplY3QnLFxuICAgIHRpdGxlOiAnVGl0bGUnLFxufTtcbmV4cG9ydCBjb25zdCBQUklNQVJZX0JMVUUgPSAncmdiYSg2MywgMTEyLCAxNzYsIDEpJztcbmV4cG9ydCBjb25zdCBQUklNQVJZX0JMVUVfSE9WRVIgPSAnaHNsKDIxNCwgNDglLCA2MCUsIDEpJztcbmV4cG9ydCBjb25zdCBQUklNQVJZX0xJR0hUX0JMVUUgPSAncmdiYSgyMjMsIDIzMywgMjQ3LCAxKSc7XG5leHBvcnQgY29uc3QgUFJJTUFSWV9QVVJQTEUgPSAnIzg3NzJBNic7XG5leHBvcnQgY29uc3QgUFJJTUFSWV9QVVJQTEVfSE9WRVIgPSAnIzdDNjU5Ric7XG5leHBvcnQgY29uc3QgREVTVFJVQ1RJVkVfUkVEID0gJ2hzbCg1LCA0NyUsIDQ3JSwgMSknO1xuZXhwb3J0IGNvbnN0IERFU1RSVUNUSVZFX1JFRF9IT1ZFUiA9ICdoc2woNSwgNDclLCA1MCUsIDEpJztcbmV4cG9ydCBjb25zdCBTRUNPTkRBUllfVEVYVF9HUkFZID0gJ3JnYmEoMTEyLCAxMTIsIDExMiwgMSknO1xuZXhwb3J0IGNvbnN0IExJR0hUX0dSQVkgPSAncmdiYSgyMzAsIDIzMCwgMjMwLCAxKSc7XG5leHBvcnQgY29uc3QgRFJPUF9TSEFET1dfREFSS19CTFVFID0gJ3JnYmEoNTEsIDYyLCA3NywgMC4xNiknO1xuZXhwb3J0IGNvbnN0IFRFWFRfU0VNSUJPTEQgPSAnNjAwJztcbmV4cG9ydCBjb25zdCBURVhUX0JPTEQgPSAnNzAwJztcbmV4cG9ydCBjb25zdCBBTklNQVRJT05fTEVOR1RIX1NURCA9ICcwLjE1cyc7XG5leHBvcnQgY29uc3QgTElORV9IRUlHSFRfU1REID0gJzEuNCc7XG5leHBvcnQgY29uc3QgQ09OVEVYVFVBTF9TRUxFQ1RPUl9IRUlHSFQgPSAnNC41cmVtJztcbmV4cG9ydCBjb25zdCBDT05URU5UX01BWF9XSURUSCA9ICc3NXJlbSc7XG5leHBvcnQgY29uc3QgQ09OVEVOVF9QQURESU5HID0gJzUuNHJlbSc7XG4iXX0=
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, inject, } from '@angular/core';
|
|
2
|
+
import { NonNullableFormBuilder, ReactiveFormsModule, Validators } from '@angular/forms';
|
|
3
|
+
import { CommonModule } from '@angular/common';
|
|
4
|
+
import { libHbllFadeInOut } from '../../animations/animations';
|
|
5
|
+
import { validateDateRangeFields } from '../utils';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/forms";
|
|
8
|
+
export class DateRangeComponent {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.fb = inject(NonNullableFormBuilder);
|
|
11
|
+
/**
|
|
12
|
+
* Emits `{ from: string, to: string }` object if both values are set to a valid or empty state by the user.
|
|
13
|
+
*/
|
|
14
|
+
this.validDateChange = new EventEmitter();
|
|
15
|
+
this.fromChange = new EventEmitter();
|
|
16
|
+
this.toChange = new EventEmitter();
|
|
17
|
+
this.dateForm = this.fb.group({
|
|
18
|
+
from: ['', [Validators.pattern('^[0-9]{4}$')]],
|
|
19
|
+
to: ['', [Validators.pattern('^[0-9]{4}$')]],
|
|
20
|
+
}, { validators: validateDateRangeFields() });
|
|
21
|
+
this.onFromChange = () => {
|
|
22
|
+
this.onInputChange();
|
|
23
|
+
this.fromChange.emit(this.dateForm.controls.from.value);
|
|
24
|
+
};
|
|
25
|
+
this.onToChange = () => {
|
|
26
|
+
this.onInputChange();
|
|
27
|
+
this.toChange.emit(this.dateForm.controls.to.value);
|
|
28
|
+
};
|
|
29
|
+
this.onInputChange = () => {
|
|
30
|
+
let from = this.dateForm.controls.from.value;
|
|
31
|
+
let to = this.dateForm.controls.to.value;
|
|
32
|
+
if (from && to && this.dateForm.controls.from.valid && this.dateForm.controls.to.valid) {
|
|
33
|
+
// Ensure 'to' contains the larger year. Swap if necessary.
|
|
34
|
+
if (parseInt(from) > parseInt(to)) {
|
|
35
|
+
const swap = from;
|
|
36
|
+
from = to;
|
|
37
|
+
to = swap;
|
|
38
|
+
}
|
|
39
|
+
this.validDateChange.emit({ from, to });
|
|
40
|
+
}
|
|
41
|
+
else if (!from && !to) {
|
|
42
|
+
this.validDateChange.emit({ from, to });
|
|
43
|
+
}
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
set from(from) {
|
|
47
|
+
if (from !== this.dateForm.controls.from.value)
|
|
48
|
+
this.dateForm.controls.from.setValue(from);
|
|
49
|
+
}
|
|
50
|
+
set to(to) {
|
|
51
|
+
if (to !== this.dateForm.controls.to.value)
|
|
52
|
+
this.dateForm.controls.to.setValue(to);
|
|
53
|
+
}
|
|
54
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: DateRangeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
55
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.1.0", type: DateRangeComponent, isStandalone: true, selector: "lib-date-range", inputs: { from: "from", to: "to" }, outputs: { validDateChange: "validDateChange", fromChange: "fromChange", toChange: "toChange" }, ngImport: i0, template: "<form id=\"from-to\" data-testid=\"from-to\" [formGroup]=\"dateForm\">\n <label for=\"fromFacet\">Between</label>\n <input\n formControlName=\"from\"\n id=\"fromFacet\"\n name=\"fromFacet\"\n data-testid=\"from\"\n placeholder=\"YYYY\"\n inputmode=\"numeric\"\n minlength=\"4\"\n maxlength=\"4\"\n (input)=\"onFromChange()\"\n [class.error]=\"\n dateForm.errors?.['fromInvalid'] || dateForm.controls.from.errors?.['pattern']\n \"\n />\n <label for=\"toFacet\">and</label>\n <input\n formControlName=\"to\"\n id=\"toFacet\"\n name=\"toFacet\"\n data-testid=\"to\"\n placeholder=\"YYYY\"\n inputmode=\"numeric\"\n minlength=\"4\"\n maxlength=\"4\"\n (input)=\"onToChange()\"\n [class.error]=\"dateForm.errors?.['toInvalid'] || dateForm.controls.to.errors?.['pattern']\"\n />\n</form>\n", styles: ["input{appearance:none;font-family:inherit;padding:.38em;border:solid 1px #707070;border-radius:4px;font-size:1em;background-color:#fff}input:focus{border-color:#3a6093}#from-to{margin-top:.6rem;margin-bottom:.8rem}#from-to label{margin-right:.7em}#from-to label:not(:first-of-type){margin-left:.7em}#from-to input{font-size:inherit;width:3.5em}#from-to input.error{border-color:#b04940}@media screen and (min-width: 690px){#from-to{margin-bottom:1.4rem}#from-to label{margin-right:.65em}#from-to label:not(:first-of-type){margin-left:.65em}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinLengthValidator, selector: "[minlength][formControlName],[minlength][formControl],[minlength][ngModel]", inputs: ["minlength"] }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }], animations: [libHbllFadeInOut], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
56
|
+
}
|
|
57
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: DateRangeComponent, decorators: [{
|
|
58
|
+
type: Component,
|
|
59
|
+
args: [{ selector: 'lib-date-range', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, ReactiveFormsModule], animations: [libHbllFadeInOut], template: "<form id=\"from-to\" data-testid=\"from-to\" [formGroup]=\"dateForm\">\n <label for=\"fromFacet\">Between</label>\n <input\n formControlName=\"from\"\n id=\"fromFacet\"\n name=\"fromFacet\"\n data-testid=\"from\"\n placeholder=\"YYYY\"\n inputmode=\"numeric\"\n minlength=\"4\"\n maxlength=\"4\"\n (input)=\"onFromChange()\"\n [class.error]=\"\n dateForm.errors?.['fromInvalid'] || dateForm.controls.from.errors?.['pattern']\n \"\n />\n <label for=\"toFacet\">and</label>\n <input\n formControlName=\"to\"\n id=\"toFacet\"\n name=\"toFacet\"\n data-testid=\"to\"\n placeholder=\"YYYY\"\n inputmode=\"numeric\"\n minlength=\"4\"\n maxlength=\"4\"\n (input)=\"onToChange()\"\n [class.error]=\"dateForm.errors?.['toInvalid'] || dateForm.controls.to.errors?.['pattern']\"\n />\n</form>\n", styles: ["input{appearance:none;font-family:inherit;padding:.38em;border:solid 1px #707070;border-radius:4px;font-size:1em;background-color:#fff}input:focus{border-color:#3a6093}#from-to{margin-top:.6rem;margin-bottom:.8rem}#from-to label{margin-right:.7em}#from-to label:not(:first-of-type){margin-left:.7em}#from-to input{font-size:inherit;width:3.5em}#from-to input.error{border-color:#b04940}@media screen and (min-width: 690px){#from-to{margin-bottom:1.4rem}#from-to label{margin-right:.65em}#from-to label:not(:first-of-type){margin-left:.65em}}\n"] }]
|
|
60
|
+
}], propDecorators: { validDateChange: [{
|
|
61
|
+
type: Output
|
|
62
|
+
}], fromChange: [{
|
|
63
|
+
type: Output
|
|
64
|
+
}], toChange: [{
|
|
65
|
+
type: Output
|
|
66
|
+
}], from: [{
|
|
67
|
+
type: Input
|
|
68
|
+
}], to: [{
|
|
69
|
+
type: Input
|
|
70
|
+
}] } });
|
|
71
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-range.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/ss-search-bar/date-range/date-range.component.ts","../../../../../../projects/components/src/lib/ss-search-bar/date-range/date-range.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    Component,\n    EventEmitter,\n    Input,\n    Output,\n    inject,\n} from '@angular/core';\nimport { NonNullableFormBuilder, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { CommonModule } from '@angular/common';\nimport { libHbllFadeInOut } from '../../animations/animations';\nimport { validateDateRangeFields } from '../utils';\n\n@Component({\n    selector: 'lib-date-range',\n    templateUrl: './date-range.component.html',\n    styleUrls: ['./date-range.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    standalone: true,\n    imports: [CommonModule, ReactiveFormsModule],\n    animations: [libHbllFadeInOut],\n})\nexport class DateRangeComponent {\n    private readonly fb = inject(NonNullableFormBuilder);\n    /**\n     * Emits `{ from: string, to: string }` object if both values are set to a valid or empty state by the user.\n     */\n    @Output() validDateChange = new EventEmitter<{ from: string; to: string }>();\n    @Output() fromChange = new EventEmitter<string>();\n    @Output() toChange = new EventEmitter<string>();\n    protected dateForm = this.fb.group(\n        {\n            from: ['', [Validators.pattern('^[0-9]{4}$')]],\n            to: ['', [Validators.pattern('^[0-9]{4}$')]],\n        },\n        { validators: validateDateRangeFields() },\n    );\n    @Input() set from(from: string) {\n        if (from !== this.dateForm.controls.from.value) this.dateForm.controls.from.setValue(from);\n    }\n    @Input() set to(to: string) {\n        if (to !== this.dateForm.controls.to.value) this.dateForm.controls.to.setValue(to);\n    }\n    protected onFromChange = () => {\n        this.onInputChange();\n        this.fromChange.emit(this.dateForm.controls.from.value);\n    };\n    protected onToChange = () => {\n        this.onInputChange();\n        this.toChange.emit(this.dateForm.controls.to.value);\n    };\n\n    private onInputChange = () => {\n        let from = this.dateForm.controls.from.value;\n        let to = this.dateForm.controls.to.value;\n        if (from && to && this.dateForm.controls.from.valid && this.dateForm.controls.to.valid) {\n            // Ensure 'to' contains the larger year. Swap if necessary.\n            if (parseInt(from) > parseInt(to)) {\n                const swap = from;\n                from = to;\n                to = swap;\n            }\n            this.validDateChange.emit({ from, to });\n        } else if (!from && !to) {\n            this.validDateChange.emit({ from, to });\n        }\n    };\n}\n","<form id=\"from-to\" data-testid=\"from-to\" [formGroup]=\"dateForm\">\n    <label for=\"fromFacet\">Between</label>\n    <input\n        formControlName=\"from\"\n        id=\"fromFacet\"\n        name=\"fromFacet\"\n        data-testid=\"from\"\n        placeholder=\"YYYY\"\n        inputmode=\"numeric\"\n        minlength=\"4\"\n        maxlength=\"4\"\n        (input)=\"onFromChange()\"\n        [class.error]=\"\n            dateForm.errors?.['fromInvalid'] || dateForm.controls.from.errors?.['pattern']\n        \"\n    />\n    <label for=\"toFacet\">and</label>\n    <input\n        formControlName=\"to\"\n        id=\"toFacet\"\n        name=\"toFacet\"\n        data-testid=\"to\"\n        placeholder=\"YYYY\"\n        inputmode=\"numeric\"\n        minlength=\"4\"\n        maxlength=\"4\"\n        (input)=\"onToChange()\"\n        [class.error]=\"dateForm.errors?.['toInvalid'] || dateForm.controls.to.errors?.['pattern']\"\n    />\n</form>\n"]}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { Pipe } from '@angular/core';
|
|
2
|
+
import { ADVANCED_SEARCH_OPTIONS, } from '../ss-search-bar.component';
|
|
3
|
+
import { ADVANCED_SEARCH_FIELD_MAP, ADVANCED_SEARCH_QUALIFIER_MAP } from '../constants';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export class SsAdvancedQueriesPipe {
|
|
6
|
+
transform(queryRows) {
|
|
7
|
+
return queryRows
|
|
8
|
+
.map((row, i) => {
|
|
9
|
+
const b = i !== 0 ? row.boolean || ADVANCED_SEARCH_OPTIONS.boolean[0] : '';
|
|
10
|
+
const f = row.field ??
|
|
11
|
+
ADVANCED_SEARCH_OPTIONS.localFields[0];
|
|
12
|
+
const o = row.qualifier ?? ADVANCED_SEARCH_OPTIONS.qualifiers[f][0];
|
|
13
|
+
return `${b ? b + ' ' : ''}${ADVANCED_SEARCH_FIELD_MAP[f]} ${ADVANCED_SEARCH_QUALIFIER_MAP[o]} <b>${row.query}</b>`;
|
|
14
|
+
})
|
|
15
|
+
.join(' ');
|
|
16
|
+
}
|
|
17
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: SsAdvancedQueriesPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
18
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.1.0", ngImport: i0, type: SsAdvancedQueriesPipe, isStandalone: true, name: "ssAdvancedQueries" }); }
|
|
19
|
+
}
|
|
20
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: SsAdvancedQueriesPipe, decorators: [{
|
|
21
|
+
type: Pipe,
|
|
22
|
+
args: [{
|
|
23
|
+
name: 'ssAdvancedQueries',
|
|
24
|
+
standalone: true,
|
|
25
|
+
}]
|
|
26
|
+
}] });
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWR2YW5jZWQtcXVlcmllcy5waXBlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50cy9zcmMvbGliL3NzLXNlYXJjaC1iYXIvcGlwZXMvYWR2YW5jZWQtcXVlcmllcy5waXBlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxJQUFJLEVBQWlCLE1BQU0sZUFBZSxDQUFDO0FBQ3BELE9BQU8sRUFDSCx1QkFBdUIsR0FHMUIsTUFBTSw0QkFBNEIsQ0FBQztBQUNwQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsNkJBQTZCLEVBQUUsTUFBTSxjQUFjLENBQUM7O0FBTXhGLE1BQU0sT0FBTyxxQkFBcUI7SUFDOUIsU0FBUyxDQUFDLFNBQW1DO1FBQ3pDLE9BQU8sU0FBUzthQUNYLEdBQUcsQ0FBQyxDQUFDLEdBQUcsRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUNaLE1BQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxPQUFPLElBQUksdUJBQXVCLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7WUFDM0UsTUFBTSxDQUFDLEdBQ0YsR0FBRyxDQUFDLEtBQXdDO2dCQUM1Qyx1QkFBdUIsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFvQyxDQUFDO1lBQy9FLE1BQU0sQ0FBQyxHQUFHLEdBQUcsQ0FBQyxTQUFTLElBQUksdUJBQXVCLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3BFLE9BQU8sR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsR0FDdEIseUJBQXlCLENBQUMsQ0FBQyxDQUMvQixJQUFJLDZCQUE2QixDQUFDLENBQStDLENBQUMsT0FBTyxHQUFHLENBQUMsS0FBSyxNQUFNLENBQUM7UUFDN0csQ0FBQyxDQUFDO2FBQ0QsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ25CLENBQUM7OEdBZFEscUJBQXFCOzRHQUFyQixxQkFBcUI7OzJGQUFyQixxQkFBcUI7a0JBSmpDLElBQUk7bUJBQUM7b0JBQ0YsSUFBSSxFQUFFLG1CQUFtQjtvQkFDekIsVUFBVSxFQUFFLElBQUk7aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGlwZSwgUGlwZVRyYW5zZm9ybSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtcbiAgICBBRFZBTkNFRF9TRUFSQ0hfT1BUSU9OUyxcbiAgICBBZHZhbmNlZFNlYXJjaExvY2FsRmllbGRPcHRpb24sXG4gICAgQWR2YW5jZWRTZWFyY2hRdWVyeVJvdyxcbn0gZnJvbSAnLi4vc3Mtc2VhcmNoLWJhci5jb21wb25lbnQnO1xuaW1wb3J0IHsgQURWQU5DRURfU0VBUkNIX0ZJRUxEX01BUCwgQURWQU5DRURfU0VBUkNIX1FVQUxJRklFUl9NQVAgfSBmcm9tICcuLi9jb25zdGFudHMnO1xuXG5AUGlwZSh7XG4gICAgbmFtZTogJ3NzQWR2YW5jZWRRdWVyaWVzJyxcbiAgICBzdGFuZGFsb25lOiB0cnVlLFxufSlcbmV4cG9ydCBjbGFzcyBTc0FkdmFuY2VkUXVlcmllc1BpcGUgaW1wbGVtZW50cyBQaXBlVHJhbnNmb3JtIHtcbiAgICB0cmFuc2Zvcm0ocXVlcnlSb3dzOiBBZHZhbmNlZFNlYXJjaFF1ZXJ5Um93W10pOiBzdHJpbmcge1xuICAgICAgICByZXR1cm4gcXVlcnlSb3dzXG4gICAgICAgICAgICAubWFwKChyb3csIGkpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCBiID0gaSAhPT0gMCA/IHJvdy5ib29sZWFuIHx8IEFEVkFOQ0VEX1NFQVJDSF9PUFRJT05TLmJvb2xlYW5bMF0gOiAnJztcbiAgICAgICAgICAgICAgICBjb25zdCBmOiBBZHZhbmNlZFNlYXJjaExvY2FsRmllbGRPcHRpb24gPVxuICAgICAgICAgICAgICAgICAgICAocm93LmZpZWxkIGFzIEFkdmFuY2VkU2VhcmNoTG9jYWxGaWVsZE9wdGlvbikgPz9cbiAgICAgICAgICAgICAgICAgICAgKEFEVkFOQ0VEX1NFQVJDSF9PUFRJT05TLmxvY2FsRmllbGRzWzBdIGFzIEFkdmFuY2VkU2VhcmNoTG9jYWxGaWVsZE9wdGlvbik7XG4gICAgICAgICAgICAgICAgY29uc3QgbyA9IHJvdy5xdWFsaWZpZXIgPz8gQURWQU5DRURfU0VBUkNIX09QVElPTlMucXVhbGlmaWVyc1tmXVswXTtcbiAgICAgICAgICAgICAgICByZXR1cm4gYCR7YiA/IGIgKyAnICcgOiAnJ30ke1xuICAgICAgICAgICAgICAgICAgICBBRFZBTkNFRF9TRUFSQ0hfRklFTERfTUFQW2ZdXG4gICAgICAgICAgICAgICAgfSAke0FEVkFOQ0VEX1NFQVJDSF9RVUFMSUZJRVJfTUFQW28gYXMga2V5b2YgdHlwZW9mIEFEVkFOQ0VEX1NFQVJDSF9RVUFMSUZJRVJfTUFQXX0gPGI+JHtyb3cucXVlcnl9PC9iPmA7XG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLmpvaW4oJyAnKTtcbiAgICB9XG59XG4iXX0=
|