@rosoftlab/core 0.0.100

Sign up to get free protection for your applications and to get access to all the features.
Files changed (90) hide show
  1. package/README.md +6 -0
  2. package/esm2020/lib/base-components/base-form-edit.component.mjs +242 -0
  3. package/esm2020/lib/base-components/field-error-display/field-error-display.component.mjs +15 -0
  4. package/esm2020/lib/base-components/generic-table/generic-table.component.mjs +431 -0
  5. package/esm2020/lib/base-components/index.mjs +5 -0
  6. package/esm2020/lib/base-components/page-not-found/page-not-found.component.mjs +15 -0
  7. package/esm2020/lib/base-components/searchable-dropdown/searchable-dropdown.component.mjs +231 -0
  8. package/esm2020/lib/base-components/under-construction/under-construction.component.mjs +12 -0
  9. package/esm2020/lib/constants/symbols.mjs +3 -0
  10. package/esm2020/lib/converters/date/date.converter.mjs +12 -0
  11. package/esm2020/lib/decorators/attribute.decorator.mjs +85 -0
  12. package/esm2020/lib/decorators/base-datastore-config.decorator.mjs +7 -0
  13. package/esm2020/lib/decorators/base-model-config.decorator.mjs +10 -0
  14. package/esm2020/lib/decorators/custom.type.decorator.mjs +11 -0
  15. package/esm2020/lib/decorators/grid-layout.decorator.mjs +23 -0
  16. package/esm2020/lib/directives/translated-content.directive.mjs +96 -0
  17. package/esm2020/lib/directives/translated-element.directive.mjs +20 -0
  18. package/esm2020/lib/index.mjs +29 -0
  19. package/esm2020/lib/interfaces/attribute-decorator-options.interface.mjs +2 -0
  20. package/esm2020/lib/interfaces/datastore-config.interface.mjs +2 -0
  21. package/esm2020/lib/interfaces/model-config.interface.mjs +2 -0
  22. package/esm2020/lib/interfaces/overrides.interface.mjs +2 -0
  23. package/esm2020/lib/interfaces/property-converter.interface.mjs +2 -0
  24. package/esm2020/lib/material.mjs +94 -0
  25. package/esm2020/lib/models/base-meta.model.mjs +7 -0
  26. package/esm2020/lib/models/base-query-data.mjs +13 -0
  27. package/esm2020/lib/models/base.model.mjs +171 -0
  28. package/esm2020/lib/models/error-response.model.mjs +9 -0
  29. package/esm2020/lib/models/grid-layout-format.enum.mjs +15 -0
  30. package/esm2020/lib/models/grid-layout.mjs +18 -0
  31. package/esm2020/lib/models/rule.mjs +6 -0
  32. package/esm2020/lib/module.mjs +55 -0
  33. package/esm2020/lib/pipes/Nl2brPipe.pipe.mjs +21 -0
  34. package/esm2020/lib/pipes/input-error.pipe.mjs +37 -0
  35. package/esm2020/lib/providers.mjs +10 -0
  36. package/esm2020/lib/services/base-datastore.service.mjs +333 -0
  37. package/esm2020/lib/services/base.service.mjs +95 -0
  38. package/esm2020/lib/services/dialog.service.mjs +150 -0
  39. package/esm2020/lib/services/grid-layout.service.mjs +21 -0
  40. package/esm2020/lib/services/index.mjs +5 -0
  41. package/esm2020/lib/validators/pattern-validator.mjs +15 -0
  42. package/esm2020/public-api.mjs +12 -0
  43. package/esm2020/rosoftlab-core.mjs +5 -0
  44. package/fesm2015/rosoftlab-core.mjs +2217 -0
  45. package/fesm2015/rosoftlab-core.mjs.map +1 -0
  46. package/fesm2020/rosoftlab-core.mjs +2210 -0
  47. package/fesm2020/rosoftlab-core.mjs.map +1 -0
  48. package/lib/base-components/base-form-edit.component.d.ts +58 -0
  49. package/lib/base-components/field-error-display/field-error-display.component.d.ts +7 -0
  50. package/lib/base-components/generic-table/generic-table.component.d.ts +93 -0
  51. package/lib/base-components/index.d.ts +4 -0
  52. package/lib/base-components/page-not-found/page-not-found.component.d.ts +8 -0
  53. package/lib/base-components/searchable-dropdown/searchable-dropdown.component.d.ts +62 -0
  54. package/lib/base-components/under-construction/under-construction.component.d.ts +6 -0
  55. package/lib/constants/symbols.d.ts +1 -0
  56. package/lib/converters/date/date.converter.d.ts +5 -0
  57. package/lib/decorators/attribute.decorator.d.ts +2 -0
  58. package/lib/decorators/base-datastore-config.decorator.d.ts +1 -0
  59. package/lib/decorators/base-model-config.decorator.d.ts +1 -0
  60. package/lib/decorators/custom.type.decorator.d.ts +1 -0
  61. package/lib/decorators/grid-layout.decorator.d.ts +2 -0
  62. package/lib/directives/translated-content.directive.d.ts +21 -0
  63. package/lib/directives/translated-element.directive.d.ts +10 -0
  64. package/lib/index.d.ts +27 -0
  65. package/lib/interfaces/attribute-decorator-options.interface.d.ts +8 -0
  66. package/lib/interfaces/datastore-config.interface.d.ts +8 -0
  67. package/lib/interfaces/model-config.interface.d.ts +8 -0
  68. package/lib/interfaces/overrides.interface.d.ts +5 -0
  69. package/lib/interfaces/property-converter.interface.d.ts +4 -0
  70. package/lib/material.d.ts +42 -0
  71. package/lib/models/base-meta.model.d.ts +5 -0
  72. package/lib/models/base-query-data.d.ts +7 -0
  73. package/lib/models/base.model.d.ts +33 -0
  74. package/lib/models/error-response.model.d.ts +17 -0
  75. package/lib/models/grid-layout-format.enum.d.ts +12 -0
  76. package/lib/models/grid-layout.d.ts +13 -0
  77. package/lib/models/rule.d.ts +5 -0
  78. package/lib/module.d.ts +14 -0
  79. package/lib/pipes/Nl2brPipe.pipe.d.ts +7 -0
  80. package/lib/pipes/input-error.pipe.d.ts +10 -0
  81. package/lib/providers.d.ts +2 -0
  82. package/lib/services/base-datastore.service.d.ts +45 -0
  83. package/lib/services/base.service.d.ts +26 -0
  84. package/lib/services/dialog.service.d.ts +21 -0
  85. package/lib/services/grid-layout.service.d.ts +9 -0
  86. package/lib/services/index.d.ts +4 -0
  87. package/lib/validators/pattern-validator.d.ts +4 -0
  88. package/package.json +53 -0
  89. package/public-api.d.ts +7 -0
  90. package/rosoftlab-core.d.ts +5 -0
@@ -0,0 +1,231 @@
1
+ import { coerceBooleanProperty } from '@angular/cdk/coercion';
2
+ import { Component, EventEmitter, Input, Optional, Output, Self } from '@angular/core';
3
+ import { FormControl } from '@angular/forms';
4
+ import { Subject } from 'rxjs';
5
+ import { debounceTime, filter, finalize, switchMap, takeUntil, tap } from 'rxjs/operators';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/cdk/a11y";
8
+ import * as i2 from "@angular/forms";
9
+ import * as i3 from "@angular/material/form-field";
10
+ import * as i4 from "@angular/material/button";
11
+ import * as i5 from "@angular/material/icon";
12
+ import * as i6 from "@angular/material/autocomplete";
13
+ import * as i7 from "@angular/material/core";
14
+ import * as i8 from "@angular/material/input";
15
+ import * as i9 from "@angular/common";
16
+ import * as i10 from "@ngx-translate/core";
17
+ export class SearchableDropdownComponent {
18
+ constructor(_focusMonitor, _elementRef, ngControl) {
19
+ this._focusMonitor = _focusMonitor;
20
+ this._elementRef = _elementRef;
21
+ this.ngControl = ngControl;
22
+ this.onChange = (_) => { };
23
+ this.onTouched = () => { };
24
+ this.stateChanges = new Subject();
25
+ this.focused = false;
26
+ this.errorState = false;
27
+ this.id = `rsl-searchable-dropdown-${SearchableDropdownComponent.nextId++}`;
28
+ // label for the search dropdown
29
+ this.label = '';
30
+ //Fields for sorting the API data
31
+ this.sortFields = '';
32
+ // The value used to populate the control value
33
+ this.valueField = 'id';
34
+ // The minumum char for search
35
+ this.minLengthTerm = 3;
36
+ this.modelSelected = new EventEmitter();
37
+ this._required = false;
38
+ this._disabled = false;
39
+ this.showSerach = true;
40
+ this.isLoading = false;
41
+ this.options = [];
42
+ this.filteredOptions = [];
43
+ /** Subject that emits when the component has been destroyed. */
44
+ this._onDestroy = new Subject();
45
+ this.searchControl = new FormControl('');
46
+ _focusMonitor.monitor(_elementRef, true).subscribe(origin => {
47
+ if (this.focused && !origin) {
48
+ this.onTouched();
49
+ }
50
+ this.focused = !!origin;
51
+ this.stateChanges.next();
52
+ });
53
+ if (this.ngControl) {
54
+ this.ngControl.valueAccessor = this;
55
+ }
56
+ }
57
+ get empty() {
58
+ return !this.value;
59
+ }
60
+ get required() { return this._required; }
61
+ set required(value) {
62
+ this._required = coerceBooleanProperty(value);
63
+ this.stateChanges.next();
64
+ }
65
+ get disabled() { return this._disabled; }
66
+ set disabled(value) {
67
+ this._disabled = coerceBooleanProperty(value);
68
+ this.stateChanges.next();
69
+ }
70
+ get value() {
71
+ return this._value;
72
+ }
73
+ set value(value) {
74
+ if (value)
75
+ if (this._value !== value) {
76
+ this.loadModel(value);
77
+ }
78
+ this._value = value;
79
+ this.stateChanges.next();
80
+ }
81
+ ngOnInit() {
82
+ if (!this.displayFields)
83
+ this.displayFields = this.searchFields;
84
+ this.loadModel(this.ngControl.value);
85
+ if (this.preloadElementsCount) {
86
+ this.serviceRef.getAll(1, this.preloadElementsCount).subscribe({
87
+ next: (data) => {
88
+ this.showSerach = false;
89
+ this.options = data.getModels();
90
+ this.filteredOptions = this.options;
91
+ }
92
+ });
93
+ }
94
+ this.searchControl.valueChanges
95
+ .pipe(filter((res) => {
96
+ const result = res !== null && res.length >= this.minLengthTerm;
97
+ if (!result) {
98
+ this.options = [];
99
+ this.filteredOptions = [];
100
+ }
101
+ this.showSerach = !result;
102
+ return result;
103
+ }), debounceTime(300), tap(() => {
104
+ this.filteredOptions = [];
105
+ this.isLoading = true;
106
+ }), switchMap((value) => this.serviceRef
107
+ .getAll(1, 10, this.sortFields, `${this.searchFields.replace(',', '|')}@=*${value}`)
108
+ .pipe(finalize(() => {
109
+ this.isLoading = false;
110
+ }))), takeUntil(this._onDestroy))
111
+ .subscribe((options) => {
112
+ this.filteredOptions = options.getModels();
113
+ });
114
+ }
115
+ onSelected($event) {
116
+ const value = $event.option.value;
117
+ this.searchControl.setValue(value);
118
+ this.modelSelected.emit(value);
119
+ if (this.valueField in value) {
120
+ this.setControlValue(value[this.valueField]);
121
+ }
122
+ }
123
+ setControlValue(value) {
124
+ this.value = value;
125
+ this.ngControl.control.setValue(value);
126
+ }
127
+ displayWith(value) {
128
+ return this.getConcatedFields(value);
129
+ }
130
+ clearSelection() {
131
+ this.modelSelected.emit(null);
132
+ this.setControlValue(null);
133
+ this.searchControl.setValue(null);
134
+ this.filteredOptions = this.options;
135
+ }
136
+ getConcatedFields(option) {
137
+ if (!option)
138
+ return null;
139
+ const filtersArr = this.displayFields.split(',');
140
+ return filtersArr.reduce((acc, cv) => {
141
+ return acc.concat(option[cv] + ' ');
142
+ }, '');
143
+ }
144
+ ngOnDestroy() {
145
+ this.searchControl = null;
146
+ this.stateChanges.complete();
147
+ this._focusMonitor.stopMonitoring(this._elementRef);
148
+ this._onDestroy.next();
149
+ this._onDestroy.complete();
150
+ }
151
+ getSearchText() {
152
+ return `Enter ${this.minLengthTerm} characters to start search`;
153
+ }
154
+ writeValue(model) {
155
+ console.log(model);
156
+ this.value = model;
157
+ }
158
+ registerOnChange(fn) {
159
+ this.onChange = fn;
160
+ }
161
+ registerOnTouched(fn) {
162
+ this.onTouched = fn;
163
+ }
164
+ setDisabledState(isDisabled) {
165
+ this.disabled = isDisabled;
166
+ }
167
+ loadModel(id) {
168
+ //Load the model from API to display the value
169
+ let needLoad = false;
170
+ if (id) {
171
+ if (!this.searchControl.value)
172
+ needLoad = true;
173
+ else {
174
+ if (this.searchControl.value[this.valueField] !== id) {
175
+ needLoad = true;
176
+ }
177
+ }
178
+ }
179
+ if (needLoad) {
180
+ this.serviceRef.get(id).subscribe({
181
+ next: (model) => {
182
+ this.options.push(model);
183
+ this.filteredOptions = this.options;
184
+ this.searchControl.setValue(model);
185
+ }
186
+ });
187
+ }
188
+ }
189
+ }
190
+ SearchableDropdownComponent.nextId = 0;
191
+ SearchableDropdownComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: SearchableDropdownComponent, deps: [{ token: i1.FocusMonitor }, { token: i0.ElementRef }, { token: i2.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component });
192
+ SearchableDropdownComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.0", type: SearchableDropdownComponent, selector: "rsl-searchable-dropdown", inputs: { label: "label", fControlName: "fControlName", serviceRef: "serviceRef", searchFields: "searchFields", displayFields: "displayFields", sortFields: "sortFields", valueField: "valueField", minLengthTerm: "minLengthTerm", preloadElementsCount: "preloadElementsCount", required: "required", disabled: "disabled", value: "value" }, outputs: { modelSelected: "modelSelected" }, host: { properties: { "class.example-floating": "shouldLabelFloat", "id": "id", "attr.aria-describedby": "describedBy" } }, ngImport: i0, template: "<div class=\"form-group\">\r\n <label>{{label | translate}}</label>\r\n <mat-form-field appearance=\"outline\" fxFlex>\r\n <input matInput [matAutocomplete]=\"auto\" [formControl]=\"searchControl\">\r\n <button *ngIf=\"searchControl.value\" matSuffix mat-icon-button aria-label=\"Clear\" (click)=\"clearSelection()\">\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"onSelected($event)\"\r\n [displayWith]=\"displayWith.bind(this)\">\r\n <mat-option *ngIf=\"isLoading\">Loading...</mat-option>\r\n <mat-option *ngIf=\"!isLoading && filteredOptions.length == 0 && !showSerach\" disabled>Not found</mat-option>\r\n <mat-option *ngIf=\"showSerach\" disabled>{{getSearchText()}}</mat-option>\r\n <ng-container *ngIf=\"!isLoading && filteredOptions.length > 0\">\r\n <mat-option *ngFor=\"let option of filteredOptions\" [value]=\"option\">\r\n <span><b>{{getConcatedFields(option)}}</b></span>\r\n </mat-option>\r\n </ng-container>\r\n\r\n </mat-autocomplete>\r\n </mat-form-field>\r\n</div>", components: [{ type: i3.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i6.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple"], exportAs: ["matAutocomplete"] }, { type: i7.MatOption, selector: "mat-option", exportAs: ["matOption"] }], directives: [{ type: i8.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { type: i6.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.MatSuffix, selector: "[matSuffix]" }, { type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "translate": i10.TranslatePipe } });
193
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: SearchableDropdownComponent, decorators: [{
194
+ type: Component,
195
+ args: [{ selector: 'rsl-searchable-dropdown', host: {
196
+ '[class.example-floating]': 'shouldLabelFloat',
197
+ '[id]': 'id',
198
+ '[attr.aria-describedby]': 'describedBy',
199
+ }, template: "<div class=\"form-group\">\r\n <label>{{label | translate}}</label>\r\n <mat-form-field appearance=\"outline\" fxFlex>\r\n <input matInput [matAutocomplete]=\"auto\" [formControl]=\"searchControl\">\r\n <button *ngIf=\"searchControl.value\" matSuffix mat-icon-button aria-label=\"Clear\" (click)=\"clearSelection()\">\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"onSelected($event)\"\r\n [displayWith]=\"displayWith.bind(this)\">\r\n <mat-option *ngIf=\"isLoading\">Loading...</mat-option>\r\n <mat-option *ngIf=\"!isLoading && filteredOptions.length == 0 && !showSerach\" disabled>Not found</mat-option>\r\n <mat-option *ngIf=\"showSerach\" disabled>{{getSearchText()}}</mat-option>\r\n <ng-container *ngIf=\"!isLoading && filteredOptions.length > 0\">\r\n <mat-option *ngFor=\"let option of filteredOptions\" [value]=\"option\">\r\n <span><b>{{getConcatedFields(option)}}</b></span>\r\n </mat-option>\r\n </ng-container>\r\n\r\n </mat-autocomplete>\r\n </mat-form-field>\r\n</div>" }]
200
+ }], ctorParameters: function () { return [{ type: i1.FocusMonitor }, { type: i0.ElementRef }, { type: i2.NgControl, decorators: [{
201
+ type: Optional
202
+ }, {
203
+ type: Self
204
+ }] }]; }, propDecorators: { label: [{
205
+ type: Input
206
+ }], fControlName: [{
207
+ type: Input
208
+ }], serviceRef: [{
209
+ type: Input
210
+ }], searchFields: [{
211
+ type: Input
212
+ }], displayFields: [{
213
+ type: Input
214
+ }], sortFields: [{
215
+ type: Input
216
+ }], valueField: [{
217
+ type: Input
218
+ }], minLengthTerm: [{
219
+ type: Input
220
+ }], preloadElementsCount: [{
221
+ type: Input
222
+ }], modelSelected: [{
223
+ type: Output
224
+ }], required: [{
225
+ type: Input
226
+ }], disabled: [{
227
+ type: Input
228
+ }], value: [{
229
+ type: Input
230
+ }] } });
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"searchable-dropdown.component.js","sourceRoot":"","sources":["../../../../../../../projects/rosoftlab/core/src/lib/base-components/searchable-dropdown/searchable-dropdown.component.ts","../../../../../../../projects/rosoftlab/core/src/lib/base-components/searchable-dropdown/searchable-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { FocusMonitor } from '@angular/cdk/a11y';\r\nimport { coerceBooleanProperty } from '@angular/cdk/coercion';\r\nimport { Component, ElementRef, EventEmitter, Input, OnDestroy, OnInit, Optional, Output, Self } from '@angular/core';\r\nimport { AbstractControl, ControlValueAccessor, FormControl, NgControl } from '@angular/forms';\r\nimport { Subject } from 'rxjs';\r\nimport { debounceTime, filter, finalize, switchMap, takeUntil, tap } from 'rxjs/operators';\r\nimport { BaseQueryData } from '../../models/base-query-data';\r\nimport { BaseModel } from '../../models/base.model';\r\nimport { BaseService } from '../../services/base.service';\r\n\r\n@Component({\r\n  selector: 'rsl-searchable-dropdown',\r\n  templateUrl: './searchable-dropdown.component.html',\r\n  // styleUrls: ['./searchable-dropdown.component.scss'],\r\n  host: {\r\n    '[class.example-floating]': 'shouldLabelFloat',\r\n    '[id]': 'id',\r\n    '[attr.aria-describedby]': 'describedBy',\r\n  }\r\n})\r\nexport class SearchableDropdownComponent<T extends BaseModel> implements OnInit, OnDestroy, ControlValueAccessor {\r\n  static nextId = 0;\r\n  onChange = (_: any) => { };\r\n  onTouched = () => { };\r\n  stateChanges = new Subject<void>();\r\n  focused = false;\r\n  errorState = false;\r\n  id = `rsl-searchable-dropdown-${SearchableDropdownComponent.nextId++}`;\r\n\r\n  // label for the search dropdown\r\n  @Input() label: string = '';\r\n\r\n  //Name of the form control\r\n  @Input() fControlName: string;\r\n\r\n  //Reference of the service\r\n  @Input() serviceRef: BaseService<T>;\r\n\r\n  //Fields for filtering\r\n  @Input() searchFields: string;\r\n\r\n  //Fields for display \r\n  @Input() displayFields: string;\r\n\r\n  //Fields for sorting the API data\r\n  @Input() sortFields: string = '';\r\n\r\n  // The value used to populate the control value\r\n  @Input() valueField: string = 'id';\r\n\r\n  // The minumum char for search\r\n  @Input() minLengthTerm: number = 3;\r\n\r\n  // The preload x elements\r\n  @Input() preloadElementsCount: number;\r\n\r\n  @Output() modelSelected: EventEmitter<BaseModel> = new EventEmitter();\r\n\r\n\r\n  get empty() {\r\n    return !this.value;\r\n  }\r\n\r\n\r\n  @Input()\r\n  get required(): boolean { return this._required; }\r\n  set required(value: boolean) {\r\n    this._required = coerceBooleanProperty(value);\r\n    this.stateChanges.next();\r\n  }\r\n  private _required = false;\r\n\r\n  @Input()\r\n  get disabled(): boolean { return this._disabled; }\r\n  set disabled(value: boolean) {\r\n    this._disabled = coerceBooleanProperty(value);\r\n    this.stateChanges.next();\r\n  }\r\n  private _disabled = false;\r\n\r\n\r\n\r\n  @Input()\r\n  get value(): any | null {\r\n    return this._value;\r\n  }\r\n  set value(value: any | null) {\r\n    if (value)\r\n      if (this._value !== value) {\r\n        this.loadModel(value);\r\n      }\r\n    this._value = value;\r\n    this.stateChanges.next();\r\n  }\r\n  private _value: any;\r\n  // Search form control\r\n  searchControl: AbstractControl;\r\n\r\n  showSerach = true;\r\n  isLoading = false;\r\n  options = [];\r\n  filteredOptions = [];\r\n  /** Subject that emits when the component has been destroyed. */\r\n  protected _onDestroy = new Subject<void>();\r\n\r\n  constructor(\r\n    private _focusMonitor: FocusMonitor,\r\n    private _elementRef: ElementRef<HTMLElement>,\r\n    @Optional() @Self() public ngControl: NgControl\r\n  ) {\r\n\r\n    this.searchControl = new FormControl('');\r\n    _focusMonitor.monitor(_elementRef, true).subscribe(origin => {\r\n      if (this.focused && !origin) {\r\n        this.onTouched();\r\n      }\r\n      this.focused = !!origin;\r\n      this.stateChanges.next();\r\n    });\r\n\r\n    if (this.ngControl) {\r\n      this.ngControl.valueAccessor = this;\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void {\r\n\r\n    if (!this.displayFields)\r\n      this.displayFields = this.searchFields;\r\n\r\n    this.loadModel(this.ngControl.value)\r\n\r\n    if (this.preloadElementsCount) {\r\n      this.serviceRef.getAll(1, this.preloadElementsCount).subscribe({\r\n        next: (data: BaseQueryData<T>) => {\r\n          this.showSerach = false;\r\n          this.options = data.getModels();\r\n          this.filteredOptions = this.options;\r\n        }\r\n      })\r\n    }\r\n\r\n    this.searchControl.valueChanges\r\n      .pipe(\r\n        filter((res) => {\r\n          const result = res !== null && res.length >= this.minLengthTerm;\r\n          if (!result) {\r\n            this.options = [];\r\n            this.filteredOptions = [];\r\n          }\r\n          this.showSerach = !result;\r\n          return result;\r\n        }),\r\n        debounceTime(300),\r\n        tap(() => {\r\n          this.filteredOptions = [];\r\n          this.isLoading = true;\r\n        }),\r\n        switchMap((value: string) =>\r\n          this.serviceRef\r\n            .getAll(1, 10, this.sortFields, `${this.searchFields.replace(',', '|')}@=*${value}`)\r\n            .pipe(\r\n              finalize(() => {\r\n                this.isLoading = false;\r\n              })\r\n            )\r\n        ),\r\n        takeUntil(this._onDestroy)\r\n      )\r\n      .subscribe((options: any) => {\r\n        this.filteredOptions = options.getModels();\r\n      });\r\n  }\r\n\r\n  onSelected($event) {\r\n    const value = $event.option.value;\r\n    this.searchControl.setValue(value);\r\n    this.modelSelected.emit(value);\r\n    if (this.valueField in value) {\r\n      this.setControlValue(value[this.valueField]);\r\n    }\r\n\r\n  }\r\n  setControlValue(value: any) {\r\n    this.value = value;\r\n    this.ngControl.control.setValue(value);\r\n  }\r\n  displayWith(value: any) {\r\n    return this.getConcatedFields(value);\r\n  }\r\n\r\n  clearSelection() {\r\n    this.modelSelected.emit(null);\r\n    this.setControlValue(null);\r\n    this.searchControl.setValue(null);\r\n    this.filteredOptions = this.options;\r\n  }\r\n\r\n  getConcatedFields(option): string {\r\n    if (!option) return null;\r\n    const filtersArr = this.displayFields.split(',');\r\n    return filtersArr.reduce((acc, cv) => {\r\n      return acc.concat(option[cv] + ' ');\r\n    }, '');\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    this.searchControl = null;\r\n    this.stateChanges.complete();\r\n    this._focusMonitor.stopMonitoring(this._elementRef);\r\n    this._onDestroy.next();\r\n    this._onDestroy.complete();\r\n  }\r\n  getSearchText(): string {\r\n    return `Enter ${this.minLengthTerm} characters to start search`;\r\n\r\n  }\r\n  writeValue(model: T | null): void {\r\n    console.log(model);\r\n    this.value = model;\r\n  }\r\n\r\n  registerOnChange(fn: any): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: any): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.disabled = isDisabled;\r\n  }\r\n  private loadModel(id: string) {\r\n    //Load the model from API to display the value\r\n    let needLoad = false;\r\n    if (id) {\r\n      if (!this.searchControl.value)\r\n        needLoad = true\r\n      else {\r\n        if (this.searchControl.value[this.valueField] !== id) {\r\n          needLoad = true\r\n        }\r\n      }\r\n    }\r\n    if (needLoad) {\r\n      this.serviceRef.get(id).subscribe({\r\n        next: (model: T) => {\r\n          this.options.push(model);\r\n          this.filteredOptions = this.options;\r\n          this.searchControl.setValue(model);\r\n        }\r\n      })\r\n    }\r\n  }\r\n}\r\n","<div class=\"form-group\">\r\n  <label>{{label | translate}}</label>\r\n  <mat-form-field appearance=\"outline\" fxFlex>\r\n    <input matInput [matAutocomplete]=\"auto\" [formControl]=\"searchControl\">\r\n    <button *ngIf=\"searchControl.value\" matSuffix mat-icon-button aria-label=\"Clear\" (click)=\"clearSelection()\">\r\n      <mat-icon>close</mat-icon>\r\n    </button>\r\n    <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"onSelected($event)\"\r\n      [displayWith]=\"displayWith.bind(this)\">\r\n      <mat-option *ngIf=\"isLoading\">Loading...</mat-option>\r\n      <mat-option *ngIf=\"!isLoading && filteredOptions.length == 0 && !showSerach\" disabled>Not found</mat-option>\r\n      <mat-option *ngIf=\"showSerach\" disabled>{{getSearchText()}}</mat-option>\r\n      <ng-container *ngIf=\"!isLoading && filteredOptions.length > 0\">\r\n        <mat-option *ngFor=\"let option of filteredOptions\" [value]=\"option\">\r\n          <span><b>{{getConcatedFields(option)}}</b></span>\r\n        </mat-option>\r\n      </ng-container>\r\n\r\n    </mat-autocomplete>\r\n  </mat-form-field>\r\n</div>"]}
@@ -0,0 +1,12 @@
1
+ import { Component } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class UnderConstructionComponent {
4
+ constructor() { }
5
+ }
6
+ UnderConstructionComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: UnderConstructionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
7
+ UnderConstructionComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.0", type: UnderConstructionComponent, selector: "rsl-under-construction", ngImport: i0, template: "<div class=\"forms-view-container\">\n <div class=\"under-construction-view-container\">\n Page under construction\n </div>\n</div>", styles: [".under-construction-view-container{height:calc(100vh - 250px);margin:60px 0 0;padding:0;display:flex;justify-content:center;align-items:center;box-sizing:border-box;font-size:1.8em;line-height:1.3em;font-weight:300;color:#58585b;text-align:center}\n"] });
8
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: UnderConstructionComponent, decorators: [{
9
+ type: Component,
10
+ args: [{ selector: 'rsl-under-construction', template: "<div class=\"forms-view-container\">\n <div class=\"under-construction-view-container\">\n Page under construction\n </div>\n</div>", styles: [".under-construction-view-container{height:calc(100vh - 250px);margin:60px 0 0;padding:0;display:flex;justify-content:center;align-items:center;box-sizing:border-box;font-size:1.8em;line-height:1.3em;font-weight:300;color:#58585b;text-align:center}\n"] }]
11
+ }], ctorParameters: function () { return []; } });
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidW5kZXItY29uc3RydWN0aW9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jvc29mdGxhYi9jb3JlL3NyYy9saWIvYmFzZS1jb21wb25lbnRzL3VuZGVyLWNvbnN0cnVjdGlvbi91bmRlci1jb25zdHJ1Y3Rpb24uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcm9zb2Z0bGFiL2NvcmUvc3JjL2xpYi9iYXNlLWNvbXBvbmVudHMvdW5kZXItY29uc3RydWN0aW9uL3VuZGVyLWNvbnN0cnVjdGlvbi5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU8xQyxNQUFNLE9BQU8sMEJBQTBCO0lBRXJDLGdCQUFnQixDQUFDOzt1SEFGTiwwQkFBMEI7MkdBQTFCLDBCQUEwQiw4RENQdkMsMElBSU07MkZER08sMEJBQTBCO2tCQUx0QyxTQUFTOytCQUNFLHdCQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICdyc2wtdW5kZXItY29uc3RydWN0aW9uJyxcclxuICB0ZW1wbGF0ZVVybDogJy4vdW5kZXItY29uc3RydWN0aW9uLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybHM6IFsnLi91bmRlci1jb25zdHJ1Y3Rpb24uY29tcG9uZW50LnNjc3MnXVxyXG59KVxyXG5leHBvcnQgY2xhc3MgVW5kZXJDb25zdHJ1Y3Rpb25Db21wb25lbnQge1xyXG5cclxuICBjb25zdHJ1Y3RvcigpIHsgfVxyXG59XHJcbiIsIjxkaXYgY2xhc3M9XCJmb3Jtcy12aWV3LWNvbnRhaW5lclwiPlxuICA8ZGl2IGNsYXNzPVwidW5kZXItY29uc3RydWN0aW9uLXZpZXctY29udGFpbmVyXCI+XG4gICAgUGFnZSB1bmRlciBjb25zdHJ1Y3Rpb25cbiAgPC9kaXY+XG48L2Rpdj4iXX0=
@@ -0,0 +1,3 @@
1
+ // tslint:disable-next-line:variable-name
2
+ export const AttributeMetadata = Symbol('AttributeMetadata');
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3ltYm9scy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jvc29mdGxhYi9jb3JlL3NyYy9saWIvY29uc3RhbnRzL3N5bWJvbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEseUNBQXlDO0FBQ3pDLE1BQU0sQ0FBQyxNQUFNLGlCQUFpQixHQUFRLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gdHNsaW50OmRpc2FibGUtbmV4dC1saW5lOnZhcmlhYmxlLW5hbWVcbmV4cG9ydCBjb25zdCBBdHRyaWJ1dGVNZXRhZGF0YTogYW55ID0gU3ltYm9sKCdBdHRyaWJ1dGVNZXRhZGF0YScpO1xuIl19
@@ -0,0 +1,12 @@
1
+ import { parseISO } from 'date-fns';
2
+ export class DateConverter {
3
+ mask(value) {
4
+ const d = parseISO(value);
5
+ return d;
6
+ }
7
+ unmask(value) {
8
+ // const result = format(value, 'YYYY-MM-DDTHH:mm:ssZ');
9
+ return value;
10
+ }
11
+ }
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0ZS5jb252ZXJ0ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yb3NvZnRsYWIvY29yZS9zcmMvbGliL2NvbnZlcnRlcnMvZGF0ZS9kYXRlLmNvbnZlcnRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQWlCLFFBQVEsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUduRCxNQUFNLE9BQU8sYUFBYTtJQUN4QixJQUFJLENBQUMsS0FBVTtRQUNiLE1BQU0sQ0FBQyxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMxQixPQUFPLENBQUMsQ0FBQztJQUNYLENBQUM7SUFDRCxNQUFNLENBQUMsS0FBVTtRQUNmLHdEQUF3RDtRQUN4RCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGZvcm1hdCwgcGFyc2UsIHBhcnNlSVNPIH0gZnJvbSAnZGF0ZS1mbnMnO1xyXG5pbXBvcnQgeyBQcm9wZXJ0eUNvbnZlcnRlciB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMvcHJvcGVydHktY29udmVydGVyLmludGVyZmFjZSc7XHJcblxyXG5leHBvcnQgY2xhc3MgRGF0ZUNvbnZlcnRlciBpbXBsZW1lbnRzIFByb3BlcnR5Q29udmVydGVyIHtcclxuICBtYXNrKHZhbHVlOiBhbnkpIHtcclxuICAgIGNvbnN0IGQgPSBwYXJzZUlTTyh2YWx1ZSk7XHJcbiAgICByZXR1cm4gZDtcclxuICB9XHJcbiAgdW5tYXNrKHZhbHVlOiBhbnkpIHtcclxuICAgIC8vIGNvbnN0IHJlc3VsdCA9IGZvcm1hdCh2YWx1ZSwgJ1lZWVktTU0tRERUSEg6bW06c3NaJyk7XHJcbiAgICByZXR1cm4gdmFsdWU7XHJcbiAgfVxyXG59XHJcbiJdfQ==
@@ -0,0 +1,85 @@
1
+ import { AttributeMetadata } from '../constants/symbols';
2
+ import { DateConverter } from '../converters/date/date.converter';
3
+ export function Attribute(options = {}) {
4
+ return (target, propertyName) => {
5
+ const converter = (dataType, value, forSerialisation = false) => {
6
+ let attrConverter;
7
+ if (dataType) {
8
+ if (options.converter) {
9
+ attrConverter = options.converter;
10
+ }
11
+ else if (dataType === Date) {
12
+ attrConverter = new DateConverter();
13
+ }
14
+ else {
15
+ const datatype = new dataType();
16
+ if (datatype.mask && datatype.unmask) {
17
+ attrConverter = datatype;
18
+ }
19
+ }
20
+ if (attrConverter) {
21
+ if (!forSerialisation) {
22
+ return attrConverter.mask(value);
23
+ }
24
+ return attrConverter.unmask(value);
25
+ }
26
+ }
27
+ return value;
28
+ };
29
+ const saveAnnotations = () => {
30
+ const metadata = Reflect.getMetadata('Attribute', target) || {};
31
+ metadata[propertyName] = {
32
+ marked: true
33
+ };
34
+ Reflect.defineMetadata('Attribute', metadata, target);
35
+ const mappingMetadata = Reflect.getMetadata('AttributeMapping', target) || {};
36
+ const serializedPropertyName = options.serializedName !== undefined ? options.serializedName : propertyName;
37
+ mappingMetadata[serializedPropertyName] = propertyName;
38
+ Reflect.defineMetadata('AttributeMapping', mappingMetadata, target);
39
+ const requiredMetadata = Reflect.getMetadata('AttributeRequired', target) || {};
40
+ requiredMetadata[serializedPropertyName] = options.required !== undefined ? options.required : false;
41
+ Reflect.defineMetadata('AttributeRequired', requiredMetadata, target);
42
+ const defaultMetadata = Reflect.getMetadata('AttributedefaultValue', target) || {};
43
+ defaultMetadata[serializedPropertyName] = options.defaultValue !== undefined ? options.defaultValue : null;
44
+ Reflect.defineMetadata('AttributedefaultValue', defaultMetadata, target);
45
+ const formSubGroupMetadata = Reflect.getMetadata('AttributeformSubGroup', target) || {};
46
+ formSubGroupMetadata[serializedPropertyName] = options.formSubGroup !== undefined ? options.formSubGroup : null;
47
+ Reflect.defineMetadata('AttributeformSubGroup', formSubGroupMetadata, target);
48
+ };
49
+ const setMetadata = (hasDirtyAttributes, instance, oldValue, newValue, isNew) => {
50
+ const targetType = Reflect.getMetadata('design:type', target, propertyName);
51
+ if (!instance[AttributeMetadata]) {
52
+ instance[AttributeMetadata] = {};
53
+ }
54
+ const propertyHasDirtyAttributes = typeof oldValue === 'undefined' && !isNew ? false : hasDirtyAttributes;
55
+ instance[AttributeMetadata][propertyName] = {
56
+ newValue,
57
+ oldValue,
58
+ serializedName: options.serializedName,
59
+ hasDirtyAttributes: propertyHasDirtyAttributes,
60
+ serialisationValue: converter(targetType, newValue, true)
61
+ };
62
+ };
63
+ const getter = function () {
64
+ return this['_' + propertyName];
65
+ };
66
+ const setter = function (newVal) {
67
+ const targetType = Reflect.getMetadata('design:type', target, propertyName);
68
+ const convertedValue = converter(targetType, newVal);
69
+ if (convertedValue !== this['_' + propertyName]) {
70
+ setMetadata(true, this, this['_' + propertyName], newVal, !this.id);
71
+ this['_' + propertyName] = convertedValue;
72
+ }
73
+ };
74
+ if (delete target[propertyName]) {
75
+ saveAnnotations();
76
+ Object.defineProperty(target, propertyName, {
77
+ get: getter,
78
+ set: setter,
79
+ enumerable: true,
80
+ configurable: true
81
+ });
82
+ }
83
+ };
84
+ }
85
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attribute.decorator.js","sourceRoot":"","sources":["../../../../../../projects/rosoftlab/core/src/lib/decorators/attribute.decorator.ts"],"names":[],"mappings":"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","sourcesContent":["import { } from 'reflect-metadata';\r\nimport { AttributeMetadata } from '../constants/symbols';\r\nimport { DateConverter } from '../converters/date/date.converter';\r\nimport { AttributeDecoratorOptions } from '../interfaces/attribute-decorator-options.interface';\r\n\r\nexport function Attribute(options: AttributeDecoratorOptions = {}): PropertyDecorator {\r\n  return (target: any, propertyName: string) => {\r\n\r\n    const converter = (dataType: any, value: any, forSerialisation = false): any => {\r\n      let attrConverter;\r\n      if (dataType) {\r\n        if (options.converter) {\r\n          attrConverter = options.converter;\r\n        } else if (dataType === Date) {\r\n          attrConverter = new DateConverter();\r\n        } else {\r\n          const datatype = new dataType();\r\n\r\n          if (datatype.mask && datatype.unmask) {\r\n            attrConverter = datatype;\r\n          }\r\n        }\r\n\r\n        if (attrConverter) {\r\n          if (!forSerialisation) {\r\n            return attrConverter.mask(value);\r\n          }\r\n          return attrConverter.unmask(value);\r\n        }\r\n      }\r\n      return value;\r\n    };\r\n\r\n    const saveAnnotations = () => {\r\n      const metadata = Reflect.getMetadata('Attribute', target) || {};\r\n\r\n      metadata[propertyName] = {\r\n        marked: true\r\n      };\r\n\r\n      Reflect.defineMetadata('Attribute', metadata, target);\r\n\r\n      const mappingMetadata = Reflect.getMetadata('AttributeMapping', target) || {};\r\n      const serializedPropertyName = options.serializedName !== undefined ? options.serializedName : propertyName;\r\n      mappingMetadata[serializedPropertyName] = propertyName;\r\n      Reflect.defineMetadata('AttributeMapping', mappingMetadata, target);\r\n\r\n      const requiredMetadata = Reflect.getMetadata('AttributeRequired', target) || {};\r\n      requiredMetadata[serializedPropertyName] = options.required !== undefined ? options.required : false;\r\n      Reflect.defineMetadata('AttributeRequired', requiredMetadata, target);\r\n\r\n      const defaultMetadata = Reflect.getMetadata('AttributedefaultValue', target) || {};\r\n      defaultMetadata[serializedPropertyName] = options.defaultValue !== undefined ? options.defaultValue : null;\r\n      Reflect.defineMetadata('AttributedefaultValue', defaultMetadata, target);\r\n\r\n      const formSubGroupMetadata = Reflect.getMetadata('AttributeformSubGroup', target) || {};\r\n      formSubGroupMetadata[serializedPropertyName] = options.formSubGroup !== undefined ? options.formSubGroup : null;\r\n      Reflect.defineMetadata('AttributeformSubGroup', formSubGroupMetadata, target);\r\n\r\n    };\r\n\r\n    const setMetadata = (\r\n      hasDirtyAttributes: boolean,\r\n      instance: any,\r\n      oldValue: any,\r\n      newValue: any,\r\n      isNew: boolean\r\n    ) => {\r\n      const targetType = Reflect.getMetadata('design:type', target, propertyName);\r\n\r\n      if (!instance[AttributeMetadata]) {\r\n        instance[AttributeMetadata] = {};\r\n      }\r\n\r\n      const propertyHasDirtyAttributes = typeof oldValue === 'undefined' && !isNew ? false : hasDirtyAttributes;\r\n\r\n      instance[AttributeMetadata][propertyName] = {\r\n        newValue,\r\n        oldValue,\r\n        serializedName: options.serializedName,\r\n        hasDirtyAttributes: propertyHasDirtyAttributes,\r\n        serialisationValue: converter(targetType, newValue, true)\r\n      };\r\n    };\r\n\r\n    const getter = function () {\r\n      return this['_' + (propertyName as string)];\r\n    };\r\n\r\n    const setter = function (newVal: any) {\r\n      const targetType = Reflect.getMetadata('design:type', target, propertyName);\r\n      const convertedValue = converter(targetType, newVal);\r\n\r\n      if (convertedValue !== this['_' + (propertyName as string)]) {\r\n        setMetadata(true, this, this['_' + (propertyName as string)], newVal, !this.id);\r\n        this['_' + (propertyName as string)] = convertedValue;\r\n      }\r\n    };\r\n\r\n    if (delete target[propertyName]) {\r\n      saveAnnotations();\r\n      Object.defineProperty(target, propertyName, {\r\n        get: getter,\r\n        set: setter,\r\n        enumerable: true,\r\n        configurable: true\r\n      });\r\n    }\r\n  };\r\n}\r\n"]}
@@ -0,0 +1,7 @@
1
+ export function BaseDatastoreConfig(config = {}) {
2
+ // tslint:disable-next-line:only-arrow-functions
3
+ return (target) => {
4
+ Reflect.defineMetadata('BaseDatastoreConfig', config, target);
5
+ };
6
+ }
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1kYXRhc3RvcmUtY29uZmlnLmRlY29yYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jvc29mdGxhYi9jb3JlL3NyYy9saWIvZGVjb3JhdG9ycy9iYXNlLWRhdGFzdG9yZS1jb25maWcuZGVjb3JhdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sVUFBVSxtQkFBbUIsQ0FBQyxTQUFjLEVBQUU7SUFDbEQsZ0RBQWdEO0lBQ2hELE9BQU8sQ0FBQyxNQUFXLEVBQUUsRUFBRTtRQUNyQixPQUFPLENBQUMsY0FBYyxDQUFDLHFCQUFxQixFQUFFLE1BQU0sRUFBRSxNQUFNLENBQUMsQ0FBQztJQUNoRSxDQUFDLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIEJhc2VEYXRhc3RvcmVDb25maWcoY29uZmlnOiBhbnkgPSB7fSkge1xyXG4gIC8vIHRzbGludDpkaXNhYmxlLW5leHQtbGluZTpvbmx5LWFycm93LWZ1bmN0aW9uc1xyXG4gIHJldHVybiAodGFyZ2V0OiBhbnkpID0+IHtcclxuICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoJ0Jhc2VEYXRhc3RvcmVDb25maWcnLCBjb25maWcsIHRhcmdldCk7XHJcbiAgfTtcclxufVxyXG4iXX0=
@@ -0,0 +1,10 @@
1
+ import { BaseMetaModel } from '../models/base-meta.model';
2
+ export function BaseModelConfig(config = {}) {
3
+ return (target) => {
4
+ if (typeof config['meta'] === 'undefined' || config['meta'] == null) {
5
+ config['meta'] = BaseMetaModel;
6
+ }
7
+ Reflect.defineMetadata('BaseModelConfig', config, target);
8
+ };
9
+ }
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1tb2RlbC1jb25maWcuZGVjb3JhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcm9zb2Z0bGFiL2NvcmUvc3JjL2xpYi9kZWNvcmF0b3JzL2Jhc2UtbW9kZWwtY29uZmlnLmRlY29yYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFFMUQsTUFBTSxVQUFVLGVBQWUsQ0FBQyxTQUFjLEVBQUU7SUFDOUMsT0FBTyxDQUFDLE1BQVcsRUFBRSxFQUFFO1FBQ3JCLElBQUksT0FBTyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssV0FBVyxJQUFJLE1BQU0sQ0FBQyxNQUFNLENBQUMsSUFBSSxJQUFJLEVBQUU7WUFDbkUsTUFBTSxDQUFDLE1BQU0sQ0FBQyxHQUFHLGFBQWEsQ0FBQztTQUNoQztRQUVELE9BQU8sQ0FBQyxjQUFjLENBQUMsaUJBQWlCLEVBQUUsTUFBTSxFQUFFLE1BQU0sQ0FBQyxDQUFDO0lBQzVELENBQUMsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBCYXNlTWV0YU1vZGVsIH0gZnJvbSAnLi4vbW9kZWxzL2Jhc2UtbWV0YS5tb2RlbCc7XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gQmFzZU1vZGVsQ29uZmlnKGNvbmZpZzogYW55ID0ge30pIHtcclxuICByZXR1cm4gKHRhcmdldDogYW55KSA9PiB7XHJcbiAgICBpZiAodHlwZW9mIGNvbmZpZ1snbWV0YSddID09PSAndW5kZWZpbmVkJyB8fCBjb25maWdbJ21ldGEnXSA9PSBudWxsKSB7XHJcbiAgICAgIGNvbmZpZ1snbWV0YSddID0gQmFzZU1ldGFNb2RlbDtcclxuICAgIH1cclxuXHJcbiAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKCdCYXNlTW9kZWxDb25maWcnLCBjb25maWcsIHRhcmdldCk7XHJcbiAgfTtcclxufVxyXG4iXX0=
@@ -0,0 +1,11 @@
1
+ export function CustomType(config = {}) {
2
+ return (target, propertyName) => {
3
+ const annotations = Reflect.getMetadata('CustomType', target) || [];
4
+ annotations.push({
5
+ propertyName,
6
+ relationship: config.key || propertyName
7
+ });
8
+ Reflect.defineMetadata('CustomType', annotations, target);
9
+ };
10
+ }
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VzdG9tLnR5cGUuZGVjb3JhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcm9zb2Z0bGFiL2NvcmUvc3JjL2xpYi9kZWNvcmF0b3JzL2N1c3RvbS50eXBlLmRlY29yYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLFVBQVUsVUFBVSxDQUFDLFNBQWMsRUFBRTtJQUN2QyxPQUFPLENBQUMsTUFBVyxFQUFFLFlBQTZCLEVBQUUsRUFBRTtRQUNsRCxNQUFNLFdBQVcsR0FBRyxPQUFPLENBQUMsV0FBVyxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7UUFFcEUsV0FBVyxDQUFDLElBQUksQ0FBQztZQUNiLFlBQVk7WUFDWixZQUFZLEVBQUUsTUFBTSxDQUFDLEdBQUcsSUFBSSxZQUFZO1NBQzNDLENBQUMsQ0FBQztRQUVILE9BQU8sQ0FBQyxjQUFjLENBQUMsWUFBWSxFQUFFLFdBQVcsRUFBRSxNQUFNLENBQUMsQ0FBQztJQUM5RCxDQUFDLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIEN1c3RvbVR5cGUoY29uZmlnOiBhbnkgPSB7fSkge1xyXG4gICAgcmV0dXJuICh0YXJnZXQ6IGFueSwgcHJvcGVydHlOYW1lOiBzdHJpbmcgfCBzeW1ib2wpID0+IHtcclxuICAgICAgICBjb25zdCBhbm5vdGF0aW9ucyA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEoJ0N1c3RvbVR5cGUnLCB0YXJnZXQpIHx8IFtdO1xyXG5cclxuICAgICAgICBhbm5vdGF0aW9ucy5wdXNoKHtcclxuICAgICAgICAgICAgcHJvcGVydHlOYW1lLFxyXG4gICAgICAgICAgICByZWxhdGlvbnNoaXA6IGNvbmZpZy5rZXkgfHwgcHJvcGVydHlOYW1lXHJcbiAgICAgICAgfSk7XHJcblxyXG4gICAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoJ0N1c3RvbVR5cGUnLCBhbm5vdGF0aW9ucywgdGFyZ2V0KTtcclxuICAgIH07XHJcbn1cclxuIl19
@@ -0,0 +1,23 @@
1
+ import { CellTextAlign, GridLayoutFormat } from '../models/grid-layout-format.enum';
2
+ export function GridLayout(translateKey, width, grow = 0, shrink = 0, subProperty, formating = GridLayoutFormat.none, format = '', order = 0, textAlign = CellTextAlign.left) {
3
+ return (target, propertyName) => {
4
+ const annotations = Reflect.getMetadata('GridLayout', target) || [];
5
+ let propName = propertyName;
6
+ if (subProperty) {
7
+ propName = propertyName.toString() + '.' + subProperty;
8
+ }
9
+ annotations.push({
10
+ propertyName: propName,
11
+ translateKey,
12
+ width,
13
+ grow,
14
+ shrink,
15
+ formating,
16
+ format,
17
+ order,
18
+ textAlign
19
+ });
20
+ Reflect.defineMetadata('GridLayout', annotations, target);
21
+ };
22
+ }
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ3JpZC1sYXlvdXQuZGVjb3JhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcm9zb2Z0bGFiL2NvcmUvc3JjL2xpYi9kZWNvcmF0b3JzL2dyaWQtbGF5b3V0LmRlY29yYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsYUFBYSxFQUFFLGdCQUFnQixFQUFFLE1BQU0sbUNBQW1DLENBQUM7QUFFcEYsTUFBTSxVQUFVLFVBQVUsQ0FBQyxZQUFvQixFQUFFLEtBQXFCLEVBQ2xFLE9BQWMsQ0FBQyxFQUFFLFNBQWdCLENBQUMsRUFBRSxXQUEyQixFQUMvRCxZQUE4QixnQkFBZ0IsQ0FBQyxJQUFJLEVBQ25ELFNBQWlCLEVBQUUsRUFBRSxRQUFnQixDQUFDLEVBQUUsWUFBMkIsYUFBYSxDQUFDLElBQUk7SUFDckYsT0FBTyxDQUFDLE1BQVcsRUFBRSxZQUE2QixFQUFFLEVBQUU7UUFDbEQsTUFBTSxXQUFXLEdBQUcsT0FBTyxDQUFDLFdBQVcsQ0FBQyxZQUFZLEVBQUUsTUFBTSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ3BFLElBQUksUUFBUSxHQUFHLFlBQVksQ0FBQTtRQUMzQixJQUFJLFdBQVcsRUFBRTtZQUNiLFFBQVEsR0FBRyxZQUFZLENBQUMsUUFBUSxFQUFFLEdBQUcsR0FBRyxHQUFHLFdBQVcsQ0FBQztTQUMxRDtRQUNELFdBQVcsQ0FBQyxJQUFJLENBQUM7WUFDYixZQUFZLEVBQUUsUUFBUTtZQUN0QixZQUFZO1lBQ1osS0FBSztZQUNMLElBQUk7WUFDSixNQUFNO1lBQ04sU0FBUztZQUNULE1BQU07WUFDTixLQUFLO1lBQ0wsU0FBUztTQUNaLENBQUMsQ0FBQztRQUVILE9BQU8sQ0FBQyxjQUFjLENBQUMsWUFBWSxFQUFFLFdBQVcsRUFBRSxNQUFNLENBQUMsQ0FBQztJQUM5RCxDQUFDLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2VsbFRleHRBbGlnbiwgR3JpZExheW91dEZvcm1hdCB9IGZyb20gJy4uL21vZGVscy9ncmlkLWxheW91dC1mb3JtYXQuZW51bSc7XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gR3JpZExheW91dCh0cmFuc2xhdGVLZXk6IHN0cmluZywgd2lkdGg/OiBudW1iZXIgfCBudWxsLFxyXG4gICAgZ3JvdzogMCB8IDEgPSAwLCBzaHJpbms6IDAgfCAxID0gMCwgc3ViUHJvcGVydHk/OiBzdHJpbmcgfCBudWxsLFxyXG4gICAgZm9ybWF0aW5nOiBHcmlkTGF5b3V0Rm9ybWF0ID0gR3JpZExheW91dEZvcm1hdC5ub25lLFxyXG4gICAgZm9ybWF0OiBzdHJpbmcgPSAnJywgb3JkZXI6IG51bWJlciA9IDAsIHRleHRBbGlnbjogQ2VsbFRleHRBbGlnbiA9IENlbGxUZXh0QWxpZ24ubGVmdCkge1xyXG4gICAgcmV0dXJuICh0YXJnZXQ6IGFueSwgcHJvcGVydHlOYW1lOiBzdHJpbmcgfCBzeW1ib2wpID0+IHtcclxuICAgICAgICBjb25zdCBhbm5vdGF0aW9ucyA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEoJ0dyaWRMYXlvdXQnLCB0YXJnZXQpIHx8IFtdO1xyXG4gICAgICAgIGxldCBwcm9wTmFtZSA9IHByb3BlcnR5TmFtZVxyXG4gICAgICAgIGlmIChzdWJQcm9wZXJ0eSkge1xyXG4gICAgICAgICAgICBwcm9wTmFtZSA9IHByb3BlcnR5TmFtZS50b1N0cmluZygpICsgJy4nICsgc3ViUHJvcGVydHk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGFubm90YXRpb25zLnB1c2goe1xyXG4gICAgICAgICAgICBwcm9wZXJ0eU5hbWU6IHByb3BOYW1lLFxyXG4gICAgICAgICAgICB0cmFuc2xhdGVLZXksXHJcbiAgICAgICAgICAgIHdpZHRoLFxyXG4gICAgICAgICAgICBncm93LFxyXG4gICAgICAgICAgICBzaHJpbmssXHJcbiAgICAgICAgICAgIGZvcm1hdGluZyxcclxuICAgICAgICAgICAgZm9ybWF0LFxyXG4gICAgICAgICAgICBvcmRlcixcclxuICAgICAgICAgICAgdGV4dEFsaWduXHJcbiAgICAgICAgfSk7XHJcblxyXG4gICAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoJ0dyaWRMYXlvdXQnLCBhbm5vdGF0aW9ucywgdGFyZ2V0KTtcclxuICAgIH07XHJcbn1cclxuIl19
@@ -0,0 +1,96 @@
1
+ import { ContentChildren, Directive, Input } from '@angular/core';
2
+ import { BehaviorSubject, combineLatest, merge } from 'rxjs';
3
+ import { map, switchMap } from 'rxjs/operators';
4
+ import { TranslatedElementDirective } from './translated-element.directive';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@ngx-translate/core";
7
+ const TOKEN_START_DEMARC = '{{';
8
+ const TOKEN_END_DEMARC = '}}';
9
+ // adapted from @kasperlauge's solution in https://github.com/ngx-translate/core/issues/223
10
+ export class TranslatedContentDirective {
11
+ constructor(viewRef, renderer, translateService, changeDetectorRef) {
12
+ this.viewRef = viewRef;
13
+ this.renderer = renderer;
14
+ this.translateService = translateService;
15
+ this.changeDetectorRef = changeDetectorRef;
16
+ this.subs = [];
17
+ }
18
+ ngOnInit() {
19
+ this.rawTranslation = merge(this.translateService.get(this.translationKey), this.translateService.onLangChange.asObservable().pipe(switchMap(() => this.translateService.get(this.translationKey))));
20
+ }
21
+ ngAfterContentInit() {
22
+ // QueryList.changes doesn't re-emit after its initial value, which we have by now
23
+ // BehaviorSubjects re-emit their initial value on subscription, so we get what we need by merging
24
+ // the BehaviorSubject and the QueryList.changes observable
25
+ const elementsSubject = new BehaviorSubject(this.elements.toArray());
26
+ const elementsChanges = merge(elementsSubject, this.elements.changes);
27
+ this.translationData = combineLatest(this.rawTranslation, elementsChanges)
28
+ .pipe(map(([rawTranslation]) => {
29
+ return {
30
+ elements: this.elements.toArray(),
31
+ rawTranslation,
32
+ };
33
+ }));
34
+ this.subs.push(this.translationData.subscribe(this.render.bind(this)));
35
+ }
36
+ render(translationData) {
37
+ if (!translationData.rawTranslation || translationData.rawTranslation === this.translationKey) {
38
+ throw new Error(`No resource matching the key '${this.translationKey}'`);
39
+ }
40
+ while (this.viewRef.element.nativeElement.firstChild) {
41
+ this.renderer.removeChild(this.viewRef.element.nativeElement, this.viewRef.element.nativeElement.firstChild);
42
+ }
43
+ let lastTokenEnd = 0;
44
+ while (lastTokenEnd < translationData.rawTranslation.length) {
45
+ const tokenStartDemarc = translationData.rawTranslation.indexOf(TOKEN_START_DEMARC, lastTokenEnd);
46
+ if (tokenStartDemarc < 0) {
47
+ break;
48
+ }
49
+ const tokenStart = tokenStartDemarc + TOKEN_START_DEMARC.length;
50
+ const tokenEnd = translationData.rawTranslation.indexOf(TOKEN_END_DEMARC, tokenStart);
51
+ if (tokenEnd < 0) {
52
+ throw new Error(`Encountered unterminated token in translation string '${this.translationKey}'`);
53
+ }
54
+ const tokenEndDemarc = tokenEnd + TOKEN_END_DEMARC.length;
55
+ const precedingText = translationData.rawTranslation.substring(lastTokenEnd, tokenStartDemarc);
56
+ const precedingTextElement = this.renderer.createText(precedingText);
57
+ this.renderer.appendChild(this.viewRef.element.nativeElement, precedingTextElement);
58
+ const elementKey = translationData.rawTranslation.substring(tokenStart, tokenEnd);
59
+ const embeddedElementTemplate = translationData.elements.find(element => element.elementKey === elementKey);
60
+ if (embeddedElementTemplate) {
61
+ const embeddedElementView = embeddedElementTemplate.viewRef.createEmbeddedView(embeddedElementTemplate.templateRef);
62
+ this.renderer.appendChild(this.viewRef.element.nativeElement, embeddedElementView.rootNodes[0]);
63
+ }
64
+ else {
65
+ const missingTokenText = translationData.rawTranslation.substring(tokenStartDemarc, tokenEndDemarc);
66
+ const missingTokenElement = this.renderer.createText(missingTokenText);
67
+ this.renderer.appendChild(this.viewRef.element.nativeElement, missingTokenElement);
68
+ }
69
+ lastTokenEnd = tokenEndDemarc;
70
+ }
71
+ const trailingText = translationData.rawTranslation.substring(lastTokenEnd);
72
+ const trailingTextElement = this.renderer.createText(trailingText);
73
+ this.renderer.appendChild(this.viewRef.element.nativeElement, trailingTextElement);
74
+ // in case the rendering happens outside of a change detection event, this ensures that any translations in the
75
+ // embedded elements are rendered
76
+ this.changeDetectorRef.detectChanges();
77
+ }
78
+ ngOnDestroy() {
79
+ this.subs.forEach(sub => sub.unsubscribe());
80
+ }
81
+ }
82
+ TranslatedContentDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: TranslatedContentDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.Renderer2 }, { token: i1.TranslateService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Directive });
83
+ TranslatedContentDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.0", type: TranslatedContentDirective, selector: "[appTranslatedContent]", inputs: { translationKey: ["appTranslatedContent", "translationKey"] }, queries: [{ propertyName: "elements", predicate: TranslatedElementDirective }], ngImport: i0 });
84
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: TranslatedContentDirective, decorators: [{
85
+ type: Directive,
86
+ args: [{
87
+ selector: '[appTranslatedContent]',
88
+ }]
89
+ }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.Renderer2 }, { type: i1.TranslateService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { translationKey: [{
90
+ type: Input,
91
+ args: ['appTranslatedContent']
92
+ }], elements: [{
93
+ type: ContentChildren,
94
+ args: [TranslatedElementDirective]
95
+ }] } });
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"translated-content.directive.js","sourceRoot":"","sources":["../../../../../../projects/rosoftlab/core/src/lib/directives/translated-content.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  AfterContentInit, ChangeDetectorRef, ContentChildren, Directive,\r\n  Input, OnDestroy, OnInit, QueryList, Renderer2, ViewContainerRef\r\n} from '@angular/core';\r\nimport { TranslateService } from '@ngx-translate/core';\r\nimport { BehaviorSubject, combineLatest, merge, Observable, Subscription } from 'rxjs';\r\nimport { map, switchMap } from 'rxjs/operators';\r\nimport { TranslatedElementDirective } from './translated-element.directive';\r\n\r\ninterface TranslationData {\r\n  elements: TranslatedElementDirective[];\r\n  rawTranslation: string;\r\n}\r\n\r\nconst TOKEN_START_DEMARC = '{{';\r\nconst TOKEN_END_DEMARC = '}}';\r\n\r\n// adapted from @kasperlauge's solution in https://github.com/ngx-translate/core/issues/223\r\n@Directive({\r\n  selector: '[appTranslatedContent]',\r\n})\r\nexport class TranslatedContentDirective implements OnInit, OnDestroy, AfterContentInit {\r\n\r\n  @Input('appTranslatedContent') translationKey: string;\r\n\r\n  @ContentChildren(TranslatedElementDirective)\r\n  private elements: QueryList<TranslatedElementDirective>;\r\n\r\n  private subs: Subscription[] = [];\r\n  private rawTranslation: Observable<string>;\r\n  private translationData: Observable<TranslationData>;\r\n\r\n  constructor(\r\n    private viewRef: ViewContainerRef,\r\n    private renderer: Renderer2,\r\n    private translateService: TranslateService,\r\n    private changeDetectorRef: ChangeDetectorRef,\r\n  ) { }\r\n\r\n  public ngOnInit(): void {\r\n    this.rawTranslation = merge(\r\n      this.translateService.get(this.translationKey),\r\n      this.translateService.onLangChange.asObservable().pipe(switchMap(() => this.translateService.get(this.translationKey)))\r\n    );\r\n  }\r\n\r\n\r\n  public ngAfterContentInit(): void {\r\n    // QueryList.changes doesn't re-emit after its initial value, which we have by now\r\n    // BehaviorSubjects re-emit their initial value on subscription, so we get what we need by merging\r\n    // the BehaviorSubject and the QueryList.changes observable\r\n    const elementsSubject = new BehaviorSubject(this.elements.toArray());\r\n    const elementsChanges = merge(elementsSubject, this.elements.changes);\r\n\r\n    this.translationData = combineLatest(this.rawTranslation, elementsChanges)\r\n      .pipe(\r\n        map(([rawTranslation]) => {\r\n          return {\r\n            elements: this.elements.toArray(),\r\n            rawTranslation,\r\n          };\r\n        })\r\n      );\r\n\r\n    this.subs.push(this.translationData.subscribe(this.render.bind(this)));\r\n  }\r\n\r\n  private render(translationData: TranslationData): void {\r\n\r\n    if (!translationData.rawTranslation || translationData.rawTranslation === this.translationKey) {\r\n      throw new Error(`No resource matching the key '${this.translationKey}'`);\r\n    }\r\n\r\n    while (this.viewRef.element.nativeElement.firstChild) {\r\n      this.renderer.removeChild(this.viewRef.element.nativeElement, this.viewRef.element.nativeElement.firstChild);\r\n    }\r\n\r\n    let lastTokenEnd = 0;\r\n    while (lastTokenEnd < translationData.rawTranslation.length) {\r\n      const tokenStartDemarc = translationData.rawTranslation.indexOf(TOKEN_START_DEMARC, lastTokenEnd);\r\n      if (tokenStartDemarc < 0) {\r\n        break;\r\n      }\r\n      const tokenStart = tokenStartDemarc + TOKEN_START_DEMARC.length;\r\n      const tokenEnd = translationData.rawTranslation.indexOf(TOKEN_END_DEMARC, tokenStart);\r\n      if (tokenEnd < 0) {\r\n        throw new Error(`Encountered unterminated token in translation string '${this.translationKey}'`);\r\n      }\r\n      const tokenEndDemarc = tokenEnd + TOKEN_END_DEMARC.length;\r\n\r\n      const precedingText = translationData.rawTranslation.substring(lastTokenEnd, tokenStartDemarc);\r\n      const precedingTextElement = this.renderer.createText(precedingText);\r\n      this.renderer.appendChild(this.viewRef.element.nativeElement, precedingTextElement);\r\n\r\n      const elementKey = translationData.rawTranslation.substring(tokenStart, tokenEnd);\r\n      const embeddedElementTemplate = translationData.elements.find(element => element.elementKey === elementKey);\r\n      if (embeddedElementTemplate) {\r\n        const embeddedElementView = embeddedElementTemplate.viewRef.createEmbeddedView(embeddedElementTemplate.templateRef);\r\n        this.renderer.appendChild(this.viewRef.element.nativeElement, embeddedElementView.rootNodes[0]);\r\n      } else {\r\n        const missingTokenText = translationData.rawTranslation.substring(tokenStartDemarc, tokenEndDemarc);\r\n        const missingTokenElement = this.renderer.createText(missingTokenText);\r\n        this.renderer.appendChild(this.viewRef.element.nativeElement, missingTokenElement);\r\n      }\r\n\r\n      lastTokenEnd = tokenEndDemarc;\r\n    }\r\n\r\n    const trailingText = translationData.rawTranslation.substring(lastTokenEnd);\r\n    const trailingTextElement = this.renderer.createText(trailingText);\r\n\r\n    this.renderer.appendChild(this.viewRef.element.nativeElement, trailingTextElement);\r\n\r\n    // in case the rendering happens outside of a change detection event, this ensures that any translations in the\r\n    // embedded elements are rendered\r\n    this.changeDetectorRef.detectChanges();\r\n\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this.subs.forEach(sub => sub.unsubscribe());\r\n  }\r\n}"]}