@sebgroup/green-angular 4.6.1 → 4.6.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/src/v-angular/base-control-value-accessor/base-control-value-accessor.component.mjs +9 -9
- package/esm2022/src/v-angular/button/button.component.mjs +5 -5
- package/esm2022/src/v-angular/character-countdown/character-countdown.directive.mjs +5 -5
- package/esm2022/src/v-angular/core/core.globals.mjs +3 -3
- package/esm2022/src/v-angular/dropdown/dropdown-list/dropdown-list.component.mjs +4 -4
- package/esm2022/src/v-angular/dropdown/typeahead/index.mjs +4 -1
- package/esm2022/src/v-angular/dropdown/typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component.mjs +5 -5
- package/esm2022/src/v-angular/dropdown/typeahead/typeahead.directive.mjs +177 -0
- package/esm2022/src/v-angular/info-circle/info-circle.component.mjs +3 -3
- package/esm2022/src/v-angular/input/input.component.mjs +7 -7
- package/esm2022/src/v-angular/input-mask/input-mask-format.pipe.mjs +5 -4
- package/esm2022/src/v-angular/input-mask/input-mask.directive.mjs +8 -8
- package/esm2022/src/v-angular/modal/dialog/dialog.component.mjs +16 -16
- package/esm2022/src/v-angular/modal/fold-out/fold-out.directive.mjs +3 -3
- package/esm2022/src/v-angular/modal/modal.globals.mjs +3 -3
- package/esm2022/src/v-angular/modal/slide-out/slide-out.component.mjs +15 -15
- package/esm2022/src/v-angular/textarea/textarea.component.mjs +4 -3
- package/esm2022/src/v-angular/textarea/textarea.module.mjs +5 -4
- package/esm2022/src/v-angular/tooltip/tooltip.directive.mjs +13 -13
- package/esm2022/v-angular/base-control-value-accessor/base-control-value-accessor.component.mjs +287 -0
- package/esm2022/v-angular/base-control-value-accessor/base-control-value-accessor.module.mjs +17 -0
- package/esm2022/v-angular/base-control-value-accessor/index.mjs +3 -0
- package/esm2022/v-angular/breadcrumbs/breadcrumbs.component.mjs +25 -0
- package/esm2022/v-angular/breadcrumbs/breadcrumbs.module.mjs +20 -0
- package/esm2022/v-angular/breadcrumbs/index.mjs +3 -0
- package/esm2022/v-angular/button/button.component.mjs +108 -0
- package/esm2022/v-angular/button/button.module.mjs +20 -0
- package/esm2022/v-angular/button/index.mjs +3 -0
- package/esm2022/v-angular/card/card.component.mjs +11 -0
- package/esm2022/v-angular/card/card.module.mjs +18 -0
- package/esm2022/v-angular/card/index.mjs +3 -0
- package/esm2022/v-angular/character-countdown/character-countdown.directive.mjs +51 -0
- package/esm2022/v-angular/character-countdown/character-countdown.module.mjs +18 -0
- package/esm2022/v-angular/character-countdown/index.mjs +3 -0
- package/esm2022/v-angular/checkbox/checkbox.component.mjs +72 -0
- package/esm2022/v-angular/checkbox/checkbox.module.mjs +19 -0
- package/esm2022/v-angular/checkbox/index.mjs +3 -0
- package/esm2022/v-angular/dropdown/dropdown-list/dropdown-list.component.mjs +256 -0
- package/esm2022/v-angular/dropdown/dropdown-list/index.mjs +2 -0
- package/esm2022/v-angular/dropdown/dropdown.component.mjs +239 -0
- package/esm2022/v-angular/dropdown/dropdown.module.mjs +22 -0
- package/esm2022/v-angular/dropdown/index.mjs +6 -0
- package/esm2022/v-angular/dropdown/typeahead/index.mjs +6 -0
- package/esm2022/v-angular/dropdown/typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component.mjs +98 -0
- package/esm2022/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.mjs +85 -0
- package/esm2022/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.mjs +132 -0
- package/esm2022/v-angular/dropdown/typeahead/typeahead.directive.mjs +177 -0
- package/esm2022/v-angular/dropdown/typeahead/typeahead.module.mjs +33 -0
- package/esm2022/v-angular/i18n/i18n.json +12 -0
- package/esm2022/v-angular/i18n/i18n.module.mjs +83 -0
- package/esm2022/v-angular/i18n/i18n.test.module.mjs +89 -0
- package/esm2022/v-angular/i18n/index.mjs +3 -0
- package/esm2022/v-angular/index.mjs +18 -0
- package/esm2022/v-angular/info-circle/index.mjs +3 -0
- package/esm2022/v-angular/info-circle/info-circle.component.mjs +28 -0
- package/esm2022/v-angular/info-circle/info-circle.module.mjs +21 -0
- package/esm2022/v-angular/input/index.mjs +3 -0
- package/esm2022/v-angular/input/input.component.mjs +221 -0
- package/esm2022/v-angular/input/input.module.mjs +32 -0
- package/esm2022/v-angular/input-mask/config.mjs +9 -0
- package/esm2022/v-angular/input-mask/constants.mjs +2 -0
- package/esm2022/v-angular/input-mask/index.mjs +6 -0
- package/esm2022/v-angular/input-mask/input-mask-format.pipe.mjs +21 -0
- package/esm2022/v-angular/input-mask/input-mask.directive.mjs +165 -0
- package/esm2022/v-angular/input-mask/input-mask.module.mjs +35 -0
- package/esm2022/v-angular/input-mask/input-mask.types.mjs +2 -0
- package/esm2022/v-angular/modal/dialog/dialog.component.mjs +190 -0
- package/esm2022/v-angular/modal/fold-out/fold-out.component.mjs +56 -0
- package/esm2022/v-angular/modal/fold-out/fold-out.directive.mjs +19 -0
- package/esm2022/v-angular/modal/index.mjs +6 -0
- package/esm2022/v-angular/modal/modal.globals.mjs +20 -0
- package/esm2022/v-angular/modal/modal.module.mjs +40 -0
- package/esm2022/v-angular/modal/modal.types.mjs +2 -0
- package/esm2022/v-angular/modal/slide-out/slide-out.component.mjs +229 -0
- package/esm2022/v-angular/radio/index.mjs +3 -0
- package/esm2022/v-angular/radio/radio.component.mjs +130 -0
- package/esm2022/v-angular/radio/radio.module.mjs +20 -0
- package/esm2022/v-angular/sebgroup-green-angular-v-angular.mjs +5 -0
- package/esm2022/v-angular/textarea/index.mjs +3 -0
- package/esm2022/v-angular/textarea/textarea.component.mjs +102 -0
- package/esm2022/v-angular/textarea/textarea.module.mjs +19 -0
- package/esm2022/v-angular/tooltip/index.mjs +3 -0
- package/esm2022/v-angular/tooltip/tooltip.directive.mjs +273 -0
- package/esm2022/v-angular/tooltip/tooltip.module.mjs +18 -0
- package/esm2022/v-angular/v-angular.module.mjs +80 -0
- package/fesm2022/sebgroup-green-angular-src-v-angular-base-control-value-accessor.mjs +8 -8
- package/fesm2022/sebgroup-green-angular-src-v-angular-base-control-value-accessor.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-button.mjs +4 -4
- package/fesm2022/sebgroup-green-angular-src-v-angular-button.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-character-countdown.mjs +4 -4
- package/fesm2022/sebgroup-green-angular-src-v-angular-character-countdown.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-core.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-core.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-dropdown.mjs +179 -10
- package/fesm2022/sebgroup-green-angular-src-v-angular-dropdown.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-info-circle.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-info-circle.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-input-mask.mjs +11 -10
- package/fesm2022/sebgroup-green-angular-src-v-angular-input-mask.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-input.mjs +6 -6
- package/fesm2022/sebgroup-green-angular-src-v-angular-input.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-modal.mjs +33 -33
- package/fesm2022/sebgroup-green-angular-src-v-angular-modal.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-textarea.mjs +7 -5
- package/fesm2022/sebgroup-green-angular-src-v-angular-textarea.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-tooltip.mjs +12 -12
- package/fesm2022/sebgroup-green-angular-src-v-angular-tooltip.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-v-angular.mjs +3408 -0
- package/fesm2022/sebgroup-green-angular-v-angular.mjs.map +1 -0
- package/package.json +8 -2
- package/src/v-angular/base-control-value-accessor/base-control-value-accessor.component.d.ts +3 -3
- package/src/v-angular/button/button.component.d.ts +2 -2
- package/src/v-angular/character-countdown/character-countdown.directive.d.ts +1 -1
- package/src/v-angular/core/core.globals.d.ts +1 -1
- package/src/v-angular/dropdown/typeahead/index.d.ts +3 -0
- package/src/v-angular/dropdown/typeahead/typeahead.directive.d.ts +76 -0
- package/src/v-angular/input/input.component.d.ts +3 -3
- package/src/v-angular/input-mask/input-mask-format.pipe.d.ts +1 -1
- package/src/v-angular/input-mask/input-mask.directive.d.ts +2 -2
- package/src/v-angular/modal/dialog/dialog.component.d.ts +5 -5
- package/src/v-angular/modal/fold-out/fold-out.directive.d.ts +1 -1
- package/src/v-angular/modal/modal.globals.d.ts +1 -1
- package/src/v-angular/modal/slide-out/slide-out.component.d.ts +6 -6
- package/src/v-angular/textarea/textarea.module.d.ts +2 -1
- package/src/v-angular/tooltip/tooltip.directive.d.ts +4 -4
- package/v-angular/base-control-value-accessor/base-control-value-accessor.component.d.ts +124 -0
- package/v-angular/base-control-value-accessor/base-control-value-accessor.module.d.ts +7 -0
- package/v-angular/base-control-value-accessor/index.d.ts +2 -0
- package/v-angular/breadcrumbs/breadcrumbs.component.d.ts +18 -0
- package/v-angular/breadcrumbs/breadcrumbs.module.d.ts +10 -0
- package/v-angular/breadcrumbs/index.d.ts +2 -0
- package/v-angular/button/button.component.d.ts +62 -0
- package/v-angular/button/button.module.d.ts +10 -0
- package/v-angular/button/index.d.ts +2 -0
- package/v-angular/card/card.component.d.ts +5 -0
- package/v-angular/card/card.module.d.ts +8 -0
- package/v-angular/card/index.d.ts +2 -0
- package/v-angular/character-countdown/character-countdown.directive.d.ts +17 -0
- package/v-angular/character-countdown/character-countdown.module.d.ts +8 -0
- package/v-angular/character-countdown/index.d.ts +2 -0
- package/v-angular/checkbox/checkbox.component.d.ts +27 -0
- package/v-angular/checkbox/checkbox.module.d.ts +9 -0
- package/v-angular/checkbox/index.d.ts +2 -0
- package/v-angular/dropdown/dropdown-list/dropdown-list.component.d.ts +89 -0
- package/v-angular/dropdown/dropdown-list/index.d.ts +1 -0
- package/v-angular/dropdown/dropdown.component.d.ts +99 -0
- package/v-angular/dropdown/dropdown.module.d.ts +12 -0
- package/v-angular/dropdown/index.d.ts +5 -0
- package/v-angular/dropdown/typeahead/index.d.ts +5 -0
- package/v-angular/dropdown/typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component.d.ts +34 -0
- package/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.d.ts +34 -0
- package/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.d.ts +59 -0
- package/v-angular/dropdown/typeahead/typeahead.directive.d.ts +76 -0
- package/v-angular/dropdown/typeahead/typeahead.module.d.ts +11 -0
- package/v-angular/i18n/i18n.module.d.ts +15 -0
- package/v-angular/i18n/i18n.test.module.d.ts +27 -0
- package/v-angular/i18n/index.d.ts +2 -0
- package/v-angular/index.d.ts +17 -0
- package/v-angular/info-circle/index.d.ts +2 -0
- package/v-angular/info-circle/info-circle.component.d.ts +16 -0
- package/v-angular/info-circle/info-circle.module.d.ts +10 -0
- package/v-angular/input/index.d.ts +2 -0
- package/v-angular/input/input.component.d.ts +90 -0
- package/v-angular/input/input.module.d.ts +11 -0
- package/v-angular/input-mask/config.d.ts +6 -0
- package/v-angular/input-mask/constants.d.ts +2 -0
- package/v-angular/input-mask/index.d.ts +5 -0
- package/v-angular/input-mask/input-mask-format.pipe.d.ts +8 -0
- package/v-angular/input-mask/input-mask.directive.d.ts +39 -0
- package/v-angular/input-mask/input-mask.module.d.ts +11 -0
- package/v-angular/input-mask/input-mask.types.d.ts +20 -0
- package/v-angular/modal/dialog/dialog.component.d.ts +55 -0
- package/v-angular/modal/fold-out/fold-out.component.d.ts +24 -0
- package/v-angular/modal/fold-out/fold-out.directive.d.ts +6 -0
- package/v-angular/modal/index.d.ts +5 -0
- package/v-angular/modal/modal.globals.d.ts +13 -0
- package/v-angular/modal/modal.module.d.ts +13 -0
- package/v-angular/modal/modal.types.d.ts +5 -0
- package/v-angular/modal/slide-out/slide-out.component.d.ts +76 -0
- package/v-angular/radio/index.d.ts +2 -0
- package/v-angular/radio/radio.component.d.ts +48 -0
- package/v-angular/radio/radio.module.d.ts +9 -0
- package/v-angular/textarea/index.d.ts +2 -0
- package/v-angular/textarea/textarea.component.d.ts +44 -0
- package/v-angular/textarea/textarea.module.d.ts +9 -0
- package/v-angular/tooltip/index.d.ts +2 -0
- package/v-angular/tooltip/tooltip.directive.d.ts +106 -0
- package/v-angular/tooltip/tooltip.module.d.ts +8 -0
- package/v-angular/v-angular.module.d.ts +23 -0
package/esm2022/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.mjs
ADDED
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/common";
|
|
4
|
+
export class NgvTypeaheadHighlightComponent {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.text = '';
|
|
7
|
+
this.input = '';
|
|
8
|
+
/**
|
|
9
|
+
* Regexp of characters that are allowed in textContent without being found in textToHighlight
|
|
10
|
+
* Allow whitespaces.
|
|
11
|
+
* */
|
|
12
|
+
this.allowedNonMatchingChars = new RegExp(/\s/);
|
|
13
|
+
}
|
|
14
|
+
ngOnChanges(changes) {
|
|
15
|
+
const { textContent, textToHighlight } = changes;
|
|
16
|
+
if (textContent?.currentValue != null)
|
|
17
|
+
this.text = `${textContent.currentValue}`;
|
|
18
|
+
if (textToHighlight?.currentValue != null)
|
|
19
|
+
this.input = `${textToHighlight.currentValue}`;
|
|
20
|
+
this.updateValues();
|
|
21
|
+
}
|
|
22
|
+
updateValues() {
|
|
23
|
+
const splittedText = this.text.toLocaleLowerCase().split('');
|
|
24
|
+
const splittedInput = this.input.toLocaleLowerCase().split('');
|
|
25
|
+
const { start, end } = this.getHighlightedPart(splittedText, splittedInput);
|
|
26
|
+
// If user input is not found within the textcontent, reset options and return.
|
|
27
|
+
if (start === -1 || end === -1) {
|
|
28
|
+
this.prefix = this.match = this.suffix = '';
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
this.prefix = this.text.substring(0, start);
|
|
32
|
+
this.match = this.text.substring(start, end);
|
|
33
|
+
this.suffix = this.text.substring(end);
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Function that finds the start and end index of where the input is located within the text to display.
|
|
37
|
+
* First occurence is returned. Allows for spaces to occur despite input not matching space.
|
|
38
|
+
* Loops through each character in splittedText and when a char in splittedText equlas the first character of
|
|
39
|
+
* splittedInput, evaluate wether it's match on the entire input.
|
|
40
|
+
* - If it's => return indexes.
|
|
41
|
+
* - If it's not => break out and check next char in outer loop.
|
|
42
|
+
* @param splittedText the text content splitted in an array
|
|
43
|
+
* @param splittedInput the input splitted in an array
|
|
44
|
+
* @returns { start: number, end: number } Indexes of where the match starts and ends in the text displatyed
|
|
45
|
+
*/
|
|
46
|
+
getHighlightedPart(splittedText, splittedInput) {
|
|
47
|
+
let start = -1;
|
|
48
|
+
let end = -1;
|
|
49
|
+
for (let i = 0; i < splittedText.length; i++) {
|
|
50
|
+
// If start and end have been set, break and return
|
|
51
|
+
if (start > -1 || end > -1)
|
|
52
|
+
break;
|
|
53
|
+
// If current char match first in input, try see if whole input match from that index
|
|
54
|
+
if (splittedText[i] === splittedInput[0]) {
|
|
55
|
+
let matches = 1;
|
|
56
|
+
for (let t = 1; t <= splittedText.length; t++) {
|
|
57
|
+
// If match on last character of input, we're finished.
|
|
58
|
+
if (matches === splittedInput.length) {
|
|
59
|
+
start = i;
|
|
60
|
+
end = start + t;
|
|
61
|
+
break;
|
|
62
|
+
}
|
|
63
|
+
if (splittedText[t + i] === splittedInput[matches])
|
|
64
|
+
matches++;
|
|
65
|
+
else if (this.allowedNonMatchingChars.test(splittedText[t + i]))
|
|
66
|
+
continue; // Ignore whitespace, continue
|
|
67
|
+
else
|
|
68
|
+
break; // No match, break out of inner loop to check next char in text
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
return { start, end };
|
|
73
|
+
}
|
|
74
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadHighlightComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
75
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: NgvTypeaheadHighlightComponent, selector: "nggv-typeahead-highlight", inputs: { textContent: "textContent", textToHighlight: "textToHighlight" }, usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"!this.match\">{{ textContent }}</ng-container>\n<!-- Note that this is sensitive to line breaks producing extra spacing between the different parts -->\n<ng-container *ngIf=\"!!this.match\">\n {{ prefix || '' }}<span class=\"match\">{{ match }}</span>{{ suffix ||''}}\n</ng-container>", styles: [":host .match{font-weight:500}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] }); }
|
|
76
|
+
}
|
|
77
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadHighlightComponent, decorators: [{
|
|
78
|
+
type: Component,
|
|
79
|
+
args: [{ selector: 'nggv-typeahead-highlight', template: "<ng-container *ngIf=\"!this.match\">{{ textContent }}</ng-container>\n<!-- Note that this is sensitive to line breaks producing extra spacing between the different parts -->\n<ng-container *ngIf=\"!!this.match\">\n {{ prefix || '' }}<span class=\"match\">{{ match }}</span>{{ suffix ||''}}\n</ng-container>", styles: [":host .match{font-weight:500}\n"] }]
|
|
80
|
+
}], propDecorators: { textContent: [{
|
|
81
|
+
type: Input
|
|
82
|
+
}], textToHighlight: [{
|
|
83
|
+
type: Input
|
|
84
|
+
}] } });
|
|
85
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typeahead-highlight.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular/src/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.ts","../../../../../../../../libs/angular/src/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core'\n\n@Component({\n  selector: 'nggv-typeahead-highlight',\n  templateUrl: './typeahead-highlight.component.html',\n  styleUrls: ['./typeahead-highlight.component.scss'],\n})\nexport class NgvTypeaheadHighlightComponent implements OnChanges {\n  /** The text that is displayed in the dropdown list */\n  @Input() textContent?: string\n  /** The substring that should be highlighted within textContent */\n  @Input() textToHighlight?: string\n\n  prefix?: string // Substring before the match\n  match?: string // The match\n  suffix?: string // Substring after the match\n\n  text = ''\n  input = ''\n\n  /**\n   * Regexp of characters that are allowed in textContent without being found in textToHighlight\n   * Allow whitespaces.\n   * */\n  allowedNonMatchingChars = new RegExp(/\\s/)\n\n  ngOnChanges(changes: SimpleChanges) {\n    const { textContent, textToHighlight } = changes\n    if (textContent?.currentValue != null)\n      this.text = `${textContent.currentValue}`\n    if (textToHighlight?.currentValue != null)\n      this.input = `${textToHighlight.currentValue}`\n    this.updateValues()\n  }\n\n  private updateValues() {\n    const splittedText = this.text.toLocaleLowerCase().split('')\n    const splittedInput = this.input.toLocaleLowerCase().split('')\n\n    const { start, end } = this.getHighlightedPart(splittedText, splittedInput)\n\n    // If user input is not found within the textcontent, reset options and return.\n    if (start === -1 || end === -1) {\n      this.prefix = this.match = this.suffix = ''\n      return\n    }\n    this.prefix = this.text.substring(0, start)\n    this.match = this.text.substring(start, end)\n    this.suffix = this.text.substring(end)\n  }\n\n  /**\n   * Function that finds the start and end index of where the input is located within the text to display.\n   * First occurence is returned. Allows for spaces to occur despite input not matching space.\n   * Loops through each character in splittedText and when a char in splittedText equlas the first character of\n   * splittedInput, evaluate wether it's match on the entire input.\n   * - If it's => return indexes.\n   * - If it's not => break out and check next char in outer loop.\n   * @param splittedText the text content splitted in an array\n   * @param splittedInput the input splitted in an array\n   * @returns { start: number, end: number } Indexes of where the match starts and ends in the text displatyed\n   */\n  private getHighlightedPart(\n    splittedText: string[],\n    splittedInput: string[],\n  ): { start: number; end: number } {\n    let start = -1\n    let end = -1\n\n    for (let i = 0; i < splittedText.length; i++) {\n      // If start and end have been set, break and return\n      if (start > -1 || end > -1) break\n\n      // If current char match first in input, try see if whole input match from that index\n      if (splittedText[i] === splittedInput[0]) {\n        let matches = 1\n        for (let t = 1; t <= splittedText.length; t++) {\n          // If match on last character of input, we're finished.\n          if (matches === splittedInput.length) {\n            start = i\n            end = start + t\n            break\n          }\n          if (splittedText[t + i] === splittedInput[matches]) matches++\n          else if (this.allowedNonMatchingChars.test(splittedText[t + i]))\n            continue // Ignore whitespace, continue\n          else break // No match, break out of inner loop to check next char in text\n        }\n      }\n    }\n    return { start, end }\n  }\n}\n","<ng-container *ngIf=\"!this.match\">{{ textContent }}</ng-container>\n<!-- Note that this is sensitive to line breaks producing extra spacing between the different parts -->\n<ng-container *ngIf=\"!!this.match\">\n    {{ prefix || '' }}<span class=\"match\">{{ match }}</span>{{ suffix ||''}}\n</ng-container>"]}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
import { ChangeDetectorRef, Component, ElementRef, HostListener, Inject, Input, Optional, Renderer2, Self, } from '@angular/core';
|
|
2
|
+
import { NgControl } from '@angular/forms';
|
|
3
|
+
import { takeUntil } from 'rxjs';
|
|
4
|
+
import { TRANSLOCO_SCOPE } from '@ngneat/transloco';
|
|
5
|
+
import { NgvDropdownComponent } from '../../dropdown.component';
|
|
6
|
+
import { NgvInputComponent } from '@sebgroup/green-angular/src/v-angular/input';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "@angular/forms";
|
|
9
|
+
import * as i2 from "@angular/common";
|
|
10
|
+
export class NgvTypeaheadInputComponent extends NgvInputComponent {
|
|
11
|
+
get dropdownButton() {
|
|
12
|
+
return this.hostComponent.inputRef?.nativeElement;
|
|
13
|
+
}
|
|
14
|
+
constructor(element, renderer2, ngControl, translocoScope, cdr) {
|
|
15
|
+
super(ngControl, translocoScope, cdr);
|
|
16
|
+
this.element = element;
|
|
17
|
+
this.renderer2 = renderer2;
|
|
18
|
+
this.ngControl = ngControl;
|
|
19
|
+
this.translocoScope = translocoScope;
|
|
20
|
+
this.cdr = cdr;
|
|
21
|
+
/** Boolean to indicate wether the dropdown is expanded or not, to apply appropriate styling */
|
|
22
|
+
this.expanded = false;
|
|
23
|
+
super.ngOnInit();
|
|
24
|
+
}
|
|
25
|
+
ngOnInit() {
|
|
26
|
+
this.autocomplete = 'off';
|
|
27
|
+
this.debounceTime = 0;
|
|
28
|
+
this.hostComponent.selectOnSingleOption = false;
|
|
29
|
+
this.moveInput();
|
|
30
|
+
this.handleExpandedChange();
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* @internal
|
|
34
|
+
* Allow space to be inputted as text
|
|
35
|
+
* @param event fired containing which key was released.
|
|
36
|
+
*/
|
|
37
|
+
onKeyUp(event) {
|
|
38
|
+
if (event.code === 'Space') {
|
|
39
|
+
event.preventDefault();
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* @internal
|
|
44
|
+
* First time the dropdown is expanded, move the input sp it becomes a child of the dropdown button
|
|
45
|
+
* to better reflect semantics and styling.
|
|
46
|
+
*/
|
|
47
|
+
moveInput() {
|
|
48
|
+
// Workaround to execute code in setTimeout due to hooks management etc.
|
|
49
|
+
setTimeout(() => {
|
|
50
|
+
// Only move if parent dropdown is found
|
|
51
|
+
if (this.dropdownButton) {
|
|
52
|
+
this.renderer2.appendChild(this.dropdownButton.querySelector('button'), this.element.nativeElement);
|
|
53
|
+
// Get the height of the parent button so the input can be explicitly set to the same height since it's absolutely positioned
|
|
54
|
+
this.buttonHeight =
|
|
55
|
+
this.dropdownButton.getBoundingClientRect().height || 32; // Default to 2em;
|
|
56
|
+
}
|
|
57
|
+
}, 0);
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* @internal
|
|
61
|
+
* When dropdown is expanded, focus on the input. If a value is selected, set it in the input.
|
|
62
|
+
* When the dropdown is collapsed, empty the input from text.
|
|
63
|
+
*/
|
|
64
|
+
handleExpandedChange() {
|
|
65
|
+
this.hostComponent.expandedChange
|
|
66
|
+
.pipe(takeUntil(this._destroy$))
|
|
67
|
+
.subscribe((state) => {
|
|
68
|
+
this.expanded = state;
|
|
69
|
+
if (this.expanded) {
|
|
70
|
+
// Weird workaround for setting focus. Didn't set focus, but wrapping in setTimeout solved it.
|
|
71
|
+
// See suggestion here: https://github.com/ionic-team/stencil/issues/3772#issuecomment-1292599609
|
|
72
|
+
setTimeout(() => this.setFocus());
|
|
73
|
+
// Format and interpolate result since return type can be other than string from the formatter
|
|
74
|
+
const formattedValue = `${this.formatSelected(this.hostComponent.state)}`;
|
|
75
|
+
this.setInput(formattedValue, false);
|
|
76
|
+
}
|
|
77
|
+
else
|
|
78
|
+
this.setInput('', true);
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* @internal Formats the selected option to display in the input. Interpolate the returned value
|
|
83
|
+
* since we don't know the return type or label type.
|
|
84
|
+
* @param value The selected value
|
|
85
|
+
* @returns The formatted value
|
|
86
|
+
*/
|
|
87
|
+
formatSelected(value) {
|
|
88
|
+
if (value?.key == null)
|
|
89
|
+
return '';
|
|
90
|
+
// If no formatter exists, return the label or empty string
|
|
91
|
+
if (!this.selectedFormatter)
|
|
92
|
+
return value.label ?? '';
|
|
93
|
+
// If a formatter exists, use it
|
|
94
|
+
return this.selectedFormatter(value) ?? '';
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Sets the input programmatically instead of native HTML input event.
|
|
98
|
+
* @param input
|
|
99
|
+
*/
|
|
100
|
+
setInput(input, triggerFilter) {
|
|
101
|
+
this.state = input;
|
|
102
|
+
if (triggerFilter) {
|
|
103
|
+
this.onChange(this.state);
|
|
104
|
+
this.inputChange$.next(this.state);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadInputComponent, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i1.NgControl, optional: true, self: true }, { token: TRANSLOCO_SCOPE, optional: true }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
108
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: NgvTypeaheadInputComponent, selector: "nggv-typeahead-input", inputs: { hostComponent: "hostComponent", resultFormatter: "resultFormatter", selectedFormatter: "selectedFormatter" }, host: { listeners: { "document:keyup": "onKeyUp($event)" } }, usesInheritance: true, ngImport: i0, template: "<ng-container>\n <div class=\"input-wrapper\"\n [ngClass]=\"{ 'expanded': expanded, 'collapsed': !expanded }\"\n [ngStyle]=\"{ 'height.px': buttonHeight }\"\n (click)=\"$event.stopPropagation();\">\n <!-- INPUT FIELD -->\n <div class=\"input-group\">\n <input #input\n [id]=\"id + '-input'\"\n class=\"sdv-field\"\n [attr.type]=\"type\"\n [attr.name]=\"name\"\n [attr.required]=\"required\"\n [attr.email]=\"email\"\n [min]=\"min\"\n [max]=\"max\"\n [step]=\"step\"\n [attr.maxlength]=\"maxlength\"\n [attr.minlength]=\"minlength\"\n [pattern]=\"pattern\"\n [disabled]=\"disabled\"\n [autocomplete]=\"autocomplete\"\n [autofocus]=\"autofocus\"\n [readOnly]=\"readonly\"\n [attr.role]=\"role\"\n [attr.placeholder]=\"placeholder\"\n [attr.aria-label]=\"description\"\n [value]=\"state\"\n (input)=\"onInput($event)\"\n (focus)=\"onFocus($event)\"\n (blur)=\"onBlur($event)\" />\n </div>\n </div>\n</ng-container>", styles: [":host{position:absolute;top:0;left:0;width:calc(100% - 2.625em)}:host .input-wrapper{border:none;box-shadow:none}:host .input-wrapper.expanded{display:flex}:host .input-wrapper.expanded .input-group input.sdv-field{height:100%;width:100%;min-height:unset;margin-left:1px;border:none;box-shadow:none}:host .input-wrapper.collapsed{display:none}:host .input-wrapper .input-group{width:100%}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); }
|
|
109
|
+
}
|
|
110
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadInputComponent, decorators: [{
|
|
111
|
+
type: Component,
|
|
112
|
+
args: [{ selector: 'nggv-typeahead-input', template: "<ng-container>\n <div class=\"input-wrapper\"\n [ngClass]=\"{ 'expanded': expanded, 'collapsed': !expanded }\"\n [ngStyle]=\"{ 'height.px': buttonHeight }\"\n (click)=\"$event.stopPropagation();\">\n <!-- INPUT FIELD -->\n <div class=\"input-group\">\n <input #input\n [id]=\"id + '-input'\"\n class=\"sdv-field\"\n [attr.type]=\"type\"\n [attr.name]=\"name\"\n [attr.required]=\"required\"\n [attr.email]=\"email\"\n [min]=\"min\"\n [max]=\"max\"\n [step]=\"step\"\n [attr.maxlength]=\"maxlength\"\n [attr.minlength]=\"minlength\"\n [pattern]=\"pattern\"\n [disabled]=\"disabled\"\n [autocomplete]=\"autocomplete\"\n [autofocus]=\"autofocus\"\n [readOnly]=\"readonly\"\n [attr.role]=\"role\"\n [attr.placeholder]=\"placeholder\"\n [attr.aria-label]=\"description\"\n [value]=\"state\"\n (input)=\"onInput($event)\"\n (focus)=\"onFocus($event)\"\n (blur)=\"onBlur($event)\" />\n </div>\n </div>\n</ng-container>", styles: [":host{position:absolute;top:0;left:0;width:calc(100% - 2.625em)}:host .input-wrapper{border:none;box-shadow:none}:host .input-wrapper.expanded{display:flex}:host .input-wrapper.expanded .input-group input.sdv-field{height:100%;width:100%;min-height:unset;margin-left:1px;border:none;box-shadow:none}:host .input-wrapper.collapsed{display:none}:host .input-wrapper .input-group{width:100%}\n"] }]
|
|
113
|
+
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i1.NgControl, decorators: [{
|
|
114
|
+
type: Self
|
|
115
|
+
}, {
|
|
116
|
+
type: Optional
|
|
117
|
+
}] }, { type: undefined, decorators: [{
|
|
118
|
+
type: Optional
|
|
119
|
+
}, {
|
|
120
|
+
type: Inject,
|
|
121
|
+
args: [TRANSLOCO_SCOPE]
|
|
122
|
+
}] }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { hostComponent: [{
|
|
123
|
+
type: Input
|
|
124
|
+
}], resultFormatter: [{
|
|
125
|
+
type: Input
|
|
126
|
+
}], selectedFormatter: [{
|
|
127
|
+
type: Input
|
|
128
|
+
}], onKeyUp: [{
|
|
129
|
+
type: HostListener,
|
|
130
|
+
args: ['document:keyup', ['$event']]
|
|
131
|
+
}] } });
|
|
132
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typeahead-input.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular/src/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.ts","../../../../../../../../libs/angular/src/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,iBAAiB,EACjB,SAAS,EACT,UAAU,EACV,YAAY,EACZ,MAAM,EACN,KAAK,EAGL,QAAQ,EACR,SAAS,EACT,IAAI,GACL,MAAM,eAAe,CAAA;AACtB,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAA;AAE1C,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAA;AAEhC,OAAO,EAAE,eAAe,EAAkB,MAAM,mBAAmB,CAAA;AAGnE,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAA;AAC/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,6CAA6C,CAAA;;;;AAO/E,MAAM,OAAO,0BACX,SAAQ,iBAAiB;IAYzB,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,aAAa,CAAC,QAAQ,EAAE,aAAa,CAAA;IACnD,CAAC;IAQD,YACU,OAAmB,EACnB,SAAoB,EACD,SAAoB,EAGrC,cAA8B,EAC9B,GAAsB;QAEhC,KAAK,CAAC,SAAS,EAAE,cAAc,EAAE,GAAG,CAAC,CAAA;QAR7B,YAAO,GAAP,OAAO,CAAY;QACnB,cAAS,GAAT,SAAS,CAAW;QACD,cAAS,GAAT,SAAS,CAAW;QAGrC,mBAAc,GAAd,cAAc,CAAgB;QAC9B,QAAG,GAAH,GAAG,CAAmB;QAblC,+FAA+F;QAC/F,aAAQ,GAAG,KAAK,CAAA;QAed,KAAK,CAAC,QAAQ,EAAE,CAAA;IAClB,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,YAAY,GAAG,KAAK,CAAA;QACzB,IAAI,CAAC,YAAY,GAAG,CAAC,CAAA;QACrB,IAAI,CAAC,aAAa,CAAC,oBAAoB,GAAG,KAAK,CAAA;QAE/C,IAAI,CAAC,SAAS,EAAE,CAAA;QAChB,IAAI,CAAC,oBAAoB,EAAE,CAAA;IAC7B,CAAC;IAED;;;;OAIG;IAEH,OAAO,CAAC,KAAoB;QAC1B,IAAI,KAAK,CAAC,IAAI,KAAK,OAAO,EAAE;YAC1B,KAAK,CAAC,cAAc,EAAE,CAAA;SACvB;IACH,CAAC;IAED;;;;OAIG;IACK,SAAS;QACf,wEAAwE;QACxE,UAAU,CAAC,GAAG,EAAE;YACd,wCAAwC;YACxC,IAAI,IAAI,CAAC,cAAc,EAAE;gBACvB,IAAI,CAAC,SAAS,CAAC,WAAW,CACxB,IAAI,CAAC,cAAc,CAAC,aAAa,CAAC,QAAQ,CAAC,EAC3C,IAAI,CAAC,OAAO,CAAC,aAAa,CAC3B,CAAA;gBACD,6HAA6H;gBAC7H,IAAI,CAAC,YAAY;oBACf,IAAI,CAAC,cAAc,CAAC,qBAAqB,EAAE,CAAC,MAAM,IAAI,EAAE,CAAA,CAAC,kBAAkB;aAC9E;QACH,CAAC,EAAE,CAAC,CAAC,CAAA;IACP,CAAC;IAED;;;;OAIG;IACK,oBAAoB;QAC1B,IAAI,CAAC,aAAa,CAAC,cAAc;aAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;aAC/B,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAA;YAErB,IAAI,IAAI,CAAC,QAAQ,EAAE;gBACjB,8FAA8F;gBAC9F,iGAAiG;gBACjG,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAA;gBACjC,8FAA8F;gBAC9F,MAAM,cAAc,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,EAAE,CAAA;gBACzE,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE,KAAK,CAAC,CAAA;aACrC;;gBAAM,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,IAAI,CAAC,CAAA;QAChC,CAAC,CAAC,CAAA;IACN,CAAC;IAED;;;;;OAKG;IACK,cAAc,CAAC,KAAsB;QAC3C,IAAI,KAAK,EAAE,GAAG,IAAI,IAAI;YAAE,OAAO,EAAE,CAAA;QACjC,2DAA2D;QAC3D,IAAI,CAAC,IAAI,CAAC,iBAAiB;YAAE,OAAO,KAAK,CAAC,KAAK,IAAI,EAAE,CAAA;QACrD,gCAAgC;QAChC,OAAO,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,IAAI,EAAE,CAAA;IAC5C,CAAC;IAED;;;OAGG;IACK,QAAQ,CAAC,KAAa,EAAE,aAAsB;QACpD,IAAI,CAAC,KAAK,GAAG,KAAK,CAAA;QAClB,IAAI,aAAa,EAAE;YACjB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YACzB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;SACnC;IACH,CAAC;+GA5HU,0BAA0B,0HA4B3B,eAAe;mGA5Bd,0BAA0B,yQC5BvC,qyCAiCe;;4FDLF,0BAA0B;kBALtC,SAAS;+BACE,sBAAsB;;0BA8B7B,IAAI;;0BAAI,QAAQ;;0BAChB,QAAQ;;0BACR,MAAM;2BAAC,eAAe;4EAvBhB,aAAa;sBAArB,KAAK;gBAGG,eAAe;sBAAvB,KAAK;gBAGG,iBAAiB;sBAAzB,KAAK;gBAwCN,OAAO;sBADN,YAAY;uBAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  HostListener,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Renderer2,\n  Self,\n} from '@angular/core'\nimport { NgControl } from '@angular/forms'\n\nimport { takeUntil } from 'rxjs'\n\nimport { TRANSLOCO_SCOPE, TranslocoScope } from '@ngneat/transloco'\n\nimport { OptionBase } from '@sebgroup/green-angular/src/v-angular/core'\nimport { NgvDropdownComponent } from '../../dropdown.component'\nimport { NgvInputComponent } from '@sebgroup/green-angular/src/v-angular/input'\n\n@Component({\n  selector: 'nggv-typeahead-input',\n  templateUrl: './typeahead-input.component.html',\n  styleUrls: ['./typeahead-input.component.scss'],\n})\nexport class NgvTypeaheadInputComponent\n  extends NgvInputComponent\n  implements OnInit, OnDestroy\n{\n  /** Reference to the host dropdown */\n  @Input() hostComponent!: NgvDropdownComponent\n\n  /** Formats each item that is displayed as an option. Only applies format if the option if it implement Option interface. */\n  @Input() resultFormatter?: (option: OptionBase<any>) => string\n\n  /** Formats the selected item in the input when dropdown is opened. If no function is provided, it will display the value of the selected objects label property */\n  @Input() selectedFormatter?: (selected: OptionBase<any>) => string\n\n  get dropdownButton(): HTMLElement {\n    return this.hostComponent.inputRef?.nativeElement\n  }\n\n  /** Boolean to indicate wether the dropdown is expanded or not, to apply appropriate styling */\n  expanded = false\n\n  /** Used to determine the height of the inputin html */\n  buttonHeight?: number\n\n  constructor(\n    private element: ElementRef,\n    private renderer2: Renderer2,\n    @Self() @Optional() public ngControl: NgControl,\n    @Optional()\n    @Inject(TRANSLOCO_SCOPE)\n    protected translocoScope: TranslocoScope,\n    protected cdr: ChangeDetectorRef,\n  ) {\n    super(ngControl, translocoScope, cdr)\n    super.ngOnInit()\n  }\n\n  ngOnInit() {\n    this.autocomplete = 'off'\n    this.debounceTime = 0\n    this.hostComponent.selectOnSingleOption = false\n\n    this.moveInput()\n    this.handleExpandedChange()\n  }\n\n  /**\n   * @internal\n   * Allow space to be inputted as text\n   * @param event fired containing which key was released.\n   */\n  @HostListener('document:keyup', ['$event'])\n  onKeyUp(event: KeyboardEvent) {\n    if (event.code === 'Space') {\n      event.preventDefault()\n    }\n  }\n\n  /**\n   * @internal\n   * First time the dropdown is expanded, move the input sp it becomes a child of the dropdown button\n   * to better reflect semantics and styling.\n   */\n  private moveInput() {\n    // Workaround to execute code in setTimeout due to hooks management etc.\n    setTimeout(() => {\n      // Only move if parent dropdown is found\n      if (this.dropdownButton) {\n        this.renderer2.appendChild(\n          this.dropdownButton.querySelector('button'),\n          this.element.nativeElement,\n        )\n        // Get the height of the parent button so the input can be explicitly set to the same height since it's absolutely positioned\n        this.buttonHeight =\n          this.dropdownButton.getBoundingClientRect().height || 32 // Default to 2em;\n      }\n    }, 0)\n  }\n\n  /**\n   * @internal\n   * When dropdown is expanded, focus on the input. If a value is selected, set it in the input.\n   * When the dropdown is collapsed, empty the input from text.\n   */\n  private handleExpandedChange() {\n    this.hostComponent.expandedChange\n      .pipe(takeUntil(this._destroy$))\n      .subscribe((state) => {\n        this.expanded = state\n\n        if (this.expanded) {\n          // Weird workaround for setting focus. Didn't set focus, but wrapping in setTimeout solved it.\n          // See suggestion here: https://github.com/ionic-team/stencil/issues/3772#issuecomment-1292599609\n          setTimeout(() => this.setFocus())\n          // Format and interpolate result since return type can be other than string from the formatter\n          const formattedValue = `${this.formatSelected(this.hostComponent.state)}`\n          this.setInput(formattedValue, false)\n        } else this.setInput('', true)\n      })\n  }\n\n  /**\n   * @internal Formats the selected option to display in the input. Interpolate the returned value\n   * since we don't know the return type or label type.\n   * @param value The selected value\n   * @returns The formatted value\n   */\n  private formatSelected(value: OptionBase<any>) {\n    if (value?.key == null) return ''\n    // If no formatter exists, return the label or empty string\n    if (!this.selectedFormatter) return value.label ?? ''\n    // If a formatter exists, use it\n    return this.selectedFormatter(value) ?? ''\n  }\n\n  /**\n   * Sets the input programmatically instead of native HTML input event.\n   * @param input\n   */\n  private setInput(input: string, triggerFilter: boolean) {\n    this.state = input\n    if (triggerFilter) {\n      this.onChange(this.state)\n      this.inputChange$.next(this.state)\n    }\n  }\n}\n","<ng-container>\n    <div class=\"input-wrapper\"\n        [ngClass]=\"{ 'expanded': expanded, 'collapsed': !expanded }\"\n        [ngStyle]=\"{ 'height.px': buttonHeight }\"\n        (click)=\"$event.stopPropagation();\">\n        <!-- INPUT FIELD -->\n        <div class=\"input-group\">\n            <input #input\n                [id]=\"id + '-input'\"\n                class=\"sdv-field\"\n                [attr.type]=\"type\"\n                [attr.name]=\"name\"\n                [attr.required]=\"required\"\n                [attr.email]=\"email\"\n                [min]=\"min\"\n                [max]=\"max\"\n                [step]=\"step\"\n                [attr.maxlength]=\"maxlength\"\n                [attr.minlength]=\"minlength\"\n                [pattern]=\"pattern\"\n                [disabled]=\"disabled\"\n                [autocomplete]=\"autocomplete\"\n                [autofocus]=\"autofocus\"\n                [readOnly]=\"readonly\"\n                [attr.role]=\"role\"\n                [attr.placeholder]=\"placeholder\"\n                [attr.aria-label]=\"description\"\n                [value]=\"state\"\n                (input)=\"onInput($event)\"\n                (focus)=\"onFocus($event)\"\n                (blur)=\"onBlur($event)\" />\n        </div>\n    </div>\n</ng-container>"]}
|
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
import { Directive, ElementRef, EventEmitter, Host, HostListener, Input, Optional, Output, ViewContainerRef, } from '@angular/core';
|
|
2
|
+
import { distinctUntilChanged, from, Subject, takeUntil, tap, withLatestFrom, } from 'rxjs';
|
|
3
|
+
import { NgvDropdownComponent } from '../dropdown.component';
|
|
4
|
+
import { NgvInputComponent } from '@sebgroup/green-angular/src/v-angular/input';
|
|
5
|
+
import { NgvTypeaheadDropdownListComponent } from '../typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component';
|
|
6
|
+
import { NgvTypeaheadInputComponent } from './typeahead-input/typeahead-input.component';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "../dropdown.component";
|
|
9
|
+
import * as i2 from "@sebgroup/green-angular/src/v-angular/input";
|
|
10
|
+
export class NgvTypeaheadDirective {
|
|
11
|
+
/** Function that filter the inputvalue */
|
|
12
|
+
set nggvTypeahead(value) {
|
|
13
|
+
this.typeaheadFunction = value;
|
|
14
|
+
// If inputsubscription already exists, unsubscribe and subscribe again
|
|
15
|
+
if (this.inputSubscription$) {
|
|
16
|
+
this.inputSubscription$?.unsubscribe();
|
|
17
|
+
this.inputSubscription$ = undefined;
|
|
18
|
+
this.handleInputChanges();
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
/** Forward text inputs to apply the filter function*/
|
|
22
|
+
onNgvInput(event) {
|
|
23
|
+
this.inputValue$.next(event);
|
|
24
|
+
}
|
|
25
|
+
/** Helper to the determine if the host is nggv-drodpown or nggv-input*/
|
|
26
|
+
get hostIsDropdown() {
|
|
27
|
+
return !!this.hostDropdown;
|
|
28
|
+
}
|
|
29
|
+
/** Predefined options */
|
|
30
|
+
get defaultNullishOption() {
|
|
31
|
+
return { key: null, label: this.unselectLabel || '\u00A0' };
|
|
32
|
+
}
|
|
33
|
+
get emptyOption() {
|
|
34
|
+
return { key: null, label: 'label.nomatchingoptions', disabled: true };
|
|
35
|
+
}
|
|
36
|
+
/** Name of the component. nggv-dropdown if NgvDropdownComponent or nggv-input if NgvInputComponent */
|
|
37
|
+
get localName() {
|
|
38
|
+
return this.element.nativeElement.localName;
|
|
39
|
+
}
|
|
40
|
+
constructor(viewContainerRef, element, hostDropdown, hostInput) {
|
|
41
|
+
this.viewContainerRef = viewContainerRef;
|
|
42
|
+
this.element = element;
|
|
43
|
+
this.hostDropdown = hostDropdown;
|
|
44
|
+
this.hostInput = hostInput;
|
|
45
|
+
/** Allow option to be unselected in the dropdown even if it is required. Defaults to true */
|
|
46
|
+
this.allowUnselect = true;
|
|
47
|
+
/** Emits the entered string the user has written in the input */
|
|
48
|
+
this.filterPhraseChange = new EventEmitter();
|
|
49
|
+
this.inputValue$ = new Subject();
|
|
50
|
+
this.onDestroy$ = new Subject();
|
|
51
|
+
}
|
|
52
|
+
ngOnInit() {
|
|
53
|
+
this.handleInputChanges();
|
|
54
|
+
this.inputValue$.next('');
|
|
55
|
+
if (this.hostIsDropdown)
|
|
56
|
+
this.createInput();
|
|
57
|
+
else
|
|
58
|
+
this.createDropdownList();
|
|
59
|
+
}
|
|
60
|
+
ngOnDestroy() {
|
|
61
|
+
this.onDestroy$.next(true);
|
|
62
|
+
this.onDestroy$.complete();
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* @internal
|
|
66
|
+
* Core functionality of typeahead. Emits input, then filters the result based on the supplied function
|
|
67
|
+
* If directive is applied on nggv-input, manually show or hide options in the list.
|
|
68
|
+
* If directive is applied on nggv-dropdown, let the dropdown itself choose when to open or close
|
|
69
|
+
*/
|
|
70
|
+
handleInputChanges() {
|
|
71
|
+
this.inputSubscription$ = this.inputValue$
|
|
72
|
+
.pipe(takeUntil(this.onDestroy$), distinctUntilChanged(), tap((inputValue) => this.filterPhraseChange.emit(inputValue)), this.typeaheadFunction ? this.typeaheadFunction : () => from([]), withLatestFrom(this.inputValue$))
|
|
73
|
+
.subscribe(([filteredValues, input]) => this.setOptions(filteredValues, input));
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* @internal
|
|
77
|
+
* Creates a nggv-input if the host itself is not a text-input
|
|
78
|
+
* Set styles to not display the input when closed
|
|
79
|
+
* Trigger filtering when changes occur in the field
|
|
80
|
+
* */
|
|
81
|
+
createInput() {
|
|
82
|
+
// Create the input component
|
|
83
|
+
this.inputComponent = this.viewContainerRef.createComponent(NgvTypeaheadInputComponent);
|
|
84
|
+
// Forward necessary info to component
|
|
85
|
+
this.inputComponent.setInput('hostComponent', this.hostDropdown);
|
|
86
|
+
this.inputComponent.setInput('selectedFormatter', this.selectedFormatter);
|
|
87
|
+
this.inputComponent.setInput('resultFormatter', this.resultFormatter);
|
|
88
|
+
// Listen to value changes
|
|
89
|
+
this.inputComponent.instance.nggvInput
|
|
90
|
+
.pipe(takeUntil(this.onDestroy$))
|
|
91
|
+
.subscribe((inputValue) => this.inputValue$.next(inputValue));
|
|
92
|
+
}
|
|
93
|
+
/** @internal Creates a nggv-dropdown-list if the host itself is a nggv-input */
|
|
94
|
+
createDropdownList() {
|
|
95
|
+
this.dropdownListComponent = this.viewContainerRef.createComponent(NgvTypeaheadDropdownListComponent);
|
|
96
|
+
this.dropdownListComponent.setInput('hostComponent', this.hostInput);
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* @internal Sets the options the user can select.
|
|
100
|
+
* If the host is a nggv-dropdown, utilize the dropdown itself to display the options
|
|
101
|
+
* If the host is a nggv-input, use the created nggv-dropdown-list to displaye the options
|
|
102
|
+
* @param filteredValues The options to display in the dropdown
|
|
103
|
+
* @param emptyInput If the input is empty
|
|
104
|
+
*/
|
|
105
|
+
setOptions(filteredValues, input) {
|
|
106
|
+
if (!filteredValues)
|
|
107
|
+
return;
|
|
108
|
+
// Conditionally add empty or nullish option if it's allowed, the input is empty and does not already contain nullish
|
|
109
|
+
const allowNullish = this.allowUnselect &&
|
|
110
|
+
!input &&
|
|
111
|
+
!(Object.keys(filteredValues[0]).includes('key') &&
|
|
112
|
+
filteredValues[0].key == null);
|
|
113
|
+
if (filteredValues.length === 0) {
|
|
114
|
+
filteredValues = [this.emptyOption];
|
|
115
|
+
}
|
|
116
|
+
else if (allowNullish) {
|
|
117
|
+
filteredValues = [this.defaultNullishOption].concat(filteredValues);
|
|
118
|
+
}
|
|
119
|
+
if (this.hostIsDropdown) {
|
|
120
|
+
// Add nullish option when no input is written (or when dropdown is epanded and has a selection)
|
|
121
|
+
this.hostDropdown.allowControlNullishOption = false;
|
|
122
|
+
this.hostDropdown.options = this.formatOptions(filteredValues);
|
|
123
|
+
this.hostDropdown.textToHighlight = `${input || ''}`;
|
|
124
|
+
this.hostDropdown.detectChanges();
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
if (!this.hostIsDropdown) {
|
|
128
|
+
this.dropdownListComponent.setInput('options', this.formatOptions(filteredValues));
|
|
129
|
+
this.dropdownListComponent.setInput('textToHighlight', `${input || ''}`);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* @internal Formats the available options to display in the dropdown list
|
|
134
|
+
* @param options The selected value
|
|
135
|
+
* @returns The formatted value
|
|
136
|
+
*/
|
|
137
|
+
formatOptions(options) {
|
|
138
|
+
if (!options)
|
|
139
|
+
return [];
|
|
140
|
+
if (!this.resultFormatter)
|
|
141
|
+
return options;
|
|
142
|
+
return options.map((value) => value?.label ? this.resultFormatter?.(value) : value);
|
|
143
|
+
}
|
|
144
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.ElementRef }, { token: i1.NgvDropdownComponent, host: true, optional: true }, { token: i2.NgvInputComponent, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
145
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: NgvTypeaheadDirective, isStandalone: true, selector: "nggv-input[nggvTypeahead], nggv-dropdown[nggvTypeahead]", inputs: { nggvTypeahead: "nggvTypeahead", resultFormatter: "resultFormatter", selectedFormatter: "selectedFormatter", allowUnselect: "allowUnselect", unselectLabel: "unselectLabel" }, outputs: { filterPhraseChange: "filterPhraseChange" }, host: { listeners: { "nggvInput": "onNgvInput($event)" } }, ngImport: i0 }); }
|
|
146
|
+
}
|
|
147
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadDirective, decorators: [{
|
|
148
|
+
type: Directive,
|
|
149
|
+
args: [{
|
|
150
|
+
selector: 'nggv-input[nggvTypeahead], nggv-dropdown[nggvTypeahead]',
|
|
151
|
+
standalone: true,
|
|
152
|
+
}]
|
|
153
|
+
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.ElementRef }, { type: i1.NgvDropdownComponent, decorators: [{
|
|
154
|
+
type: Optional
|
|
155
|
+
}, {
|
|
156
|
+
type: Host
|
|
157
|
+
}] }, { type: i2.NgvInputComponent, decorators: [{
|
|
158
|
+
type: Optional
|
|
159
|
+
}, {
|
|
160
|
+
type: Host
|
|
161
|
+
}] }]; }, propDecorators: { nggvTypeahead: [{
|
|
162
|
+
type: Input
|
|
163
|
+
}], resultFormatter: [{
|
|
164
|
+
type: Input
|
|
165
|
+
}], selectedFormatter: [{
|
|
166
|
+
type: Input
|
|
167
|
+
}], allowUnselect: [{
|
|
168
|
+
type: Input
|
|
169
|
+
}], unselectLabel: [{
|
|
170
|
+
type: Input
|
|
171
|
+
}], filterPhraseChange: [{
|
|
172
|
+
type: Output
|
|
173
|
+
}], onNgvInput: [{
|
|
174
|
+
type: HostListener,
|
|
175
|
+
args: ['nggvInput', ['$event']]
|
|
176
|
+
}] } });
|
|
177
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typeahead.directive.js","sourceRoot":"","sources":["../../../../../../../libs/angular/src/v-angular/dropdown/typeahead/typeahead.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAEL,SAAS,EACT,UAAU,EACV,YAAY,EACZ,IAAI,EACJ,YAAY,EACZ,KAAK,EAGL,QAAQ,EACR,MAAM,EACN,gBAAgB,GACjB,MAAM,eAAe,CAAA;AAEtB,OAAO,EACL,oBAAoB,EACpB,IAAI,EAEJ,OAAO,EAEP,SAAS,EACT,GAAG,EACH,cAAc,GACf,MAAM,MAAM,CAAA;AAGb,OAAO,EAAE,oBAAoB,EAAE,MAAM,uBAAuB,CAAA;AAC5D,OAAO,EAAE,iBAAiB,EAAE,MAAM,6CAA6C,CAAA;AAC/E,OAAO,EAAE,iCAAiC,EAAE,MAAM,wEAAwE,CAAA;AAC1H,OAAO,EAAE,0BAA0B,EAAE,MAAM,6CAA6C,CAAA;;;;AAMxF,MAAM,OAAO,qBAAqB;IAOhC,0CAA0C;IAC1C,IAAa,aAAa,CAAC,KAAgD;QACzE,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAA;QAE9B,uEAAuE;QACvE,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI,CAAC,kBAAkB,EAAE,WAAW,EAAE,CAAA;YACtC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAA;YACnC,IAAI,CAAC,kBAAkB,EAAE,CAAA;SAC1B;IACH,CAAC;IAiBD,sDAAsD;IACf,UAAU,CAAC,KAAU;QAC1D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IAC9B,CAAC;IAED,wEAAwE;IACxE,IAAI,cAAc;QAChB,OAAO,CAAC,CAAC,IAAI,CAAC,YAAY,CAAA;IAC5B,CAAC;IAED,yBAAyB;IACzB,IAAI,oBAAoB;QACtB,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,aAAa,IAAI,QAAQ,EAAE,CAAA;IAC7D,CAAC;IAED,IAAI,WAAW;QACb,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,KAAK,EAAE,yBAAyB,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAA;IACxE,CAAC;IAED,sGAAsG;IACtG,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,SAAS,CAAA;IAC7C,CAAC;IAUD,YACU,gBAAkC,EAClC,OAAmB,EACC,YAAkC,EAClC,SAA4B;QAHhD,qBAAgB,GAAhB,gBAAgB,CAAkB;QAClC,YAAO,GAAP,OAAO,CAAY;QACC,iBAAY,GAAZ,YAAY,CAAsB;QAClC,cAAS,GAAT,SAAS,CAAmB;QA7C1D,6FAA6F;QACpF,kBAAa,GAAG,IAAI,CAAA;QAK7B,iEAAiE;QACvD,uBAAkB,GAAG,IAAI,YAAY,EAAU,CAAA;QA8BjD,gBAAW,GAAG,IAAI,OAAO,EAAO,CAAA;QAEhC,eAAU,GAAG,IAAI,OAAO,EAAW,CAAA;IAOxC,CAAC;IAEJ,QAAQ;QACN,IAAI,CAAC,kBAAkB,EAAE,CAAA;QACzB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;QAEzB,IAAI,IAAI,CAAC,cAAc;YAAE,IAAI,CAAC,WAAW,EAAE,CAAA;;YACtC,IAAI,CAAC,kBAAkB,EAAE,CAAA;IAChC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC1B,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAA;IAC5B,CAAC;IAED;;;;;OAKG;IACK,kBAAkB;QACxB,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,WAAW;aACvC,IAAI,CACH,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,EAC1B,oBAAoB,EAAE,EACtB,GAAG,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,EAC7D,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,EAChE,cAAc,CAAC,IAAI,CAAC,WAAW,CAAC,CACjC;aACA,SAAS,CAAC,CAAC,CAAC,cAAc,EAAE,KAAK,CAAC,EAAE,EAAE,CACrC,IAAI,CAAC,UAAU,CAAC,cAAc,EAAE,KAAK,CAAC,CACvC,CAAA;IACL,CAAC;IAED;;;;;SAKK;IACG,WAAW;QACjB,6BAA6B;QAC7B,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,gBAAgB,CAAC,eAAe,CACzD,0BAA0B,CAC3B,CAAA;QACD,sCAAsC;QACtC,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,eAAe,EAAE,IAAI,CAAC,YAAY,CAAC,CAAA;QAChE,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,mBAAmB,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAA;QACzE,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,iBAAiB,EAAE,IAAI,CAAC,eAAe,CAAC,CAAA;QACrE,0BAA0B;QAC1B,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,SAAS;aACnC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;aAChC,SAAS,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAA;IACjE,CAAC;IAED,gFAAgF;IACxE,kBAAkB;QACxB,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,gBAAgB,CAAC,eAAe,CAChE,iCAAiC,CAClC,CAAA;QACD,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,eAAe,EAAE,IAAI,CAAC,SAAS,CAAC,CAAA;IACtE,CAAC;IAED;;;;;;OAMG;IACK,UAAU,CAAC,cAAmB,EAAE,KAAa;QACnD,IAAI,CAAC,cAAc;YAAE,OAAM;QAC3B,qHAAqH;QACrH,MAAM,YAAY,GAChB,IAAI,CAAC,aAAa;YAClB,CAAC,KAAK;YACN,CAAC,CACC,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC;gBAC9C,cAAc,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,IAAI,CAC9B,CAAA;QACH,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE;YAC/B,cAAc,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,CAAA;SACpC;aAAM,IAAI,YAAY,EAAE;YACvB,cAAc,GAAG,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,MAAM,CAAC,cAAc,CAAC,CAAA;SACpE;QAED,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,gGAAgG;YAChG,IAAI,CAAC,YAAY,CAAC,yBAAyB,GAAG,KAAK,CAAA;YACnD,IAAI,CAAC,YAAY,CAAC,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,cAAc,CAAC,CAAA;YAC9D,IAAI,CAAC,YAAY,CAAC,eAAe,GAAG,GAAG,KAAK,IAAI,EAAE,EAAE,CAAA;YACpD,IAAI,CAAC,YAAY,CAAC,aAAa,EAAE,CAAA;YACjC,OAAM;SACP;QAED,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACxB,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CACjC,SAAS,EACT,IAAI,CAAC,aAAa,CAAC,cAAc,CAAC,CACnC,CAAA;YACD,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,iBAAiB,EAAE,GAAG,KAAK,IAAI,EAAE,EAAE,CAAC,CAAA;SACzE;IACH,CAAC;IAED;;;;OAIG;IACK,aAAa,CAAC,OAAY;QAChC,IAAI,CAAC,OAAO;YAAE,OAAO,EAAE,CAAA;QACvB,IAAI,CAAC,IAAI,CAAC,eAAe;YAAE,OAAO,OAAO,CAAA;QACzC,OAAO,OAAO,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAC3B,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CACrD,CAAA;IACH,CAAC;+GA3LU,qBAAqB;mGAArB,qBAAqB;;4FAArB,qBAAqB;kBAJjC,SAAS;mBAAC;oBACT,QAAQ,EAAE,yDAAyD;oBACnE,UAAU,EAAE,IAAI;iBACjB;;0BAsEI,QAAQ;;0BAAI,IAAI;;0BAChB,QAAQ;;0BAAI,IAAI;4CA9DN,aAAa;sBAAzB,KAAK;gBAYG,eAAe;sBAAvB,KAAK;gBAGG,iBAAiB;sBAAzB,KAAK;gBAGG,aAAa;sBAArB,KAAK;gBAGG,aAAa;sBAArB,KAAK;gBAGI,kBAAkB;sBAA3B,MAAM;gBAGgC,UAAU;sBAAhD,YAAY;uBAAC,WAAW,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import {\n  ComponentRef,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  Host,\n  HostListener,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  ViewContainerRef,\n} from '@angular/core'\n\nimport {\n  distinctUntilChanged,\n  from,\n  OperatorFunction,\n  Subject,\n  Subscription,\n  takeUntil,\n  tap,\n  withLatestFrom,\n} from 'rxjs'\n\nimport { Option, OptionBase } from '@sebgroup/green-angular/src/v-angular/core'\nimport { NgvDropdownComponent } from '../dropdown.component'\nimport { NgvInputComponent } from '@sebgroup/green-angular/src/v-angular/input'\nimport { NgvTypeaheadDropdownListComponent } from '../typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component'\nimport { NgvTypeaheadInputComponent } from './typeahead-input/typeahead-input.component'\n\n@Directive({\n  selector: 'nggv-input[nggvTypeahead], nggv-dropdown[nggvTypeahead]',\n  standalone: true,\n})\nexport class NgvTypeaheadDirective<\n    K = string | null | undefined,\n    V = string | null | undefined,\n    T extends Option<K, V> = Option<K, V>,\n  >\n  implements OnInit, OnDestroy\n{\n  /** Function that filter the inputvalue */\n  @Input() set nggvTypeahead(value: OperatorFunction<string, T[]> | undefined) {\n    this.typeaheadFunction = value\n\n    // If inputsubscription already exists, unsubscribe and subscribe again\n    if (this.inputSubscription$) {\n      this.inputSubscription$?.unsubscribe()\n      this.inputSubscription$ = undefined\n      this.handleInputChanges()\n    }\n  }\n\n  /** Formats each item that is displayed as an option. Only applies format if the option if it implement Option interface. */\n  @Input() resultFormatter?: (option: OptionBase<any>) => string\n\n  /** Formats the selected item in the input when dropdown is opened. If no function is provided, it will display the value of the selected objects label property */\n  @Input() selectedFormatter?: (selected: OptionBase<any>) => string\n\n  /** Allow option to be unselected in the dropdown even if it is required. Defaults to true */\n  @Input() allowUnselect = true\n\n  /** Custom label for the unselect option */\n  @Input() unselectLabel?: string\n\n  /** Emits the entered string the user has written in the input */\n  @Output() filterPhraseChange = new EventEmitter<string>()\n\n  /** Forward text inputs to apply the filter function*/\n  @HostListener('nggvInput', ['$event']) onNgvInput(event: any) {\n    this.inputValue$.next(event)\n  }\n\n  /** Helper to the determine if the host is nggv-drodpown or nggv-input*/\n  get hostIsDropdown() {\n    return !!this.hostDropdown\n  }\n\n  /** Predefined options */\n  get defaultNullishOption(): OptionBase<any> {\n    return { key: null, label: this.unselectLabel || '\\u00A0' }\n  }\n\n  get emptyOption(): OptionBase<any> {\n    return { key: null, label: 'label.nomatchingoptions', disabled: true }\n  }\n\n  /** Name of the component. nggv-dropdown if NgvDropdownComponent or nggv-input if NgvInputComponent */\n  get localName() {\n    return this.element.nativeElement.localName\n  }\n\n  dropdownListComponent!: ComponentRef<NgvTypeaheadDropdownListComponent>\n  inputComponent!: ComponentRef<NgvTypeaheadInputComponent>\n\n  private typeaheadFunction?: OperatorFunction<string, T[]>\n  private inputValue$ = new Subject<any>()\n  private inputSubscription$?: Subscription\n  private onDestroy$ = new Subject<boolean>()\n\n  constructor(\n    private viewContainerRef: ViewContainerRef,\n    private element: ElementRef,\n    @Optional() @Host() private hostDropdown: NgvDropdownComponent,\n    @Optional() @Host() private hostInput: NgvInputComponent,\n  ) {}\n\n  ngOnInit() {\n    this.handleInputChanges()\n    this.inputValue$.next('')\n\n    if (this.hostIsDropdown) this.createInput()\n    else this.createDropdownList()\n  }\n\n  ngOnDestroy(): void {\n    this.onDestroy$.next(true)\n    this.onDestroy$.complete()\n  }\n\n  /**\n   * @internal\n   * Core functionality of typeahead. Emits input, then filters the result based on the supplied function\n   * If directive is applied on nggv-input, manually show or hide options in the list.\n   * If directive is applied on nggv-dropdown, let the dropdown itself choose when to open or close\n   */\n  private handleInputChanges() {\n    this.inputSubscription$ = this.inputValue$\n      .pipe(\n        takeUntil(this.onDestroy$),\n        distinctUntilChanged(),\n        tap((inputValue) => this.filterPhraseChange.emit(inputValue)),\n        this.typeaheadFunction ? this.typeaheadFunction : () => from([]),\n        withLatestFrom(this.inputValue$),\n      )\n      .subscribe(([filteredValues, input]) =>\n        this.setOptions(filteredValues, input),\n      )\n  }\n\n  /**\n   * @internal\n   * Creates a nggv-input if the host itself is not a text-input\n   * Set styles to not display the input when closed\n   * Trigger filtering when changes occur in the field\n   * */\n  private createInput() {\n    // Create the input component\n    this.inputComponent = this.viewContainerRef.createComponent(\n      NgvTypeaheadInputComponent,\n    )\n    // Forward necessary info to component\n    this.inputComponent.setInput('hostComponent', this.hostDropdown)\n    this.inputComponent.setInput('selectedFormatter', this.selectedFormatter)\n    this.inputComponent.setInput('resultFormatter', this.resultFormatter)\n    // Listen to value changes\n    this.inputComponent.instance.nggvInput\n      .pipe(takeUntil(this.onDestroy$))\n      .subscribe((inputValue) => this.inputValue$.next(inputValue))\n  }\n\n  /** @internal Creates a nggv-dropdown-list if the host itself is a nggv-input */\n  private createDropdownList() {\n    this.dropdownListComponent = this.viewContainerRef.createComponent(\n      NgvTypeaheadDropdownListComponent,\n    )\n    this.dropdownListComponent.setInput('hostComponent', this.hostInput)\n  }\n\n  /**\n   * @internal Sets the options the user can select.\n   * If the host is a nggv-dropdown, utilize the dropdown itself to display the options\n   * If the host is a nggv-input, use the created nggv-dropdown-list to displaye the options\n   * @param filteredValues The options to display in the dropdown\n   * @param emptyInput If the input is empty\n   */\n  private setOptions(filteredValues: T[], input: string) {\n    if (!filteredValues) return\n    // Conditionally add empty or nullish option if it's allowed, the input is empty and does not already contain nullish\n    const allowNullish =\n      this.allowUnselect &&\n      !input &&\n      !(\n        Object.keys(filteredValues[0]).includes('key') &&\n        filteredValues[0].key == null\n      )\n    if (filteredValues.length === 0) {\n      filteredValues = [this.emptyOption]\n    } else if (allowNullish) {\n      filteredValues = [this.defaultNullishOption].concat(filteredValues)\n    }\n\n    if (this.hostIsDropdown) {\n      // Add nullish option when no input is written (or when dropdown is epanded and has a selection)\n      this.hostDropdown.allowControlNullishOption = false\n      this.hostDropdown.options = this.formatOptions(filteredValues)\n      this.hostDropdown.textToHighlight = `${input || ''}`\n      this.hostDropdown.detectChanges()\n      return\n    }\n\n    if (!this.hostIsDropdown) {\n      this.dropdownListComponent.setInput(\n        'options',\n        this.formatOptions(filteredValues),\n      )\n      this.dropdownListComponent.setInput('textToHighlight', `${input || ''}`)\n    }\n  }\n\n  /**\n   * @internal Formats the available options to display in the dropdown list\n   * @param options The selected value\n   * @returns The formatted value\n   */\n  private formatOptions(options: T[]): OptionBase<any>[] {\n    if (!options) return []\n    if (!this.resultFormatter) return options\n    return options.map((value) =>\n      value?.label ? this.resultFormatter?.(value) : value,\n    )\n  }\n}\n"]}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { NgModule } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { NgvTypeaheadHighlightComponent } from './typeahead-highlight/typeahead-highlight.component';
|
|
4
|
+
import { NgvTypeaheadInputComponent } from './typeahead-input/typeahead-input.component';
|
|
5
|
+
import { NgvTooltipModule } from '@sebgroup/green-angular/src/v-angular/tooltip';
|
|
6
|
+
import { NgvTypeaheadDropdownListComponent } from './typeahead-dropdown-list/typeahead-dropdown-list.component';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export class NgvTypeaheadModule {
|
|
9
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
|
|
10
|
+
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadModule, declarations: [NgvTypeaheadHighlightComponent,
|
|
11
|
+
NgvTypeaheadInputComponent,
|
|
12
|
+
NgvTypeaheadDropdownListComponent], imports: [CommonModule, NgvTooltipModule], exports: [NgvTypeaheadHighlightComponent,
|
|
13
|
+
NgvTypeaheadInputComponent,
|
|
14
|
+
NgvTypeaheadDropdownListComponent] }); }
|
|
15
|
+
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadModule, imports: [CommonModule, NgvTooltipModule] }); }
|
|
16
|
+
}
|
|
17
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvTypeaheadModule, decorators: [{
|
|
18
|
+
type: NgModule,
|
|
19
|
+
args: [{
|
|
20
|
+
declarations: [
|
|
21
|
+
NgvTypeaheadHighlightComponent,
|
|
22
|
+
NgvTypeaheadInputComponent,
|
|
23
|
+
NgvTypeaheadDropdownListComponent,
|
|
24
|
+
],
|
|
25
|
+
imports: [CommonModule, NgvTooltipModule],
|
|
26
|
+
exports: [
|
|
27
|
+
NgvTypeaheadHighlightComponent,
|
|
28
|
+
NgvTypeaheadInputComponent,
|
|
29
|
+
NgvTypeaheadDropdownListComponent,
|
|
30
|
+
],
|
|
31
|
+
}]
|
|
32
|
+
}] });
|
|
33
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZWFoZWFkLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci9zcmMvdi1hbmd1bGFyL2Ryb3Bkb3duL3R5cGVhaGVhZC90eXBlYWhlYWQubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFDeEMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzlDLE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLHFEQUFxRCxDQUFBO0FBQ3BHLE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLDZDQUE2QyxDQUFBO0FBQ3hGLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLCtDQUErQyxDQUFBO0FBQ2hGLE9BQU8sRUFBRSxpQ0FBaUMsRUFBRSxNQUFNLDZEQUE2RCxDQUFBOztBQWUvRyxNQUFNLE9BQU8sa0JBQWtCOytHQUFsQixrQkFBa0I7Z0hBQWxCLGtCQUFrQixpQkFYM0IsOEJBQThCO1lBQzlCLDBCQUEwQjtZQUMxQixpQ0FBaUMsYUFFekIsWUFBWSxFQUFFLGdCQUFnQixhQUV0Qyw4QkFBOEI7WUFDOUIsMEJBQTBCO1lBQzFCLGlDQUFpQztnSEFHeEIsa0JBQWtCLFlBUG5CLFlBQVksRUFBRSxnQkFBZ0I7OzRGQU83QixrQkFBa0I7a0JBYjlCLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFO3dCQUNaLDhCQUE4Qjt3QkFDOUIsMEJBQTBCO3dCQUMxQixpQ0FBaUM7cUJBQ2xDO29CQUNELE9BQU8sRUFBRSxDQUFDLFlBQVksRUFBRSxnQkFBZ0IsQ0FBQztvQkFDekMsT0FBTyxFQUFFO3dCQUNQLDhCQUE4Qjt3QkFDOUIsMEJBQTBCO3dCQUMxQixpQ0FBaUM7cUJBQ2xDO2lCQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJ1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJ1xuaW1wb3J0IHsgTmd2VHlwZWFoZWFkSGlnaGxpZ2h0Q29tcG9uZW50IH0gZnJvbSAnLi90eXBlYWhlYWQtaGlnaGxpZ2h0L3R5cGVhaGVhZC1oaWdobGlnaHQuY29tcG9uZW50J1xuaW1wb3J0IHsgTmd2VHlwZWFoZWFkSW5wdXRDb21wb25lbnQgfSBmcm9tICcuL3R5cGVhaGVhZC1pbnB1dC90eXBlYWhlYWQtaW5wdXQuY29tcG9uZW50J1xuaW1wb3J0IHsgTmd2VG9vbHRpcE1vZHVsZSB9IGZyb20gJ0BzZWJncm91cC9ncmVlbi1hbmd1bGFyL3NyYy92LWFuZ3VsYXIvdG9vbHRpcCdcbmltcG9ydCB7IE5ndlR5cGVhaGVhZERyb3Bkb3duTGlzdENvbXBvbmVudCB9IGZyb20gJy4vdHlwZWFoZWFkLWRyb3Bkb3duLWxpc3QvdHlwZWFoZWFkLWRyb3Bkb3duLWxpc3QuY29tcG9uZW50J1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBOZ3ZUeXBlYWhlYWRIaWdobGlnaHRDb21wb25lbnQsXG4gICAgTmd2VHlwZWFoZWFkSW5wdXRDb21wb25lbnQsXG4gICAgTmd2VHlwZWFoZWFkRHJvcGRvd25MaXN0Q29tcG9uZW50LFxuICBdLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBOZ3ZUb29sdGlwTW9kdWxlXSxcbiAgZXhwb3J0czogW1xuICAgIE5ndlR5cGVhaGVhZEhpZ2hsaWdodENvbXBvbmVudCxcbiAgICBOZ3ZUeXBlYWhlYWRJbnB1dENvbXBvbmVudCxcbiAgICBOZ3ZUeXBlYWhlYWREcm9wZG93bkxpc3RDb21wb25lbnQsXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIE5ndlR5cGVhaGVhZE1vZHVsZSB7fVxuIl19
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
{
|
|
2
|
+
"error.fieldinputmask": "Invalid value pattern",
|
|
3
|
+
"error.fieldrequired": "Field must have content",
|
|
4
|
+
"error.fieldmaxlength": "Field content should not be longer than {{requiredLength}} characters",
|
|
5
|
+
"label.defaultlabel": "Label",
|
|
6
|
+
"label.maxlength": "characters left",
|
|
7
|
+
"label.optional": "Optional",
|
|
8
|
+
"button_cancel": "Cancel",
|
|
9
|
+
"button_apply": "Apply",
|
|
10
|
+
"button_save": "Save",
|
|
11
|
+
"text_dialogue": "You can supply the content seen here either through the <code>[content]=\"string\"</code> property or</br>by passing children between the opening and closing tags <code><c-dialog> ...children </c-dialog></code>"
|
|
12
|
+
}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { Inject, Injectable, isDevMode, NgModule } from '@angular/core';
|
|
2
|
+
import { delay, lastValueFrom, of } from 'rxjs';
|
|
3
|
+
import { provideTransloco, provideTranslocoMissingHandler, TRANSLOCO_TRANSPILER, TranslocoModule, } from '@ngneat/transloco';
|
|
4
|
+
import defaultLang from './i18n.json';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
export class NgvMissingHandler {
|
|
7
|
+
constructor(transpiler) {
|
|
8
|
+
this.transpiler = transpiler;
|
|
9
|
+
}
|
|
10
|
+
handle(key, _, params) {
|
|
11
|
+
const keyWithoutLocale = key.charAt(2) === '.' ? key.substring(3) : key;
|
|
12
|
+
const withoutScope = keyWithoutLocale.replace(/^((?:\w+)(?<!label|heading|button|alt|link|title|href|fieldhelp|error|text|image|list)(?:\.))/, '');
|
|
13
|
+
const transpiledKey = this.transpiler.transpile(defaultLang[keyWithoutLocale], params, {}, keyWithoutLocale);
|
|
14
|
+
return transpiledKey || withoutScope;
|
|
15
|
+
}
|
|
16
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvMissingHandler, deps: [{ token: TRANSLOCO_TRANSPILER }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
17
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvMissingHandler }); }
|
|
18
|
+
}
|
|
19
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvMissingHandler, decorators: [{
|
|
20
|
+
type: Injectable
|
|
21
|
+
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
|
22
|
+
type: Inject,
|
|
23
|
+
args: [TRANSLOCO_TRANSPILER]
|
|
24
|
+
}] }]; } });
|
|
25
|
+
const en = {
|
|
26
|
+
'error.fieldinputmask': 'Invalid value pattern',
|
|
27
|
+
'error.fieldrequired': 'Field must have content',
|
|
28
|
+
'error.fieldmaxlength': 'Field content should not be longer than {{requiredLength}} characters',
|
|
29
|
+
'label.defaultlabel': 'Label',
|
|
30
|
+
'label.maxlength': 'characters left',
|
|
31
|
+
'label.optional': 'Optional',
|
|
32
|
+
};
|
|
33
|
+
const sv = {
|
|
34
|
+
'error.fieldinputmask': 'Icke giltigt tecken mönster',
|
|
35
|
+
'error.fieldrequired': 'Fältet får inte lämnas tomt',
|
|
36
|
+
'error.fieldmaxlength': 'Fältinnehållet måste vara längre än {{requiredLength}} tecken',
|
|
37
|
+
'label.maxlength': 'tecken kvar',
|
|
38
|
+
};
|
|
39
|
+
class TranslocoInlineLoader {
|
|
40
|
+
getTranslation(lang) {
|
|
41
|
+
if (lang === 'sv') {
|
|
42
|
+
return lastValueFrom(of(sv).pipe(delay(1500)));
|
|
43
|
+
}
|
|
44
|
+
return lastValueFrom(of(en).pipe(delay(500)));
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
export class NgvI18nModule {
|
|
48
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvI18nModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
|
|
49
|
+
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "16.2.12", ngImport: i0, type: NgvI18nModule, exports: [TranslocoModule] }); }
|
|
50
|
+
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvI18nModule, providers: [
|
|
51
|
+
provideTransloco({
|
|
52
|
+
config: {
|
|
53
|
+
availableLangs: ['en', 'sv'],
|
|
54
|
+
defaultLang: 'en',
|
|
55
|
+
// Remove this option if your application doesn't support changing language in runtime.
|
|
56
|
+
reRenderOnLangChange: true,
|
|
57
|
+
prodMode: !isDevMode(),
|
|
58
|
+
},
|
|
59
|
+
loader: TranslocoInlineLoader,
|
|
60
|
+
}),
|
|
61
|
+
provideTranslocoMissingHandler(NgvMissingHandler),
|
|
62
|
+
], imports: [TranslocoModule] }); }
|
|
63
|
+
}
|
|
64
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NgvI18nModule, decorators: [{
|
|
65
|
+
type: NgModule,
|
|
66
|
+
args: [{
|
|
67
|
+
providers: [
|
|
68
|
+
provideTransloco({
|
|
69
|
+
config: {
|
|
70
|
+
availableLangs: ['en', 'sv'],
|
|
71
|
+
defaultLang: 'en',
|
|
72
|
+
// Remove this option if your application doesn't support changing language in runtime.
|
|
73
|
+
reRenderOnLangChange: true,
|
|
74
|
+
prodMode: !isDevMode(),
|
|
75
|
+
},
|
|
76
|
+
loader: TranslocoInlineLoader,
|
|
77
|
+
}),
|
|
78
|
+
provideTranslocoMissingHandler(NgvMissingHandler),
|
|
79
|
+
],
|
|
80
|
+
exports: [TranslocoModule],
|
|
81
|
+
}]
|
|
82
|
+
}] });
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaTE4bi5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvc3JjL3YtYW5ndWxhci9pMThuL2kxOG4ubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLEVBQUUsVUFBVSxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFeEUsT0FBTyxFQUFFLEtBQUssRUFBRSxhQUFhLEVBQUUsRUFBRSxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRWhELE9BQU8sRUFFTCxnQkFBZ0IsRUFDaEIsOEJBQThCLEVBQzlCLG9CQUFvQixFQUlwQixlQUFlLEdBRWhCLE1BQU0sbUJBQW1CLENBQUM7QUFFM0IsT0FBTyxXQUFXLE1BQU0sYUFBYSxDQUFDOztBQUd0QyxNQUFNLE9BQU8saUJBQWlCO0lBQzVCLFlBQWtELFVBQStCO1FBQS9CLGVBQVUsR0FBVixVQUFVLENBQXFCO0lBQUcsQ0FBQztJQUNyRixNQUFNLENBQUMsR0FBVyxFQUFFLENBQThCLEVBQUUsTUFBZTtRQUNqRSxNQUFNLGdCQUFnQixHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEtBQUssR0FBRyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUM7UUFDeEUsTUFBTSxZQUFZLEdBQUcsZ0JBQWdCLENBQUMsT0FBTyxDQUMzQywrRkFBK0YsRUFDL0YsRUFBRSxDQUNILENBQUM7UUFDRixNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FDN0MsV0FBVyxDQUFDLGdCQUE0QyxDQUFDLEVBQ3pELE1BQU0sRUFDTixFQUFFLEVBQ0YsZ0JBQWdCLENBQ2pCLENBQUM7UUFDRixPQUFPLGFBQWEsSUFBSSxZQUFZLENBQUM7SUFDdkMsQ0FBQzsrR0FmVSxpQkFBaUIsa0JBQ1Isb0JBQW9CO21IQUQ3QixpQkFBaUI7OzRGQUFqQixpQkFBaUI7a0JBRDdCLFVBQVU7OzBCQUVJLE1BQU07MkJBQUMsb0JBQW9COztBQWlCMUMsTUFBTSxFQUFFLEdBQUc7SUFDVCxzQkFBc0IsRUFBRSx1QkFBdUI7SUFDL0MscUJBQXFCLEVBQUUseUJBQXlCO0lBQ2hELHNCQUFzQixFQUFFLHVFQUF1RTtJQUMvRixvQkFBb0IsRUFBRSxPQUFPO0lBQzdCLGlCQUFpQixFQUFFLGlCQUFpQjtJQUNwQyxnQkFBZ0IsRUFBRSxVQUFVO0NBQzdCLENBQUM7QUFFRixNQUFNLEVBQUUsR0FBRztJQUNULHNCQUFzQixFQUFFLDZCQUE2QjtJQUNyRCxxQkFBcUIsRUFBRSw2QkFBNkI7SUFDcEQsc0JBQXNCLEVBQUUsK0RBQStEO0lBQ3ZGLGlCQUFpQixFQUFFLGFBQWE7Q0FDakMsQ0FBQztBQUNGLE1BQU0scUJBQXFCO0lBQ3pCLGNBQWMsQ0FBQyxJQUFZO1FBQ3pCLElBQUksSUFBSSxLQUFLLElBQUksRUFBRTtZQUNqQixPQUFPLGFBQWEsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7U0FDaEQ7UUFDRCxPQUFPLGFBQWEsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDaEQsQ0FBQztDQUNGO0FBa0JELE1BQU0sT0FBTyxhQUFhOytHQUFiLGFBQWE7Z0hBQWIsYUFBYSxZQUZkLGVBQWU7Z0hBRWQsYUFBYSxhQWZiO1lBQ1QsZ0JBQWdCLENBQUM7Z0JBQ2YsTUFBTSxFQUFFO29CQUNOLGNBQWMsRUFBRSxDQUFDLElBQUksRUFBRSxJQUFJLENBQUM7b0JBQzVCLFdBQVcsRUFBRSxJQUFJO29CQUNqQix1RkFBdUY7b0JBQ3ZGLG9CQUFvQixFQUFFLElBQUk7b0JBQzFCLFFBQVEsRUFBRSxDQUFDLFNBQVMsRUFBRTtpQkFDdkI7Z0JBQ0QsTUFBTSxFQUFFLHFCQUFxQjthQUM5QixDQUFDO1lBQ0YsOEJBQThCLENBQUMsaUJBQWlCLENBQUM7U0FDbEQsWUFDUyxlQUFlOzs0RkFFZCxhQUFhO2tCQWhCekIsUUFBUTttQkFBQztvQkFDUixTQUFTLEVBQUU7d0JBQ1QsZ0JBQWdCLENBQUM7NEJBQ2YsTUFBTSxFQUFFO2dDQUNOLGNBQWMsRUFBRSxDQUFDLElBQUksRUFBRSxJQUFJLENBQUM7Z0NBQzVCLFdBQVcsRUFBRSxJQUFJO2dDQUNqQix1RkFBdUY7Z0NBQ3ZGLG9CQUFvQixFQUFFLElBQUk7Z0NBQzFCLFFBQVEsRUFBRSxDQUFDLFNBQVMsRUFBRTs2QkFDdkI7NEJBQ0QsTUFBTSxFQUFFLHFCQUFxQjt5QkFDOUIsQ0FBQzt3QkFDRiw4QkFBOEIsQ0FBQyxpQkFBaUIsQ0FBQztxQkFDbEQ7b0JBQ0QsT0FBTyxFQUFFLENBQUMsZUFBZSxDQUFDO2lCQUMzQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdCwgSW5qZWN0YWJsZSwgaXNEZXZNb2RlLCBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQgeyBkZWxheSwgbGFzdFZhbHVlRnJvbSwgb2YgfSBmcm9tICdyeGpzJztcblxuaW1wb3J0IHtcbiAgSGFzaE1hcCxcbiAgcHJvdmlkZVRyYW5zbG9jbyxcbiAgcHJvdmlkZVRyYW5zbG9jb01pc3NpbmdIYW5kbGVyLFxuICBUUkFOU0xPQ09fVFJBTlNQSUxFUixcbiAgVHJhbnNsb2NvTG9hZGVyLFxuICBUcmFuc2xvY29NaXNzaW5nSGFuZGxlcixcbiAgVHJhbnNsb2NvTWlzc2luZ0hhbmRsZXJEYXRhLFxuICBUcmFuc2xvY29Nb2R1bGUsXG4gIFRyYW5zbG9jb1RyYW5zcGlsZXIsXG59IGZyb20gJ0BuZ25lYXQvdHJhbnNsb2NvJztcblxuaW1wb3J0IGRlZmF1bHRMYW5nIGZyb20gJy4vaTE4bi5qc29uJztcblxuQEluamVjdGFibGUoKVxuZXhwb3J0IGNsYXNzIE5ndk1pc3NpbmdIYW5kbGVyIGltcGxlbWVudHMgVHJhbnNsb2NvTWlzc2luZ0hhbmRsZXIge1xuICBjb25zdHJ1Y3RvcihASW5qZWN0KFRSQU5TTE9DT19UUkFOU1BJTEVSKSBwcml2YXRlIHRyYW5zcGlsZXI6IFRyYW5zbG9jb1RyYW5zcGlsZXIpIHt9XG4gIGhhbmRsZShrZXk6IHN0cmluZywgXzogVHJhbnNsb2NvTWlzc2luZ0hhbmRsZXJEYXRhLCBwYXJhbXM6IEhhc2hNYXApIHtcbiAgICBjb25zdCBrZXlXaXRob3V0TG9jYWxlID0ga2V5LmNoYXJBdCgyKSA9PT0gJy4nID8ga2V5LnN1YnN0cmluZygzKSA6IGtleTtcbiAgICBjb25zdCB3aXRob3V0U2NvcGUgPSBrZXlXaXRob3V0TG9jYWxlLnJlcGxhY2UoXG4gICAgICAvXigoPzpcXHcrKSg/PCFsYWJlbHxoZWFkaW5nfGJ1dHRvbnxhbHR8bGlua3x0aXRsZXxocmVmfGZpZWxkaGVscHxlcnJvcnx0ZXh0fGltYWdlfGxpc3QpKD86XFwuKSkvLFxuICAgICAgJycsXG4gICAgKTtcbiAgICBjb25zdCB0cmFuc3BpbGVkS2V5ID0gdGhpcy50cmFuc3BpbGVyLnRyYW5zcGlsZShcbiAgICAgIGRlZmF1bHRMYW5nW2tleVdpdGhvdXRMb2NhbGUgYXMga2V5b2YgdHlwZW9mIGRlZmF1bHRMYW5nXSxcbiAgICAgIHBhcmFtcyxcbiAgICAgIHt9LFxuICAgICAga2V5V2l0aG91dExvY2FsZSxcbiAgICApO1xuICAgIHJldHVybiB0cmFuc3BpbGVkS2V5IHx8IHdpdGhvdXRTY29wZTtcbiAgfVxufVxuXG5jb25zdCBlbiA9IHtcbiAgJ2Vycm9yLmZpZWxkaW5wdXRtYXNrJzogJ0ludmFsaWQgdmFsdWUgcGF0dGVybicsXG4gICdlcnJvci5maWVsZHJlcXVpcmVkJzogJ0ZpZWxkIG11c3QgaGF2ZSBjb250ZW50JyxcbiAgJ2Vycm9yLmZpZWxkbWF4bGVuZ3RoJzogJ0ZpZWxkIGNvbnRlbnQgc2hvdWxkIG5vdCBiZSBsb25nZXIgdGhhbiB7e3JlcXVpcmVkTGVuZ3RofX0gY2hhcmFjdGVycycsXG4gICdsYWJlbC5kZWZhdWx0bGFiZWwnOiAnTGFiZWwnLFxuICAnbGFiZWwubWF4bGVuZ3RoJzogJ2NoYXJhY3RlcnMgbGVmdCcsXG4gICdsYWJlbC5vcHRpb25hbCc6ICdPcHRpb25hbCcsXG59O1xuXG5jb25zdCBzdiA9IHtcbiAgJ2Vycm9yLmZpZWxkaW5wdXRtYXNrJzogJ0lja2UgZ2lsdGlndCB0ZWNrZW4gbcO2bnN0ZXInLFxuICAnZXJyb3IuZmllbGRyZXF1aXJlZCc6ICdGw6RsdGV0IGbDpXIgaW50ZSBsw6RtbmFzIHRvbXQnLFxuICAnZXJyb3IuZmllbGRtYXhsZW5ndGgnOiAnRsOkbHRpbm5laMOlbGxldCBtw6VzdGUgdmFyYSBsw6RuZ3JlIMOkbiB7e3JlcXVpcmVkTGVuZ3RofX0gdGVja2VuJyxcbiAgJ2xhYmVsLm1heGxlbmd0aCc6ICd0ZWNrZW4ga3ZhcicsXG59O1xuY2xhc3MgVHJhbnNsb2NvSW5saW5lTG9hZGVyIGltcGxlbWVudHMgVHJhbnNsb2NvTG9hZGVyIHtcbiAgZ2V0VHJhbnNsYXRpb24obGFuZzogc3RyaW5nKSB7XG4gICAgaWYgKGxhbmcgPT09ICdzdicpIHtcbiAgICAgIHJldHVybiBsYXN0VmFsdWVGcm9tKG9mKHN2KS5waXBlKGRlbGF5KDE1MDApKSk7XG4gICAgfVxuICAgIHJldHVybiBsYXN0VmFsdWVGcm9tKG9mKGVuKS5waXBlKGRlbGF5KDUwMCkpKTtcbiAgfVxufVxuXG5ATmdNb2R1bGUoe1xuICBwcm92aWRlcnM6IFtcbiAgICBwcm92aWRlVHJhbnNsb2NvKHtcbiAgICAgIGNvbmZpZzoge1xuICAgICAgICBhdmFpbGFibGVMYW5nczogWydlbicsICdzdiddLFxuICAgICAgICBkZWZhdWx0TGFuZzogJ2VuJyxcbiAgICAgICAgLy8gUmVtb3ZlIHRoaXMgb3B0aW9uIGlmIHlvdXIgYXBwbGljYXRpb24gZG9lc24ndCBzdXBwb3J0IGNoYW5naW5nIGxhbmd1YWdlIGluIHJ1bnRpbWUuXG4gICAgICAgIHJlUmVuZGVyT25MYW5nQ2hhbmdlOiB0cnVlLFxuICAgICAgICBwcm9kTW9kZTogIWlzRGV2TW9kZSgpLFxuICAgICAgfSxcbiAgICAgIGxvYWRlcjogVHJhbnNsb2NvSW5saW5lTG9hZGVyLFxuICAgIH0pLFxuICAgIHByb3ZpZGVUcmFuc2xvY29NaXNzaW5nSGFuZGxlcihOZ3ZNaXNzaW5nSGFuZGxlciksXG4gIF0sXG4gIGV4cG9ydHM6IFtUcmFuc2xvY29Nb2R1bGVdLFxufSlcbmV4cG9ydCBjbGFzcyBOZ3ZJMThuTW9kdWxlIHt9XG4iXX0=
|