@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.
Files changed (189) hide show
  1. package/esm2022/src/v-angular/base-control-value-accessor/base-control-value-accessor.component.mjs +9 -9
  2. package/esm2022/src/v-angular/button/button.component.mjs +5 -5
  3. package/esm2022/src/v-angular/character-countdown/character-countdown.directive.mjs +5 -5
  4. package/esm2022/src/v-angular/core/core.globals.mjs +3 -3
  5. package/esm2022/src/v-angular/dropdown/dropdown-list/dropdown-list.component.mjs +4 -4
  6. package/esm2022/src/v-angular/dropdown/typeahead/index.mjs +4 -1
  7. package/esm2022/src/v-angular/dropdown/typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component.mjs +5 -5
  8. package/esm2022/src/v-angular/dropdown/typeahead/typeahead.directive.mjs +177 -0
  9. package/esm2022/src/v-angular/info-circle/info-circle.component.mjs +3 -3
  10. package/esm2022/src/v-angular/input/input.component.mjs +7 -7
  11. package/esm2022/src/v-angular/input-mask/input-mask-format.pipe.mjs +5 -4
  12. package/esm2022/src/v-angular/input-mask/input-mask.directive.mjs +8 -8
  13. package/esm2022/src/v-angular/modal/dialog/dialog.component.mjs +16 -16
  14. package/esm2022/src/v-angular/modal/fold-out/fold-out.directive.mjs +3 -3
  15. package/esm2022/src/v-angular/modal/modal.globals.mjs +3 -3
  16. package/esm2022/src/v-angular/modal/slide-out/slide-out.component.mjs +15 -15
  17. package/esm2022/src/v-angular/textarea/textarea.component.mjs +4 -3
  18. package/esm2022/src/v-angular/textarea/textarea.module.mjs +5 -4
  19. package/esm2022/src/v-angular/tooltip/tooltip.directive.mjs +13 -13
  20. package/esm2022/v-angular/base-control-value-accessor/base-control-value-accessor.component.mjs +287 -0
  21. package/esm2022/v-angular/base-control-value-accessor/base-control-value-accessor.module.mjs +17 -0
  22. package/esm2022/v-angular/base-control-value-accessor/index.mjs +3 -0
  23. package/esm2022/v-angular/breadcrumbs/breadcrumbs.component.mjs +25 -0
  24. package/esm2022/v-angular/breadcrumbs/breadcrumbs.module.mjs +20 -0
  25. package/esm2022/v-angular/breadcrumbs/index.mjs +3 -0
  26. package/esm2022/v-angular/button/button.component.mjs +108 -0
  27. package/esm2022/v-angular/button/button.module.mjs +20 -0
  28. package/esm2022/v-angular/button/index.mjs +3 -0
  29. package/esm2022/v-angular/card/card.component.mjs +11 -0
  30. package/esm2022/v-angular/card/card.module.mjs +18 -0
  31. package/esm2022/v-angular/card/index.mjs +3 -0
  32. package/esm2022/v-angular/character-countdown/character-countdown.directive.mjs +51 -0
  33. package/esm2022/v-angular/character-countdown/character-countdown.module.mjs +18 -0
  34. package/esm2022/v-angular/character-countdown/index.mjs +3 -0
  35. package/esm2022/v-angular/checkbox/checkbox.component.mjs +72 -0
  36. package/esm2022/v-angular/checkbox/checkbox.module.mjs +19 -0
  37. package/esm2022/v-angular/checkbox/index.mjs +3 -0
  38. package/esm2022/v-angular/dropdown/dropdown-list/dropdown-list.component.mjs +256 -0
  39. package/esm2022/v-angular/dropdown/dropdown-list/index.mjs +2 -0
  40. package/esm2022/v-angular/dropdown/dropdown.component.mjs +239 -0
  41. package/esm2022/v-angular/dropdown/dropdown.module.mjs +22 -0
  42. package/esm2022/v-angular/dropdown/index.mjs +6 -0
  43. package/esm2022/v-angular/dropdown/typeahead/index.mjs +6 -0
  44. package/esm2022/v-angular/dropdown/typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component.mjs +98 -0
  45. package/esm2022/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.mjs +85 -0
  46. package/esm2022/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.mjs +132 -0
  47. package/esm2022/v-angular/dropdown/typeahead/typeahead.directive.mjs +177 -0
  48. package/esm2022/v-angular/dropdown/typeahead/typeahead.module.mjs +33 -0
  49. package/esm2022/v-angular/i18n/i18n.json +12 -0
  50. package/esm2022/v-angular/i18n/i18n.module.mjs +83 -0
  51. package/esm2022/v-angular/i18n/i18n.test.module.mjs +89 -0
  52. package/esm2022/v-angular/i18n/index.mjs +3 -0
  53. package/esm2022/v-angular/index.mjs +18 -0
  54. package/esm2022/v-angular/info-circle/index.mjs +3 -0
  55. package/esm2022/v-angular/info-circle/info-circle.component.mjs +28 -0
  56. package/esm2022/v-angular/info-circle/info-circle.module.mjs +21 -0
  57. package/esm2022/v-angular/input/index.mjs +3 -0
  58. package/esm2022/v-angular/input/input.component.mjs +221 -0
  59. package/esm2022/v-angular/input/input.module.mjs +32 -0
  60. package/esm2022/v-angular/input-mask/config.mjs +9 -0
  61. package/esm2022/v-angular/input-mask/constants.mjs +2 -0
  62. package/esm2022/v-angular/input-mask/index.mjs +6 -0
  63. package/esm2022/v-angular/input-mask/input-mask-format.pipe.mjs +21 -0
  64. package/esm2022/v-angular/input-mask/input-mask.directive.mjs +165 -0
  65. package/esm2022/v-angular/input-mask/input-mask.module.mjs +35 -0
  66. package/esm2022/v-angular/input-mask/input-mask.types.mjs +2 -0
  67. package/esm2022/v-angular/modal/dialog/dialog.component.mjs +190 -0
  68. package/esm2022/v-angular/modal/fold-out/fold-out.component.mjs +56 -0
  69. package/esm2022/v-angular/modal/fold-out/fold-out.directive.mjs +19 -0
  70. package/esm2022/v-angular/modal/index.mjs +6 -0
  71. package/esm2022/v-angular/modal/modal.globals.mjs +20 -0
  72. package/esm2022/v-angular/modal/modal.module.mjs +40 -0
  73. package/esm2022/v-angular/modal/modal.types.mjs +2 -0
  74. package/esm2022/v-angular/modal/slide-out/slide-out.component.mjs +229 -0
  75. package/esm2022/v-angular/radio/index.mjs +3 -0
  76. package/esm2022/v-angular/radio/radio.component.mjs +130 -0
  77. package/esm2022/v-angular/radio/radio.module.mjs +20 -0
  78. package/esm2022/v-angular/sebgroup-green-angular-v-angular.mjs +5 -0
  79. package/esm2022/v-angular/textarea/index.mjs +3 -0
  80. package/esm2022/v-angular/textarea/textarea.component.mjs +102 -0
  81. package/esm2022/v-angular/textarea/textarea.module.mjs +19 -0
  82. package/esm2022/v-angular/tooltip/index.mjs +3 -0
  83. package/esm2022/v-angular/tooltip/tooltip.directive.mjs +273 -0
  84. package/esm2022/v-angular/tooltip/tooltip.module.mjs +18 -0
  85. package/esm2022/v-angular/v-angular.module.mjs +80 -0
  86. package/fesm2022/sebgroup-green-angular-src-v-angular-base-control-value-accessor.mjs +8 -8
  87. package/fesm2022/sebgroup-green-angular-src-v-angular-base-control-value-accessor.mjs.map +1 -1
  88. package/fesm2022/sebgroup-green-angular-src-v-angular-button.mjs +4 -4
  89. package/fesm2022/sebgroup-green-angular-src-v-angular-button.mjs.map +1 -1
  90. package/fesm2022/sebgroup-green-angular-src-v-angular-character-countdown.mjs +4 -4
  91. package/fesm2022/sebgroup-green-angular-src-v-angular-character-countdown.mjs.map +1 -1
  92. package/fesm2022/sebgroup-green-angular-src-v-angular-core.mjs +2 -2
  93. package/fesm2022/sebgroup-green-angular-src-v-angular-core.mjs.map +1 -1
  94. package/fesm2022/sebgroup-green-angular-src-v-angular-dropdown.mjs +179 -10
  95. package/fesm2022/sebgroup-green-angular-src-v-angular-dropdown.mjs.map +1 -1
  96. package/fesm2022/sebgroup-green-angular-src-v-angular-info-circle.mjs +2 -2
  97. package/fesm2022/sebgroup-green-angular-src-v-angular-info-circle.mjs.map +1 -1
  98. package/fesm2022/sebgroup-green-angular-src-v-angular-input-mask.mjs +11 -10
  99. package/fesm2022/sebgroup-green-angular-src-v-angular-input-mask.mjs.map +1 -1
  100. package/fesm2022/sebgroup-green-angular-src-v-angular-input.mjs +6 -6
  101. package/fesm2022/sebgroup-green-angular-src-v-angular-input.mjs.map +1 -1
  102. package/fesm2022/sebgroup-green-angular-src-v-angular-modal.mjs +33 -33
  103. package/fesm2022/sebgroup-green-angular-src-v-angular-modal.mjs.map +1 -1
  104. package/fesm2022/sebgroup-green-angular-src-v-angular-textarea.mjs +7 -5
  105. package/fesm2022/sebgroup-green-angular-src-v-angular-textarea.mjs.map +1 -1
  106. package/fesm2022/sebgroup-green-angular-src-v-angular-tooltip.mjs +12 -12
  107. package/fesm2022/sebgroup-green-angular-src-v-angular-tooltip.mjs.map +1 -1
  108. package/fesm2022/sebgroup-green-angular-v-angular.mjs +3408 -0
  109. package/fesm2022/sebgroup-green-angular-v-angular.mjs.map +1 -0
  110. package/package.json +8 -2
  111. package/src/v-angular/base-control-value-accessor/base-control-value-accessor.component.d.ts +3 -3
  112. package/src/v-angular/button/button.component.d.ts +2 -2
  113. package/src/v-angular/character-countdown/character-countdown.directive.d.ts +1 -1
  114. package/src/v-angular/core/core.globals.d.ts +1 -1
  115. package/src/v-angular/dropdown/typeahead/index.d.ts +3 -0
  116. package/src/v-angular/dropdown/typeahead/typeahead.directive.d.ts +76 -0
  117. package/src/v-angular/input/input.component.d.ts +3 -3
  118. package/src/v-angular/input-mask/input-mask-format.pipe.d.ts +1 -1
  119. package/src/v-angular/input-mask/input-mask.directive.d.ts +2 -2
  120. package/src/v-angular/modal/dialog/dialog.component.d.ts +5 -5
  121. package/src/v-angular/modal/fold-out/fold-out.directive.d.ts +1 -1
  122. package/src/v-angular/modal/modal.globals.d.ts +1 -1
  123. package/src/v-angular/modal/slide-out/slide-out.component.d.ts +6 -6
  124. package/src/v-angular/textarea/textarea.module.d.ts +2 -1
  125. package/src/v-angular/tooltip/tooltip.directive.d.ts +4 -4
  126. package/v-angular/base-control-value-accessor/base-control-value-accessor.component.d.ts +124 -0
  127. package/v-angular/base-control-value-accessor/base-control-value-accessor.module.d.ts +7 -0
  128. package/v-angular/base-control-value-accessor/index.d.ts +2 -0
  129. package/v-angular/breadcrumbs/breadcrumbs.component.d.ts +18 -0
  130. package/v-angular/breadcrumbs/breadcrumbs.module.d.ts +10 -0
  131. package/v-angular/breadcrumbs/index.d.ts +2 -0
  132. package/v-angular/button/button.component.d.ts +62 -0
  133. package/v-angular/button/button.module.d.ts +10 -0
  134. package/v-angular/button/index.d.ts +2 -0
  135. package/v-angular/card/card.component.d.ts +5 -0
  136. package/v-angular/card/card.module.d.ts +8 -0
  137. package/v-angular/card/index.d.ts +2 -0
  138. package/v-angular/character-countdown/character-countdown.directive.d.ts +17 -0
  139. package/v-angular/character-countdown/character-countdown.module.d.ts +8 -0
  140. package/v-angular/character-countdown/index.d.ts +2 -0
  141. package/v-angular/checkbox/checkbox.component.d.ts +27 -0
  142. package/v-angular/checkbox/checkbox.module.d.ts +9 -0
  143. package/v-angular/checkbox/index.d.ts +2 -0
  144. package/v-angular/dropdown/dropdown-list/dropdown-list.component.d.ts +89 -0
  145. package/v-angular/dropdown/dropdown-list/index.d.ts +1 -0
  146. package/v-angular/dropdown/dropdown.component.d.ts +99 -0
  147. package/v-angular/dropdown/dropdown.module.d.ts +12 -0
  148. package/v-angular/dropdown/index.d.ts +5 -0
  149. package/v-angular/dropdown/typeahead/index.d.ts +5 -0
  150. package/v-angular/dropdown/typeahead/typeahead-dropdown-list/typeahead-dropdown-list.component.d.ts +34 -0
  151. package/v-angular/dropdown/typeahead/typeahead-highlight/typeahead-highlight.component.d.ts +34 -0
  152. package/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.d.ts +59 -0
  153. package/v-angular/dropdown/typeahead/typeahead.directive.d.ts +76 -0
  154. package/v-angular/dropdown/typeahead/typeahead.module.d.ts +11 -0
  155. package/v-angular/i18n/i18n.module.d.ts +15 -0
  156. package/v-angular/i18n/i18n.test.module.d.ts +27 -0
  157. package/v-angular/i18n/index.d.ts +2 -0
  158. package/v-angular/index.d.ts +17 -0
  159. package/v-angular/info-circle/index.d.ts +2 -0
  160. package/v-angular/info-circle/info-circle.component.d.ts +16 -0
  161. package/v-angular/info-circle/info-circle.module.d.ts +10 -0
  162. package/v-angular/input/index.d.ts +2 -0
  163. package/v-angular/input/input.component.d.ts +90 -0
  164. package/v-angular/input/input.module.d.ts +11 -0
  165. package/v-angular/input-mask/config.d.ts +6 -0
  166. package/v-angular/input-mask/constants.d.ts +2 -0
  167. package/v-angular/input-mask/index.d.ts +5 -0
  168. package/v-angular/input-mask/input-mask-format.pipe.d.ts +8 -0
  169. package/v-angular/input-mask/input-mask.directive.d.ts +39 -0
  170. package/v-angular/input-mask/input-mask.module.d.ts +11 -0
  171. package/v-angular/input-mask/input-mask.types.d.ts +20 -0
  172. package/v-angular/modal/dialog/dialog.component.d.ts +55 -0
  173. package/v-angular/modal/fold-out/fold-out.component.d.ts +24 -0
  174. package/v-angular/modal/fold-out/fold-out.directive.d.ts +6 -0
  175. package/v-angular/modal/index.d.ts +5 -0
  176. package/v-angular/modal/modal.globals.d.ts +13 -0
  177. package/v-angular/modal/modal.module.d.ts +13 -0
  178. package/v-angular/modal/modal.types.d.ts +5 -0
  179. package/v-angular/modal/slide-out/slide-out.component.d.ts +76 -0
  180. package/v-angular/radio/index.d.ts +2 -0
  181. package/v-angular/radio/radio.component.d.ts +48 -0
  182. package/v-angular/radio/radio.module.d.ts +9 -0
  183. package/v-angular/textarea/index.d.ts +2 -0
  184. package/v-angular/textarea/textarea.component.d.ts +44 -0
  185. package/v-angular/textarea/textarea.module.d.ts +9 -0
  186. package/v-angular/tooltip/index.d.ts +2 -0
  187. package/v-angular/tooltip/tooltip.directive.d.ts +106 -0
  188. package/v-angular/tooltip/tooltip.module.d.ts +8 -0
  189. package/v-angular/v-angular.module.d.ts +23 -0
@@ -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":"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","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":"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","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>&lt;c-dialog&gt; ...children &lt;/c-dialog&gt;</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=