@ecodev/natural 58.0.3 → 58.0.5

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.
@@ -17,14 +17,6 @@ function isNaturalDialogTriggerProvidedData(dialogData) {
17
17
  }
18
18
  // @dynamic
19
19
  export class NaturalAbstractDetail extends NaturalAbstractPanel {
20
- #dialogData;
21
- /**
22
- * Once set, this must not change anymore, especially not right after the creation mutation,
23
- * so the form does not switch from creation mode to update mode without an actual reload of
24
- * model from DB (by navigating to update page).
25
- */
26
- #isUpdatePage;
27
- #changes;
28
20
  constructor(key, service) {
29
21
  super();
30
22
  this.key = key;
@@ -63,14 +55,14 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
63
55
  * Injected service
64
56
  */
65
57
  this.route = inject(ActivatedRoute);
66
- this.#dialogData = inject(MAT_DIALOG_DATA, { optional: true });
58
+ this._dialogData = inject(MAT_DIALOG_DATA, { optional: true });
67
59
  /**
68
60
  * Once set, this must not change anymore, especially not right after the creation mutation,
69
61
  * so the form does not switch from creation mode to update mode without an actual reload of
70
62
  * model from DB (by navigating to update page).
71
63
  */
72
- this.#isUpdatePage = false;
73
- this.#changes = new CumulativeChanges();
64
+ this._isUpdatePage = false;
65
+ this.changes = new CumulativeChanges();
74
66
  }
75
67
  /**
76
68
  * You probably should not override this method. Instead, consider overriding `initForm()`.
@@ -80,8 +72,8 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
80
72
  this.initForm();
81
73
  }
82
74
  else {
83
- const route = isNaturalDialogTriggerProvidedData(this.#dialogData)
84
- ? this.#dialogData.activatedRoute
75
+ const route = isNaturalDialogTriggerProvidedData(this._dialogData)
76
+ ? this._dialogData.activatedRoute
85
77
  : this.route;
86
78
  this.#subscribeToModelFromResolvedData(route);
87
79
  }
@@ -117,7 +109,7 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
117
109
  * This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing
118
110
  */
119
111
  isUpdatePage() {
120
- return this.#isUpdatePage;
112
+ return this._isUpdatePage;
121
113
  }
122
114
  /**
123
115
  * Update the object on the server with the values from the form fields that were modified since
@@ -133,15 +125,15 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
133
125
  ifValid(this.form).subscribe(() => {
134
126
  const newValues = this.form.getRawValue();
135
127
  if (submitAllFields) {
136
- this.#changes.initialize({});
128
+ this.changes.initialize({});
137
129
  }
138
130
  const toSubmit = {
139
131
  id: this.data.model.id,
140
- ...this.#changes.differences(newValues),
132
+ ...this.changes.differences(newValues),
141
133
  };
142
134
  const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);
143
135
  update.subscribe(model => {
144
- this.#changes.commit(newValues);
136
+ this.changes.commit(newValues);
145
137
  this.alertService.info($localize `Mis à jour`);
146
138
  this.postUpdate(model);
147
139
  });
@@ -227,9 +219,9 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
227
219
  * will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.
228
220
  */
229
221
  initForm() {
230
- this.#isUpdatePage = !!this.data.model.id;
222
+ this._isUpdatePage = !!this.data.model.id;
231
223
  this.form = this.service.getFormGroup(this.data.model);
232
- this.#changes.initialize(this.form.getRawValue());
224
+ this.changes.initialize(this.form.getRawValue());
233
225
  }
234
226
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: NaturalAbstractDetail, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive }); }
235
227
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.5", type: NaturalAbstractDetail, usesInheritance: true, ngImport: i0 }); }
@@ -237,4 +229,4 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
237
229
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: NaturalAbstractDetail, decorators: [{
238
230
  type: Directive
239
231
  }], ctorParameters: () => [{ type: undefined }, { type: undefined }] });
240
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-detail.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/abstract-detail.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase} from 'lodash-es';\nimport {NaturalAlertService} from '../modules/alert/alert.service';\nimport {NaturalAbstractPanel} from '../modules/panels/abstract-panel';\nimport {NaturalAbstractModelService} from '../services/abstract-model.service';\nimport {ExtractResolve, ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {EMPTY, endWith, finalize, last, Observable, switchMap, tap} from 'rxjs';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {CumulativeChanges} from './cumulative-changes';\nimport {MAT_DIALOG_DATA} from '@angular/material/dialog';\nimport {NaturalDialogTriggerProvidedData} from '../modules/dialog-trigger/dialog-trigger.component';\n\n/**\n * `Data` contains in `model` either the model fetched from DB or default values (without ID). And besides `model`,\n * any other extra keys defined by Extra.\n */\ntype Data<TService, Extra> = {model: {id?: string} & ExtractResolve<TService>} & Extra;\n\nfunction isNaturalDialogTriggerProvidedData(\n    dialogData: unknown,\n): dialogData is NaturalDialogTriggerProvidedData<never> {\n    return (\n        !!dialogData &&\n        typeof dialogData === 'object' &&\n        'activatedRoute' in dialogData &&\n        dialogData.activatedRoute instanceof ActivatedRoute\n    );\n}\n\n// @dynamic\n@Directive()\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            {id: string},\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n        ExtraResolve extends Literal = Record<never, never>,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Data retrieved by the server via route resolvers.\n     *\n     * The key `model` is special. It is readonly and represents the model being updated\n     * as it exists on server. The value is kept up to date when Apollo mutates it on server.\n     *\n     * The only time when `model` is not readonly is during creation. Only then can we modify the model values directly.\n     *\n     * Other keys, if present, are whatever is returned from route resolvers as-is.\n     */\n    public override data: Data<TService, ExtraResolve> = {\n        model: this.service.getDefaultForServer(),\n    } as Data<TService, ExtraResolve>;\n\n    /**\n     * Form that manages the data from the controller\n     */\n    public form: UntypedFormGroup = new UntypedFormGroup({});\n\n    /**\n     * Show / hides the bottom fab button (mostly to hide it when we are on other tabs where semantic of button can conflict with ...\n     * semantic of data on other tab, like relations that list other objects)\n     */\n    public showFabButton = true;\n\n    /**\n     * Injected service\n     */\n    protected readonly alertService = inject(NaturalAlertService);\n\n    /**\n     * Injected service\n     */\n    protected readonly router = inject(Router);\n\n    /**\n     * Injected service\n     */\n    protected readonly route = inject(ActivatedRoute);\n\n    #dialogData: unknown = inject(MAT_DIALOG_DATA, {optional: true});\n\n    /**\n     * Once set, this must not change anymore, especially not right after the creation mutation,\n     * so the form does not switch from creation mode to update mode without an actual reload of\n     * model from DB (by navigating to update page).\n     */\n    #isUpdatePage = false;\n    readonly #changes = new CumulativeChanges<typeof this.form.getRawValue>();\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n    ) {\n        super();\n    }\n\n    /**\n     * You probably should not override this method. Instead, consider overriding `initForm()`.\n     */\n    public ngOnInit(): void {\n        if (this.isPanel) {\n            this.initForm();\n        } else {\n            const route = isNaturalDialogTriggerProvidedData(this.#dialogData)\n                ? this.#dialogData.activatedRoute\n                : this.route;\n            this.#subscribeToModelFromResolvedData(route);\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    #subscribeToModelFromResolvedData(route: ActivatedRoute): void {\n        let previousId = -1;\n        route.data\n            .pipe(\n                switchMap(data => {\n                    if (!(data.model instanceof Observable)) {\n                        throw new Error(\n                            'Resolved data must include the key `model`, and it must be an observable (usually one from Apollo).',\n                        );\n                    }\n\n                    // Subscribe to model to know when Apollo cache is changed, so we can reflect it into `data.model`\n                    return data.model.pipe(\n                        tap((model: ExtractResolve<TService>) => {\n                            this.data = {\n                                ...data,\n                                model: model,\n                            } as Data<TService, ExtraResolve>;\n\n                            if (previousId !== model.id) {\n                                previousId = model.id;\n                                this.initForm();\n                            }\n                        }),\n                    );\n                }),\n            )\n            .subscribe();\n    }\n\n    /**\n     * Returns whether `data.model` was fetched from DB, so we are on an update page, or if it is a new object\n     * with (only) default values, so we are on a creation page.\n     *\n     * This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing\n     */\n    protected isUpdatePage(): this is {data: {model: ExtractTone<TService>}} {\n        return this.#isUpdatePage;\n    }\n\n    /**\n     * Update the object on the server with the values from the form fields that were modified since\n     * the initialization, or since the previous successful update.\n     *\n     * Form fields that are never modified are **not** sent to the server, unless if you specify `submitAllFields`.\n     */\n    public update(now = false, submitAllFields = false): void {\n        if (!this.isUpdatePage()) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            const newValues = this.form.getRawValue();\n            if (submitAllFields) {\n                this.#changes.initialize({});\n            }\n\n            const toSubmit = {\n                id: this.data.model.id,\n                ...this.#changes.differences(newValues),\n            };\n\n            const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);\n            update.subscribe(model => {\n                this.#changes.commit(newValues);\n                this.alertService.info($localize`Mis à jour`);\n                this.postUpdate(model);\n            });\n        });\n    }\n\n    public create(redirect = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        const newValues = this.form.getRawValue();\n        this.form.disable();\n\n        this.service\n            .create(newValues)\n            .pipe(\n                switchMap(model => {\n                    this.alertService.info($localize`Créé`);\n\n                    return this.postCreate(model).pipe(endWith(model), last());\n                }),\n                switchMap(model => {\n                    if (redirect) {\n                        if (this.isPanel) {\n                            const oldUrl = this.router.url;\n                            const nextUrl = this.panelData?.config.params.nextRoute;\n                            const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');\n                            return this.router.navigateByUrl(newUrl); // replace /new by /123\n                        } else {\n                            return this.router.navigate(['..', model.id], {relativeTo: this.route});\n                        }\n                    }\n\n                    return EMPTY;\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    /**\n     * `confirmer` can be used to open a custom dialog, or anything else, to confirm the deletion, instead of the standard dialog\n     */\n    public delete(redirectionRoute?: unknown[], confirmer?: Observable<boolean | undefined>): void {\n        this.form.disable();\n\n        (\n            confirmer ??\n            this.alertService.confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n        )\n            .pipe(\n                switchMap(confirmed => {\n                    if (!confirmed || !this.isUpdatePage()) {\n                        return EMPTY;\n                    }\n\n                    this.preDelete(this.data.model);\n\n                    return this.service.delete([this.data.model]).pipe(\n                        switchMap(() => {\n                            this.alertService.info($localize`Supprimé`);\n\n                            if (this.isPanel) {\n                                this.panelService?.goToPenultimatePanel();\n\n                                return EMPTY;\n                            } else {\n                                const defaultRoute = ['../../' + kebabCase(this.key)];\n                                return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {\n                                    relativeTo: this.route,\n                                });\n                            }\n                        }),\n                    );\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postUpdate(model: ExtractTupdate<TService>): void {\n        // noop\n    }\n\n    /**\n     * Returns an observable that will be subscribed to immediately and the\n     * redirect navigation will only happen after the observable completes.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected preDelete(model: ExtractTone<TService>): void {\n        // noop\n    }\n\n    /**\n     * Initialize the form whenever it is needed.\n     *\n     * You should override this method, and not `ngOnInit()` if you need to customize the form. Because this will\n     * correctly be called more than one time per component instance if needed, when the route changes. But `ngOnInit()`\n     * will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.\n     */\n    protected initForm(): void {\n        this.#isUpdatePage = !!this.data.model.id;\n        this.form = this.service.getFormGroup(this.data.model);\n        this.#changes.initialize(this.form.getRawValue());\n    }\n}\n"]}
232
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-detail.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/abstract-detail.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase} from 'lodash-es';\nimport {NaturalAlertService} from '../modules/alert/alert.service';\nimport {NaturalAbstractPanel} from '../modules/panels/abstract-panel';\nimport {NaturalAbstractModelService} from '../services/abstract-model.service';\nimport {ExtractResolve, ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {EMPTY, endWith, finalize, last, Observable, switchMap, tap} from 'rxjs';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {CumulativeChanges} from './cumulative-changes';\nimport {MAT_DIALOG_DATA} from '@angular/material/dialog';\nimport {NaturalDialogTriggerProvidedData} from '../modules/dialog-trigger/dialog-trigger.component';\n\n/**\n * `Data` contains in `model` either the model fetched from DB or default values (without ID). And besides `model`,\n * any other extra keys defined by Extra.\n */\ntype Data<TService, Extra> = {model: {id?: string} & ExtractResolve<TService>} & Extra;\n\nfunction isNaturalDialogTriggerProvidedData(\n    dialogData: unknown,\n): dialogData is NaturalDialogTriggerProvidedData<never> {\n    return (\n        !!dialogData &&\n        typeof dialogData === 'object' &&\n        'activatedRoute' in dialogData &&\n        dialogData.activatedRoute instanceof ActivatedRoute\n    );\n}\n\n// @dynamic\n@Directive()\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            {id: string},\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n        ExtraResolve extends Literal = Record<never, never>,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Data retrieved by the server via route resolvers.\n     *\n     * The key `model` is special. It is readonly and represents the model being updated\n     * as it exists on server. The value is kept up to date when Apollo mutates it on server.\n     *\n     * The only time when `model` is not readonly is during creation. Only then can we modify the model values directly.\n     *\n     * Other keys, if present, are whatever is returned from route resolvers as-is.\n     */\n    public override data: Data<TService, ExtraResolve> = {\n        model: this.service.getDefaultForServer(),\n    } as Data<TService, ExtraResolve>;\n\n    /**\n     * Form that manages the data from the controller\n     */\n    public form: UntypedFormGroup = new UntypedFormGroup({});\n\n    /**\n     * Show / hides the bottom fab button (mostly to hide it when we are on other tabs where semantic of button can conflict with ...\n     * semantic of data on other tab, like relations that list other objects)\n     */\n    public showFabButton = true;\n\n    /**\n     * Injected service\n     */\n    protected readonly alertService = inject(NaturalAlertService);\n\n    /**\n     * Injected service\n     */\n    protected readonly router = inject(Router);\n\n    /**\n     * Injected service\n     */\n    protected readonly route = inject(ActivatedRoute);\n\n    private _dialogData: unknown = inject(MAT_DIALOG_DATA, {optional: true});\n\n    /**\n     * Once set, this must not change anymore, especially not right after the creation mutation,\n     * so the form does not switch from creation mode to update mode without an actual reload of\n     * model from DB (by navigating to update page).\n     */\n    private _isUpdatePage = false;\n    private readonly changes = new CumulativeChanges<typeof this.form.getRawValue>();\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n    ) {\n        super();\n    }\n\n    /**\n     * You probably should not override this method. Instead, consider overriding `initForm()`.\n     */\n    public ngOnInit(): void {\n        if (this.isPanel) {\n            this.initForm();\n        } else {\n            const route = isNaturalDialogTriggerProvidedData(this._dialogData)\n                ? this._dialogData.activatedRoute\n                : this.route;\n            this.#subscribeToModelFromResolvedData(route);\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    #subscribeToModelFromResolvedData(route: ActivatedRoute): void {\n        let previousId = -1;\n        route.data\n            .pipe(\n                switchMap(data => {\n                    if (!(data.model instanceof Observable)) {\n                        throw new Error(\n                            'Resolved data must include the key `model`, and it must be an observable (usually one from Apollo).',\n                        );\n                    }\n\n                    // Subscribe to model to know when Apollo cache is changed, so we can reflect it into `data.model`\n                    return data.model.pipe(\n                        tap((model: ExtractResolve<TService>) => {\n                            this.data = {\n                                ...data,\n                                model: model,\n                            } as Data<TService, ExtraResolve>;\n\n                            if (previousId !== model.id) {\n                                previousId = model.id;\n                                this.initForm();\n                            }\n                        }),\n                    );\n                }),\n            )\n            .subscribe();\n    }\n\n    /**\n     * Returns whether `data.model` was fetched from DB, so we are on an update page, or if it is a new object\n     * with (only) default values, so we are on a creation page.\n     *\n     * This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing\n     */\n    protected isUpdatePage(): this is {data: {model: ExtractTone<TService>}} {\n        return this._isUpdatePage;\n    }\n\n    /**\n     * Update the object on the server with the values from the form fields that were modified since\n     * the initialization, or since the previous successful update.\n     *\n     * Form fields that are never modified are **not** sent to the server, unless if you specify `submitAllFields`.\n     */\n    public update(now = false, submitAllFields = false): void {\n        if (!this.isUpdatePage()) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            const newValues = this.form.getRawValue();\n            if (submitAllFields) {\n                this.changes.initialize({});\n            }\n\n            const toSubmit = {\n                id: this.data.model.id,\n                ...this.changes.differences(newValues),\n            };\n\n            const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);\n            update.subscribe(model => {\n                this.changes.commit(newValues);\n                this.alertService.info($localize`Mis à jour`);\n                this.postUpdate(model);\n            });\n        });\n    }\n\n    public create(redirect = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        const newValues = this.form.getRawValue();\n        this.form.disable();\n\n        this.service\n            .create(newValues)\n            .pipe(\n                switchMap(model => {\n                    this.alertService.info($localize`Créé`);\n\n                    return this.postCreate(model).pipe(endWith(model), last());\n                }),\n                switchMap(model => {\n                    if (redirect) {\n                        if (this.isPanel) {\n                            const oldUrl = this.router.url;\n                            const nextUrl = this.panelData?.config.params.nextRoute;\n                            const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');\n                            return this.router.navigateByUrl(newUrl); // replace /new by /123\n                        } else {\n                            return this.router.navigate(['..', model.id], {relativeTo: this.route});\n                        }\n                    }\n\n                    return EMPTY;\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    /**\n     * `confirmer` can be used to open a custom dialog, or anything else, to confirm the deletion, instead of the standard dialog\n     */\n    public delete(redirectionRoute?: unknown[], confirmer?: Observable<boolean | undefined>): void {\n        this.form.disable();\n\n        (\n            confirmer ??\n            this.alertService.confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n        )\n            .pipe(\n                switchMap(confirmed => {\n                    if (!confirmed || !this.isUpdatePage()) {\n                        return EMPTY;\n                    }\n\n                    this.preDelete(this.data.model);\n\n                    return this.service.delete([this.data.model]).pipe(\n                        switchMap(() => {\n                            this.alertService.info($localize`Supprimé`);\n\n                            if (this.isPanel) {\n                                this.panelService?.goToPenultimatePanel();\n\n                                return EMPTY;\n                            } else {\n                                const defaultRoute = ['../../' + kebabCase(this.key)];\n                                return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {\n                                    relativeTo: this.route,\n                                });\n                            }\n                        }),\n                    );\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postUpdate(model: ExtractTupdate<TService>): void {\n        // noop\n    }\n\n    /**\n     * Returns an observable that will be subscribed to immediately and the\n     * redirect navigation will only happen after the observable completes.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected preDelete(model: ExtractTone<TService>): void {\n        // noop\n    }\n\n    /**\n     * Initialize the form whenever it is needed.\n     *\n     * You should override this method, and not `ngOnInit()` if you need to customize the form. Because this will\n     * correctly be called more than one time per component instance if needed, when the route changes. But `ngOnInit()`\n     * will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.\n     */\n    protected initForm(): void {\n        this._isUpdatePage = !!this.data.model.id;\n        this.form = this.service.getFormGroup(this.data.model);\n        this.changes.initialize(this.form.getRawValue());\n    }\n}\n"]}
@@ -3,14 +3,16 @@ import { cloneDeep, isEqual } from 'lodash-es';
3
3
  * Cumulate all changes made to an object over time
4
4
  */
5
5
  export class CumulativeChanges {
6
- #original = {};
7
- #diff = {};
6
+ constructor() {
7
+ this.original = {};
8
+ this.diff = {};
9
+ }
8
10
  /**
9
11
  * Initialize the original values, should be called exactly one time per instance
10
12
  */
11
13
  initialize(originalValues) {
12
- this.#original = cloneDeep(originalValues);
13
- this.#diff = {};
14
+ this.original = cloneDeep(originalValues);
15
+ this.diff = {};
14
16
  }
15
17
  /**
16
18
  * Returns a literal that contains only the keys whose values have been changed by this call or any previous calls.
@@ -24,27 +26,27 @@ export class CumulativeChanges {
24
26
  */
25
27
  differences(newValues) {
26
28
  Object.keys(newValues).forEach(key => {
27
- if (key in this.#diff ||
29
+ if (key in this.diff ||
28
30
  (newValues[key] !== undefined &&
29
- (!(key in this.#original) || !isEqual(this.#original[key], newValues[key])))) {
30
- this.#diff[key] = newValues[key];
31
+ (!(key in this.original) || !isEqual(this.original[key], newValues[key])))) {
32
+ this.diff[key] = newValues[key];
31
33
  }
32
34
  });
33
- return Object.keys(this.#diff).length ? this.#diff : null;
35
+ return Object.keys(this.diff).length ? this.diff : null;
34
36
  }
35
37
  /**
36
38
  * Commit the given new values, so they are not treated as differences anymore.
37
39
  */
38
40
  commit(newValues) {
39
- this.#original = {
40
- ...this.#original,
41
+ this.original = {
42
+ ...this.original,
41
43
  ...cloneDeep(newValues),
42
44
  };
43
45
  Object.keys(newValues).forEach(key => {
44
- if (key in this.#diff && isEqual(this.#diff[key], newValues[key])) {
45
- delete this.#diff[key];
46
+ if (key in this.diff && isEqual(this.diff[key], newValues[key])) {
47
+ delete this.diff[key];
46
48
  }
47
49
  });
48
50
  }
49
51
  }
50
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VtdWxhdGl2ZS1jaGFuZ2VzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL2NsYXNzZXMvY3VtdWxhdGl2ZS1jaGFuZ2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsT0FBTyxFQUFDLE1BQU0sV0FBVyxDQUFDO0FBSTdDOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGlCQUFpQjtJQUMxQixTQUFTLEdBQU0sRUFBTyxDQUFDO0lBQ3ZCLEtBQUssR0FBZSxFQUFFLENBQUM7SUFFdkI7O09BRUc7SUFDSSxVQUFVLENBQUMsY0FBMkI7UUFDekMsSUFBSSxDQUFDLFNBQVMsR0FBRyxTQUFTLENBQUMsY0FBYyxDQUFDLENBQUM7UUFDM0MsSUFBSSxDQUFDLEtBQUssR0FBRyxFQUFFLENBQUM7SUFDcEIsQ0FBQztJQUVEOzs7Ozs7Ozs7T0FTRztJQUNJLFdBQVcsQ0FBQyxTQUEwQjtRQUN6QyxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsRUFBRTtZQUNqQyxJQUNJLEdBQUcsSUFBSSxJQUFJLENBQUMsS0FBSztnQkFDakIsQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLEtBQUssU0FBUztvQkFDekIsQ0FBQyxDQUFDLENBQUMsR0FBRyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxFQUFFLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFDbEYsQ0FBQztnQkFDRSxJQUFJLENBQUMsS0FBYSxDQUFDLEdBQUcsQ0FBQyxHQUFHLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQztZQUM5QyxDQUFDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7UUFFSCxPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0lBQzlELENBQUM7SUFFRDs7T0FFRztJQUNJLE1BQU0sQ0FBQyxTQUEwQjtRQUNwQyxJQUFJLENBQUMsU0FBUyxHQUFHO1lBQ2IsR0FBRyxJQUFJLENBQUMsU0FBUztZQUNqQixHQUFHLFNBQVMsQ0FBQyxTQUFTLENBQUM7U0FDMUIsQ0FBQztRQUVGLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxFQUFFO1lBQ2pDLElBQUksR0FBRyxJQUFJLElBQUksQ0FBQyxLQUFLLElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEVBQUUsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsQ0FBQztnQkFDaEUsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQzNCLENBQUM7UUFDTCxDQUFDLENBQUMsQ0FBQztJQUNQLENBQUM7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Y2xvbmVEZWVwLCBpc0VxdWFsfSBmcm9tICdsb2Rhc2gtZXMnO1xuaW1wb3J0IHtMaXRlcmFsfSBmcm9tICcuLi90eXBlcy90eXBlcyc7XG5pbXBvcnQge1JlYWRvbmx5RGVlcH0gZnJvbSAndHlwZS1mZXN0JztcblxuLyoqXG4gKiBDdW11bGF0ZSBhbGwgY2hhbmdlcyBtYWRlIHRvIGFuIG9iamVjdCBvdmVyIHRpbWVcbiAqL1xuZXhwb3J0IGNsYXNzIEN1bXVsYXRpdmVDaGFuZ2VzPFQgZXh0ZW5kcyBMaXRlcmFsPiB7XG4gICAgI29yaWdpbmFsOiBUID0ge30gYXMgVDtcbiAgICAjZGlmZjogUGFydGlhbDxUPiA9IHt9O1xuXG4gICAgLyoqXG4gICAgICogSW5pdGlhbGl6ZSB0aGUgb3JpZ2luYWwgdmFsdWVzLCBzaG91bGQgYmUgY2FsbGVkIGV4YWN0bHkgb25lIHRpbWUgcGVyIGluc3RhbmNlXG4gICAgICovXG4gICAgcHVibGljIGluaXRpYWxpemUob3JpZ2luYWxWYWx1ZXM6IFJlYWRvbmx5PFQ+KTogdm9pZCB7XG4gICAgICAgIHRoaXMuI29yaWdpbmFsID0gY2xvbmVEZWVwKG9yaWdpbmFsVmFsdWVzKTtcbiAgICAgICAgdGhpcy4jZGlmZiA9IHt9O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybnMgYSBsaXRlcmFsIHRoYXQgY29udGFpbnMgb25seSB0aGUga2V5cyB3aG9zZSB2YWx1ZXMgaGF2ZSBiZWVuIGNoYW5nZWQgYnkgdGhpcyBjYWxsIG9yIGFueSBwcmV2aW91cyBjYWxscy5cbiAgICAgKlxuICAgICAqIEVnOlxuICAgICAqXG4gICAgICogYGBgdHNcbiAgICAgKiBjaGFuZ2VzLmluaXRpYWxpemUoe2E6IDEsIGI6IDJ9KTtcbiAgICAgKiBjaGFuZ2VzLmRpZmZlcmVuY2VzKHthOiAxLCBiOiAzfSk7IC8vID0+IHtiOiAzfVxuICAgICAqIGBgYFxuICAgICAqL1xuICAgIHB1YmxpYyBkaWZmZXJlbmNlcyhuZXdWYWx1ZXM6IFJlYWRvbmx5RGVlcDxUPik6IFBhcnRpYWw8VD4gfCBudWxsIHtcbiAgICAgICAgT2JqZWN0LmtleXMobmV3VmFsdWVzKS5mb3JFYWNoKGtleSA9PiB7XG4gICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgICAga2V5IGluIHRoaXMuI2RpZmYgfHxcbiAgICAgICAgICAgICAgICAobmV3VmFsdWVzW2tleV0gIT09IHVuZGVmaW5lZCAmJlxuICAgICAgICAgICAgICAgICAgICAoIShrZXkgaW4gdGhpcy4jb3JpZ2luYWwpIHx8ICFpc0VxdWFsKHRoaXMuI29yaWdpbmFsW2tleV0sIG5ld1ZhbHVlc1trZXldKSkpXG4gICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgICAodGhpcy4jZGlmZiBhcyBhbnkpW2tleV0gPSBuZXdWYWx1ZXNba2V5XTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgcmV0dXJuIE9iamVjdC5rZXlzKHRoaXMuI2RpZmYpLmxlbmd0aCA/IHRoaXMuI2RpZmYgOiBudWxsO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENvbW1pdCB0aGUgZ2l2ZW4gbmV3IHZhbHVlcywgc28gdGhleSBhcmUgbm90IHRyZWF0ZWQgYXMgZGlmZmVyZW5jZXMgYW55bW9yZS5cbiAgICAgKi9cbiAgICBwdWJsaWMgY29tbWl0KG5ld1ZhbHVlczogUmVhZG9ubHlEZWVwPFQ+KTogdm9pZCB7XG4gICAgICAgIHRoaXMuI29yaWdpbmFsID0ge1xuICAgICAgICAgICAgLi4udGhpcy4jb3JpZ2luYWwsXG4gICAgICAgICAgICAuLi5jbG9uZURlZXAobmV3VmFsdWVzKSxcbiAgICAgICAgfTtcblxuICAgICAgICBPYmplY3Qua2V5cyhuZXdWYWx1ZXMpLmZvckVhY2goa2V5ID0+IHtcbiAgICAgICAgICAgIGlmIChrZXkgaW4gdGhpcy4jZGlmZiAmJiBpc0VxdWFsKHRoaXMuI2RpZmZba2V5XSwgbmV3VmFsdWVzW2tleV0pKSB7XG4gICAgICAgICAgICAgICAgZGVsZXRlIHRoaXMuI2RpZmZba2V5XTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG4gICAgfVxufVxuIl19
52
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VtdWxhdGl2ZS1jaGFuZ2VzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL2NsYXNzZXMvY3VtdWxhdGl2ZS1jaGFuZ2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsT0FBTyxFQUFDLE1BQU0sV0FBVyxDQUFDO0FBSTdDOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGlCQUFpQjtJQUE5QjtRQUNZLGFBQVEsR0FBTSxFQUFPLENBQUM7UUFDdEIsU0FBSSxHQUFlLEVBQUUsQ0FBQztJQWlEbEMsQ0FBQztJQS9DRzs7T0FFRztJQUNJLFVBQVUsQ0FBQyxjQUEyQjtRQUN6QyxJQUFJLENBQUMsUUFBUSxHQUFHLFNBQVMsQ0FBQyxjQUFjLENBQUMsQ0FBQztRQUMxQyxJQUFJLENBQUMsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUNuQixDQUFDO0lBRUQ7Ozs7Ozs7OztPQVNHO0lBQ0ksV0FBVyxDQUFDLFNBQTBCO1FBQ3pDLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxFQUFFO1lBQ2pDLElBQ0ksR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJO2dCQUNoQixDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsS0FBSyxTQUFTO29CQUN6QixDQUFDLENBQUMsQ0FBQyxHQUFHLElBQUksSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLEVBQUUsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUNoRixDQUFDO2dCQUNFLElBQUksQ0FBQyxJQUFZLENBQUMsR0FBRyxDQUFDLEdBQUcsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQzdDLENBQUM7UUFDTCxDQUFDLENBQUMsQ0FBQztRQUVILE9BQU8sTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7SUFDNUQsQ0FBQztJQUVEOztPQUVHO0lBQ0ksTUFBTSxDQUFDLFNBQTBCO1FBQ3BDLElBQUksQ0FBQyxRQUFRLEdBQUc7WUFDWixHQUFHLElBQUksQ0FBQyxRQUFRO1lBQ2hCLEdBQUcsU0FBUyxDQUFDLFNBQVMsQ0FBQztTQUMxQixDQUFDO1FBRUYsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLEVBQUU7WUFDakMsSUFBSSxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksSUFBSSxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxTQUFTLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDO2dCQUM5RCxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDMUIsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ1AsQ0FBQztDQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtjbG9uZURlZXAsIGlzRXF1YWx9IGZyb20gJ2xvZGFzaC1lcyc7XG5pbXBvcnQge0xpdGVyYWx9IGZyb20gJy4uL3R5cGVzL3R5cGVzJztcbmltcG9ydCB7UmVhZG9ubHlEZWVwfSBmcm9tICd0eXBlLWZlc3QnO1xuXG4vKipcbiAqIEN1bXVsYXRlIGFsbCBjaGFuZ2VzIG1hZGUgdG8gYW4gb2JqZWN0IG92ZXIgdGltZVxuICovXG5leHBvcnQgY2xhc3MgQ3VtdWxhdGl2ZUNoYW5nZXM8VCBleHRlbmRzIExpdGVyYWw+IHtcbiAgICBwcml2YXRlIG9yaWdpbmFsOiBUID0ge30gYXMgVDtcbiAgICBwcml2YXRlIGRpZmY6IFBhcnRpYWw8VD4gPSB7fTtcblxuICAgIC8qKlxuICAgICAqIEluaXRpYWxpemUgdGhlIG9yaWdpbmFsIHZhbHVlcywgc2hvdWxkIGJlIGNhbGxlZCBleGFjdGx5IG9uZSB0aW1lIHBlciBpbnN0YW5jZVxuICAgICAqL1xuICAgIHB1YmxpYyBpbml0aWFsaXplKG9yaWdpbmFsVmFsdWVzOiBSZWFkb25seTxUPik6IHZvaWQge1xuICAgICAgICB0aGlzLm9yaWdpbmFsID0gY2xvbmVEZWVwKG9yaWdpbmFsVmFsdWVzKTtcbiAgICAgICAgdGhpcy5kaWZmID0ge307XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogUmV0dXJucyBhIGxpdGVyYWwgdGhhdCBjb250YWlucyBvbmx5IHRoZSBrZXlzIHdob3NlIHZhbHVlcyBoYXZlIGJlZW4gY2hhbmdlZCBieSB0aGlzIGNhbGwgb3IgYW55IHByZXZpb3VzIGNhbGxzLlxuICAgICAqXG4gICAgICogRWc6XG4gICAgICpcbiAgICAgKiBgYGB0c1xuICAgICAqIGNoYW5nZXMuaW5pdGlhbGl6ZSh7YTogMSwgYjogMn0pO1xuICAgICAqIGNoYW5nZXMuZGlmZmVyZW5jZXMoe2E6IDEsIGI6IDN9KTsgLy8gPT4ge2I6IDN9XG4gICAgICogYGBgXG4gICAgICovXG4gICAgcHVibGljIGRpZmZlcmVuY2VzKG5ld1ZhbHVlczogUmVhZG9ubHlEZWVwPFQ+KTogUGFydGlhbDxUPiB8IG51bGwge1xuICAgICAgICBPYmplY3Qua2V5cyhuZXdWYWx1ZXMpLmZvckVhY2goa2V5ID0+IHtcbiAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgICBrZXkgaW4gdGhpcy5kaWZmIHx8XG4gICAgICAgICAgICAgICAgKG5ld1ZhbHVlc1trZXldICE9PSB1bmRlZmluZWQgJiZcbiAgICAgICAgICAgICAgICAgICAgKCEoa2V5IGluIHRoaXMub3JpZ2luYWwpIHx8ICFpc0VxdWFsKHRoaXMub3JpZ2luYWxba2V5XSwgbmV3VmFsdWVzW2tleV0pKSlcbiAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICAgICh0aGlzLmRpZmYgYXMgYW55KVtrZXldID0gbmV3VmFsdWVzW2tleV07XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuXG4gICAgICAgIHJldHVybiBPYmplY3Qua2V5cyh0aGlzLmRpZmYpLmxlbmd0aCA/IHRoaXMuZGlmZiA6IG51bGw7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ29tbWl0IHRoZSBnaXZlbiBuZXcgdmFsdWVzLCBzbyB0aGV5IGFyZSBub3QgdHJlYXRlZCBhcyBkaWZmZXJlbmNlcyBhbnltb3JlLlxuICAgICAqL1xuICAgIHB1YmxpYyBjb21taXQobmV3VmFsdWVzOiBSZWFkb25seURlZXA8VD4pOiB2b2lkIHtcbiAgICAgICAgdGhpcy5vcmlnaW5hbCA9IHtcbiAgICAgICAgICAgIC4uLnRoaXMub3JpZ2luYWwsXG4gICAgICAgICAgICAuLi5jbG9uZURlZXAobmV3VmFsdWVzKSxcbiAgICAgICAgfTtcblxuICAgICAgICBPYmplY3Qua2V5cyhuZXdWYWx1ZXMpLmZvckVhY2goa2V5ID0+IHtcbiAgICAgICAgICAgIGlmIChrZXkgaW4gdGhpcy5kaWZmICYmIGlzRXF1YWwodGhpcy5kaWZmW2tleV0sIG5ld1ZhbHVlc1trZXldKSkge1xuICAgICAgICAgICAgICAgIGRlbGV0ZSB0aGlzLmRpZmZba2V5XTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG4gICAgfVxufVxuIl19
@@ -23,7 +23,6 @@ function getTabId(tab) {
23
23
  * </mat-tab-group>
24
24
  */
25
25
  export class NaturalLinkableTabDirective extends NaturalAbstractController {
26
- #isLoadingRouteConfig;
27
26
  constructor(component, route, router) {
28
27
  super();
29
28
  this.component = component;
@@ -33,13 +32,13 @@ export class NaturalLinkableTabDirective extends NaturalAbstractController {
33
32
  * If false, disables the persistent navigation
34
33
  */
35
34
  this.naturalLinkableTab = true;
36
- this.#isLoadingRouteConfig = false;
35
+ this.isLoadingRouteConfig = false;
37
36
  router.events.pipe(takeUntilDestroyed()).subscribe(event => {
38
37
  if (event instanceof RouteConfigLoadStart) {
39
- this.#isLoadingRouteConfig = true;
38
+ this.isLoadingRouteConfig = true;
40
39
  }
41
40
  else if (event instanceof RouteConfigLoadEnd) {
42
- this.#isLoadingRouteConfig = false;
41
+ this.isLoadingRouteConfig = false;
43
42
  }
44
43
  });
45
44
  }
@@ -59,7 +58,7 @@ export class NaturalLinkableTabDirective extends NaturalAbstractController {
59
58
  });
60
59
  // When mat-tab-groups selected tab change, update url
61
60
  this.component.selectedTabChange.pipe(takeUntil(this.ngUnsubscribe)).subscribe(event => {
62
- if (this.#isLoadingRouteConfig) {
61
+ if (this.isLoadingRouteConfig) {
63
62
  return;
64
63
  }
65
64
  const activatedTabName = getTabId(event.tab);
@@ -92,4 +91,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.5", ngImpor
92
91
  }], ctorParameters: () => [{ type: i1.MatTabGroup }, { type: i2.ActivatedRoute }, { type: i2.Router }], propDecorators: { naturalLinkableTab: [{
93
92
  type: Input
94
93
  }] } });
95
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linkable-tab.directive.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/common/directives/linkable-tab.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAgB,SAAS,EAAE,KAAK,EAAC,MAAM,eAAe,CAAC;AAE9D,OAAO,EAAiB,kBAAkB,EAAE,oBAAoB,EAAS,MAAM,iBAAiB,CAAC;AACjG,OAAO,EAAC,KAAK,EAAC,MAAM,WAAW,CAAC;AAChC,OAAO,EAAC,SAAS,EAAC,MAAM,gBAAgB,CAAC;AACzC,OAAO,EAAC,yBAAyB,EAAC,MAAM,sCAAsC,CAAC;AAC/E,OAAO,EAAC,kBAAkB,EAAC,MAAM,4BAA4B,CAAC;;;;AAE9D;;GAEG;AACH,SAAS,QAAQ,CAAC,GAAW;IACzB,OAAO,GAAG,CAAC,OAAO,EAAE,gBAAgB,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE,IAAI,EAAE,CAAC;AACxE,CAAC;AAED;;;;;;;;GAQG;AAKH,MAAM,OAAO,2BAA4B,SAAQ,yBAAyB;IAKtE,qBAAqB,CAAS;IAE9B,YACqB,SAAsB,EACtB,KAAqB,EACrB,MAAc;QAE/B,KAAK,EAAE,CAAC;QAJS,cAAS,GAAT,SAAS,CAAa;QACtB,UAAK,GAAL,KAAK,CAAgB;QACrB,WAAM,GAAN,MAAM,CAAQ;QATnC;;WAEG;QACa,uBAAkB,GAAiB,IAAI,CAAC;QACxD,0BAAqB,GAAG,KAAK,CAAC;QAS1B,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YACvD,IAAI,KAAK,YAAY,oBAAoB,EAAE,CAAC;gBACxC,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;YACtC,CAAC;iBAAM,IAAI,KAAK,YAAY,kBAAkB,EAAE,CAAC;gBAC7C,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC;YACvC,CAAC;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEM,eAAe;QAClB,IAAI,IAAI,CAAC,kBAAkB,KAAK,KAAK,EAAE,CAAC;YACpC,OAAO;QACX,CAAC;QAED,gEAAgE;QAChE,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE;YACzE,4CAA4C;YAC5C,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;YAE5C,oDAAoD;YACpD,6EAA6E;YAC7E,IAAI,QAAQ,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;gBAC7B,IAAI,CAAC,SAAS,CAAC,aAAa,GAAG,QAAQ,CAAC;YAC5C,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,sDAAsD;QACtD,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YACnF,IAAI,IAAI,CAAC,qBAAqB,EAAE,CAAC;gBAC7B,OAAO;YACX,CAAC;YAED,MAAM,gBAAgB,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC7C,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC;YACzC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACnB,uGAAuG;gBACvG,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;YAC9E,CAAC;YAED,wFAAwF;YACxF,MAAM,MAAM,GAAG,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC;YAE/D,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,GAAG,EAAE,MAAM,CAAC,EAAE;gBAChC,UAAU,EAAE,IAAI,CAAC,KAAK;gBACtB,mBAAmB,EAAE,UAAU;gBAC/B,QAAQ,EAAE,gBAAgB,EAAE,MAAM,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,SAAS;aACpE,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,WAAW,CAAC,QAAuB;QACvC,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,QAAQ,KAAK,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC;IACvF,CAAC;8GAlEQ,2BAA2B;kGAA3B,2BAA2B;;2FAA3B,2BAA2B;kBAJvC,SAAS;mBAAC;oBACP,QAAQ,EAAE,mCAAmC;oBAC7C,UAAU,EAAE,IAAI;iBACnB;kIAKmB,kBAAkB;sBAAjC,KAAK","sourcesContent":["import {AfterViewInit, Directive, Input} from '@angular/core';\nimport {MatTab, MatTabGroup} from '@angular/material/tabs';\nimport {ActivatedRoute, RouteConfigLoadEnd, RouteConfigLoadStart, Router} from '@angular/router';\nimport {clone} from 'lodash-es';\nimport {takeUntil} from 'rxjs/operators';\nimport {NaturalAbstractController} from '../../../classes/abstract-controller';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * Returns an identifier for the tab\n */\nfunction getTabId(tab: MatTab): string {\n    return tab.content?.viewContainerRef.element.nativeElement.id ?? '';\n}\n\n/**\n * Usage :\n *\n * <mat-tab-group [naturalLinkableTab]=\"!isPanel\">\n *     <mat-tab label=\"Tab 1\">Tab 1</mat-tab> // First tab doesn't need id. This keeps url clean on default one\n *     <mat-tab label=\"Tab 2\" id=\"second-tab\">Tab 2</mat-tab>\n *     ...\n * </mat-tab-group>\n */\n@Directive({\n    selector: 'mat-tab-group[naturalLinkableTab]',\n    standalone: true,\n})\nexport class NaturalLinkableTabDirective extends NaturalAbstractController implements AfterViewInit {\n    /**\n     * If false, disables the persistent navigation\n     */\n    @Input() public naturalLinkableTab: boolean | '' = true;\n    #isLoadingRouteConfig = false;\n\n    public constructor(\n        private readonly component: MatTabGroup,\n        private readonly route: ActivatedRoute,\n        private readonly router: Router,\n    ) {\n        super();\n\n        router.events.pipe(takeUntilDestroyed()).subscribe(event => {\n            if (event instanceof RouteConfigLoadStart) {\n                this.#isLoadingRouteConfig = true;\n            } else if (event instanceof RouteConfigLoadEnd) {\n                this.#isLoadingRouteConfig = false;\n            }\n        });\n    }\n\n    public ngAfterViewInit(): void {\n        if (this.naturalLinkableTab === false) {\n            return;\n        }\n\n        // When url params change, update the mat-tab-group selected tab\n        this.route.fragment.pipe(takeUntil(this.ngUnsubscribe)).subscribe(fragment => {\n            // Get index of tab that matches wanted name\n            const tabIndex = this.getTabIndex(fragment);\n\n            // If tab index is valid (>= 0) go to given fragment\n            // If there is no fragment at all, go to first tab (index is -1 in this case)\n            if (tabIndex >= 0 || !fragment) {\n                this.component.selectedIndex = tabIndex;\n            }\n        });\n\n        // When mat-tab-groups selected tab change, update url\n        this.component.selectedTabChange.pipe(takeUntil(this.ngUnsubscribe)).subscribe(event => {\n            if (this.#isLoadingRouteConfig) {\n                return;\n            }\n\n            const activatedTabName = getTabId(event.tab);\n            const segments = this.route.snapshot.url;\n            if (!segments.length) {\n                // This should never happen in normal usage, because it would means there is no route at all in the app\n                throw new Error('Cannot update URL for tabs without any segments in URL');\n            }\n\n            // Get url matrix params (/segment;matrix=param) only without route params (segment/:id)\n            const params = clone(segments[segments.length - 1].parameters);\n\n            this.router.navigate(['.', params], {\n                relativeTo: this.route,\n                queryParamsHandling: 'preserve',\n                fragment: activatedTabName?.length ? activatedTabName : undefined,\n            });\n        });\n    }\n\n    private getTabIndex(fragment: string | null): number {\n        return this.component._tabs.toArray().findIndex(tab => fragment === getTabId(tab));\n    }\n}\n"]}
94
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linkable-tab.directive.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/common/directives/linkable-tab.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Directive, Input} from '@angular/core';\nimport {MatTab, MatTabGroup} from '@angular/material/tabs';\nimport {ActivatedRoute, RouteConfigLoadEnd, RouteConfigLoadStart, Router} from '@angular/router';\nimport {clone} from 'lodash-es';\nimport {takeUntil} from 'rxjs/operators';\nimport {NaturalAbstractController} from '../../../classes/abstract-controller';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * Returns an identifier for the tab\n */\nfunction getTabId(tab: MatTab): string {\n    return tab.content?.viewContainerRef.element.nativeElement.id ?? '';\n}\n\n/**\n * Usage :\n *\n * <mat-tab-group [naturalLinkableTab]=\"!isPanel\">\n *     <mat-tab label=\"Tab 1\">Tab 1</mat-tab> // First tab doesn't need id. This keeps url clean on default one\n *     <mat-tab label=\"Tab 2\" id=\"second-tab\">Tab 2</mat-tab>\n *     ...\n * </mat-tab-group>\n */\n@Directive({\n    selector: 'mat-tab-group[naturalLinkableTab]',\n    standalone: true,\n})\nexport class NaturalLinkableTabDirective extends NaturalAbstractController implements AfterViewInit {\n    /**\n     * If false, disables the persistent navigation\n     */\n    @Input() public naturalLinkableTab: boolean | '' = true;\n    private isLoadingRouteConfig = false;\n\n    public constructor(\n        private readonly component: MatTabGroup,\n        private readonly route: ActivatedRoute,\n        private readonly router: Router,\n    ) {\n        super();\n\n        router.events.pipe(takeUntilDestroyed()).subscribe(event => {\n            if (event instanceof RouteConfigLoadStart) {\n                this.isLoadingRouteConfig = true;\n            } else if (event instanceof RouteConfigLoadEnd) {\n                this.isLoadingRouteConfig = false;\n            }\n        });\n    }\n\n    public ngAfterViewInit(): void {\n        if (this.naturalLinkableTab === false) {\n            return;\n        }\n\n        // When url params change, update the mat-tab-group selected tab\n        this.route.fragment.pipe(takeUntil(this.ngUnsubscribe)).subscribe(fragment => {\n            // Get index of tab that matches wanted name\n            const tabIndex = this.getTabIndex(fragment);\n\n            // If tab index is valid (>= 0) go to given fragment\n            // If there is no fragment at all, go to first tab (index is -1 in this case)\n            if (tabIndex >= 0 || !fragment) {\n                this.component.selectedIndex = tabIndex;\n            }\n        });\n\n        // When mat-tab-groups selected tab change, update url\n        this.component.selectedTabChange.pipe(takeUntil(this.ngUnsubscribe)).subscribe(event => {\n            if (this.isLoadingRouteConfig) {\n                return;\n            }\n\n            const activatedTabName = getTabId(event.tab);\n            const segments = this.route.snapshot.url;\n            if (!segments.length) {\n                // This should never happen in normal usage, because it would means there is no route at all in the app\n                throw new Error('Cannot update URL for tabs without any segments in URL');\n            }\n\n            // Get url matrix params (/segment;matrix=param) only without route params (segment/:id)\n            const params = clone(segments[segments.length - 1].parameters);\n\n            this.router.navigate(['.', params], {\n                relativeTo: this.route,\n                queryParamsHandling: 'preserve',\n                fragment: activatedTabName?.length ? activatedTabName : undefined,\n            });\n        });\n    }\n\n    private getTabIndex(fragment: string | null): number {\n        return this.component._tabs.toArray().findIndex(tab => fragment === getTabId(tab));\n    }\n}\n"]}
@@ -47,11 +47,11 @@ export class NaturalDetailHeaderComponent {
47
47
  return this.getRootLink().concat([id]);
48
48
  }
49
49
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: NaturalDetailHeaderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
50
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.5", type: NaturalDetailHeaderComponent, isStandalone: true, selector: "natural-detail-header", inputs: { currentBaseUrl: "currentBaseUrl", isPanel: "isPanel", icon: "icon", label: "label", rootLabel: "rootLabel", newLabel: "newLabel", model: "model", breadcrumbs: "breadcrumbs", listRoute: "listRoute", listFragment: "listFragment", link: "link" }, ngImport: i0, template: "<div class=\"breadcrumb\">\n @if (rootLabel || label) {\n <a [routerLink]=\"isPanel ? [] : getRootLink()\" [fragment]=\"listFragment\" color=\"primary\" mat-button>{{\n rootLabel || label\n }}</a>\n }\n @for (parent of breadcrumbs; track parent) {\n /\n <a [routerLink]=\"isPanel ? [] : getLink(parent.id)\" color=\"primary\" mat-button>\n {{ parent?.fullName || parent?.name }}</a\n >\n }\n</div>\n\n<div class=\"body\">\n @if (icon) {\n <div style=\"width: 30px\">\n <mat-icon [naturalIcon]=\"icon\" />\n </div>\n }\n @if (!model.id) {\n <div class=\"mat-headline-5 no-margin newLabel\">{{ newLabel }}</div>\n }\n @if (model.id) {\n <div class=\"mat-headline-5 no-margin label\">{{ model.name || model.fullName || label }}</div>\n }\n <div>\n <ng-content />\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host .breadcrumb,:host .body{display:flex;flex-direction:row;align-items:center}:host .breadcrumb{position:relative;top:5px}:host .body{min-height:40px}:host .body>*:not(:last-child){margin-right:5px}:host .body .label,:host .body .newLabel{flex:1}@media screen and (max-width: 600px){:host .body{flex-direction:column;align-items:flex-start}:host .body>*:not(:last-child){margin-bottom:10px!important}:host .body mat-icon{display:none}}\n"], dependencies: [{ kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i1.MatAnchor, selector: "a[mat-button], a[mat-raised-button], a[mat-flat-button], a[mat-stroked-button]", exportAs: ["matButton", "matAnchor"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }] }); }
50
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.5", type: NaturalDetailHeaderComponent, isStandalone: true, selector: "natural-detail-header", inputs: { currentBaseUrl: "currentBaseUrl", isPanel: "isPanel", icon: "icon", label: "label", rootLabel: "rootLabel", newLabel: "newLabel", model: "model", breadcrumbs: "breadcrumbs", listRoute: "listRoute", listFragment: "listFragment", link: "link" }, ngImport: i0, template: "<div class=\"breadcrumb\">\n @if (rootLabel || label) {\n <a [routerLink]=\"isPanel ? [] : getRootLink()\" [fragment]=\"listFragment\" color=\"primary\" mat-button>{{\n rootLabel || label\n }}</a>\n }\n @for (parent of breadcrumbs; track parent) {\n /\n <a [routerLink]=\"isPanel ? [] : getLink(parent.id)\" color=\"primary\" mat-button>\n {{ parent?.fullName || parent?.name }}</a\n >\n }\n</div>\n\n<div class=\"body\">\n @if (icon) {\n <div style=\"width: 30px\">\n <mat-icon [naturalIcon]=\"icon\" />\n </div>\n }\n @if (!model.id) {\n <div class=\"mat-headline-5 nat-no-margin newLabel\">{{ newLabel }}</div>\n }\n @if (model.id) {\n <div class=\"mat-headline-5 nat-no-margin label\">{{ model.name || model.fullName || label }}</div>\n }\n <div>\n <ng-content />\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host .breadcrumb,:host .body{display:flex;flex-direction:row;align-items:center}:host .breadcrumb{position:relative;top:5px}:host .body{min-height:40px}:host .body>*:not(:last-child){margin-right:5px}:host .body .label,:host .body .newLabel{flex:1}@media screen and (max-width: 600px){:host .body{flex-direction:column;align-items:flex-start}:host .body>*:not(:last-child){margin-bottom:10px!important}:host .body mat-icon{display:none}}\n"], dependencies: [{ kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i1.MatAnchor, selector: "a[mat-button], a[mat-raised-button], a[mat-flat-button], a[mat-stroked-button]", exportAs: ["matButton", "matAnchor"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }] }); }
51
51
  }
52
52
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: NaturalDetailHeaderComponent, decorators: [{
53
53
  type: Component,
54
- args: [{ selector: 'natural-detail-header', standalone: true, imports: [MatButtonModule, RouterLink, MatIconModule, NaturalIconDirective], template: "<div class=\"breadcrumb\">\n @if (rootLabel || label) {\n <a [routerLink]=\"isPanel ? [] : getRootLink()\" [fragment]=\"listFragment\" color=\"primary\" mat-button>{{\n rootLabel || label\n }}</a>\n }\n @for (parent of breadcrumbs; track parent) {\n /\n <a [routerLink]=\"isPanel ? [] : getLink(parent.id)\" color=\"primary\" mat-button>\n {{ parent?.fullName || parent?.name }}</a\n >\n }\n</div>\n\n<div class=\"body\">\n @if (icon) {\n <div style=\"width: 30px\">\n <mat-icon [naturalIcon]=\"icon\" />\n </div>\n }\n @if (!model.id) {\n <div class=\"mat-headline-5 no-margin newLabel\">{{ newLabel }}</div>\n }\n @if (model.id) {\n <div class=\"mat-headline-5 no-margin label\">{{ model.name || model.fullName || label }}</div>\n }\n <div>\n <ng-content />\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host .breadcrumb,:host .body{display:flex;flex-direction:row;align-items:center}:host .breadcrumb{position:relative;top:5px}:host .body{min-height:40px}:host .body>*:not(:last-child){margin-right:5px}:host .body .label,:host .body .newLabel{flex:1}@media screen and (max-width: 600px){:host .body{flex-direction:column;align-items:flex-start}:host .body>*:not(:last-child){margin-bottom:10px!important}:host .body mat-icon{display:none}}\n"] }]
54
+ args: [{ selector: 'natural-detail-header', standalone: true, imports: [MatButtonModule, RouterLink, MatIconModule, NaturalIconDirective], template: "<div class=\"breadcrumb\">\n @if (rootLabel || label) {\n <a [routerLink]=\"isPanel ? [] : getRootLink()\" [fragment]=\"listFragment\" color=\"primary\" mat-button>{{\n rootLabel || label\n }}</a>\n }\n @for (parent of breadcrumbs; track parent) {\n /\n <a [routerLink]=\"isPanel ? [] : getLink(parent.id)\" color=\"primary\" mat-button>\n {{ parent?.fullName || parent?.name }}</a\n >\n }\n</div>\n\n<div class=\"body\">\n @if (icon) {\n <div style=\"width: 30px\">\n <mat-icon [naturalIcon]=\"icon\" />\n </div>\n }\n @if (!model.id) {\n <div class=\"mat-headline-5 nat-no-margin newLabel\">{{ newLabel }}</div>\n }\n @if (model.id) {\n <div class=\"mat-headline-5 nat-no-margin label\">{{ model.name || model.fullName || label }}</div>\n }\n <div>\n <ng-content />\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host .breadcrumb,:host .body{display:flex;flex-direction:row;align-items:center}:host .breadcrumb{position:relative;top:5px}:host .body{min-height:40px}:host .body>*:not(:last-child){margin-right:5px}:host .body .label,:host .body .newLabel{flex:1}@media screen and (max-width: 600px){:host .body{flex-direction:column;align-items:flex-start}:host .body>*:not(:last-child){margin-bottom:10px!important}:host .body mat-icon{display:none}}\n"] }]
55
55
  }], propDecorators: { currentBaseUrl: [{
56
56
  type: Input
57
57
  }], isPanel: [{
@@ -76,4 +76,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.5", ngImpor
76
76
  }], link: [{
77
77
  type: Input
78
78
  }] } });
79
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLWhlYWRlci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9kZXRhaWwtaGVhZGVyL2RldGFpbC1oZWFkZXIuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZGV0YWlsLWhlYWRlci9kZXRhaWwtaGVhZGVyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsS0FBSyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRS9DLE9BQU8sRUFBQyxvQkFBb0IsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQzVELE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSx3QkFBd0IsQ0FBQztBQUNyRCxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0saUJBQWlCLENBQUM7QUFDM0MsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLDBCQUEwQixDQUFDOzs7O0FBU3pELE1BQU0sT0FBTyw0QkFBNEI7SUFQekM7UUFhSTs7V0FFRztRQUNhLFlBQU8sR0FBRyxLQUFLLENBQUM7UUFFaEM7O1dBRUc7UUFDYSxTQUFJLEdBQUcsRUFBRSxDQUFDO1FBQzFCOzs7O1dBSUc7UUFDYSxVQUFLLEdBQUcsRUFBRSxDQUFDO1FBRTNCOzs7O1dBSUc7UUFDYSxjQUFTLEdBQUcsRUFBRSxDQUFDO1FBRS9COzs7O1dBSUc7UUFDYSxhQUFRLEdBQUcsRUFBRSxDQUFDO1FBRWQsZ0JBQVcsR0FBYyxFQUFFLENBQUM7UUFDNUIsY0FBUyxHQUFVLEVBQUUsQ0FBQztLQWV6QztJQVhVLFdBQVc7UUFDZCxPQUFPLENBQUMsSUFBSSxDQUFDLGNBQWMsSUFBSSxHQUFHLENBQUMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQy9ELENBQUM7SUFFTSxPQUFPLENBQUMsRUFBVTtRQUNyQixJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNaLE9BQU8sSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDcEQsQ0FBQztRQUVELE9BQU8sSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDM0MsQ0FBQzs4R0FuRFEsNEJBQTRCO2tHQUE1Qiw0QkFBNEIsK1VDZHpDLHU1QkE4QkEsMGhCRGxCYyxlQUFlLDRNQUFFLFVBQVUsbU9BQUUsYUFBYSxvTEFBRSxvQkFBb0I7OzJGQUVqRSw0QkFBNEI7a0JBUHhDLFNBQVM7K0JBQ0ksdUJBQXVCLGNBR3JCLElBQUksV0FDUCxDQUFDLGVBQWUsRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLG9CQUFvQixDQUFDOzhCQU0zRCxjQUFjO3NCQUE3QixLQUFLO2dCQUtVLE9BQU87c0JBQXRCLEtBQUs7Z0JBS1UsSUFBSTtzQkFBbkIsS0FBSztnQkFNVSxLQUFLO3NCQUFwQixLQUFLO2dCQU9VLFNBQVM7c0JBQXhCLEtBQUs7Z0JBT1UsUUFBUTtzQkFBdkIsS0FBSztnQkFDMEIsS0FBSztzQkFBcEMsS0FBSzt1QkFBQyxFQUFDLFFBQVEsRUFBRSxJQUFJLEVBQUM7Z0JBQ1AsV0FBVztzQkFBMUIsS0FBSztnQkFDVSxTQUFTO3NCQUF4QixLQUFLO2dCQUNVLFlBQVk7c0JBQTNCLEtBQUs7Z0JBQ1UsSUFBSTtzQkFBbkIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29tcG9uZW50LCBJbnB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0xpdGVyYWx9IGZyb20gJy4uLy4uL3R5cGVzL3R5cGVzJztcbmltcG9ydCB7TmF0dXJhbEljb25EaXJlY3RpdmV9IGZyb20gJy4uL2ljb24vaWNvbi5kaXJlY3RpdmUnO1xuaW1wb3J0IHtNYXRJY29uTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9pY29uJztcbmltcG9ydCB7Um91dGVyTGlua30gZnJvbSAnQGFuZ3VsYXIvcm91dGVyJztcbmltcG9ydCB7TWF0QnV0dG9uTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9idXR0b24nO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ25hdHVyYWwtZGV0YWlsLWhlYWRlcicsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2RldGFpbC1oZWFkZXIuY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsOiAnLi9kZXRhaWwtaGVhZGVyLmNvbXBvbmVudC5zY3NzJyxcbiAgICBzdGFuZGFsb25lOiB0cnVlLFxuICAgIGltcG9ydHM6IFtNYXRCdXR0b25Nb2R1bGUsIFJvdXRlckxpbmssIE1hdEljb25Nb2R1bGUsIE5hdHVyYWxJY29uRGlyZWN0aXZlXSxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbERldGFpbEhlYWRlckNvbXBvbmVudCB7XG4gICAgLyoqXG4gICAgICogQmFzZSBVUkwgdXNlZCB0byBidWlsZCBsaW5rcywgZGVmYXVsdHMgdG8gJy8nXG4gICAgICovXG4gICAgQElucHV0KCkgcHVibGljIGN1cnJlbnRCYXNlVXJsPzogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogTXVzdCBiZSBzZXQgdG8gZ2V0IHByb3BlciBsaW5rcyB3aGVuIHVzZWQgaW4gcGFuZWxzXG4gICAgICovXG4gICAgQElucHV0KCkgcHVibGljIGlzUGFuZWwgPSBmYWxzZTtcblxuICAgIC8qKlxuICAgICAqIElmIGdpdmVuIHdpbGwgc2hvdyBpY29uIGJlZm9yZSB0aXRsZVxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBpY29uID0gJyc7XG4gICAgLyoqXG4gICAgICogVGl0bGUgc2hvd24gaWYgbW9kZWwgaGFzIG5vIG5hbWUsIG9yIGVtcHR5IG5hbWUuXG4gICAgICpcbiAgICAgKiBUeXBpY2FsbHkgc2hvdWxkIGJlIHRoZSBodW1hbiBuYW1lIGZvciB0aGUgb2JqZWN0IHR5cGUsIGVnOiAnUHJvZHVjdCdcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbGFiZWwgPSAnJztcblxuICAgIC8qKlxuICAgICAqIExhYmVsIG9mIHRoZSByb290IG9mIHRoZSBicmVhZGNydW1iLCBkZWZhdWx0cyB0byB0aGUgdmFsdWUgb2YgYGxhYmVsYC5cbiAgICAgKlxuICAgICAqIFR5cGljYWxseSBzaG91bGQgYmUgdGhlIHBsdXJhbCBmb3JtIG9mIHRoZSBvYmplY3QgdHlwZSwgZWc6ICdQcm9kdWN0cydcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgcm9vdExhYmVsID0gJyc7XG5cbiAgICAvKipcbiAgICAgKiBUaXRsZSBzaG93biBpZiBtb2RlbCBoYXMgbm8gaWQuXG4gICAgICpcbiAgICAgKiBUeXBpY2FsbHkgc2hvdWxkIGJlIHNpbWlsYXIgdG8gJ05ldyBwcm9kdWN0Jy5cbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbmV3TGFiZWwgPSAnJztcbiAgICBASW5wdXQoe3JlcXVpcmVkOiB0cnVlfSkgcHVibGljIG1vZGVsITogTGl0ZXJhbDtcbiAgICBASW5wdXQoKSBwdWJsaWMgYnJlYWRjcnVtYnM6IExpdGVyYWxbXSA9IFtdO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBsaXN0Um91dGU6IGFueVtdID0gW107XG4gICAgQElucHV0KCkgcHVibGljIGxpc3RGcmFnbWVudDogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBsaW5rPzogKGlkOiBzdHJpbmcpID0+IGFueVtdO1xuXG4gICAgcHVibGljIGdldFJvb3RMaW5rKCk6IHN0cmluZ1tdIHtcbiAgICAgICAgcmV0dXJuIFt0aGlzLmN1cnJlbnRCYXNlVXJsIHx8ICcvJ10uY29uY2F0KHRoaXMubGlzdFJvdXRlKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0TGluayhpZDogc3RyaW5nKTogYW55W10ge1xuICAgICAgICBpZiAodGhpcy5saW5rKSB7XG4gICAgICAgICAgICByZXR1cm4gdGhpcy5nZXRSb290TGluaygpLmNvbmNhdCh0aGlzLmxpbmsoaWQpKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiB0aGlzLmdldFJvb3RMaW5rKCkuY29uY2F0KFtpZF0pO1xuICAgIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJicmVhZGNydW1iXCI+XG4gICAgQGlmIChyb290TGFiZWwgfHwgbGFiZWwpIHtcbiAgICAgICAgPGEgW3JvdXRlckxpbmtdPVwiaXNQYW5lbCA/IFtdIDogZ2V0Um9vdExpbmsoKVwiIFtmcmFnbWVudF09XCJsaXN0RnJhZ21lbnRcIiBjb2xvcj1cInByaW1hcnlcIiBtYXQtYnV0dG9uPnt7XG4gICAgICAgICAgICByb290TGFiZWwgfHwgbGFiZWxcbiAgICAgICAgfX08L2E+XG4gICAgfVxuICAgIEBmb3IgKHBhcmVudCBvZiBicmVhZGNydW1iczsgdHJhY2sgcGFyZW50KSB7XG4gICAgICAgIC9cbiAgICAgICAgPGEgW3JvdXRlckxpbmtdPVwiaXNQYW5lbCA/IFtdIDogZ2V0TGluayhwYXJlbnQuaWQpXCIgY29sb3I9XCJwcmltYXJ5XCIgbWF0LWJ1dHRvbj5cbiAgICAgICAgICAgIHt7IHBhcmVudD8uZnVsbE5hbWUgfHwgcGFyZW50Py5uYW1lIH19PC9hXG4gICAgICAgID5cbiAgICB9XG48L2Rpdj5cblxuPGRpdiBjbGFzcz1cImJvZHlcIj5cbiAgICBAaWYgKGljb24pIHtcbiAgICAgICAgPGRpdiBzdHlsZT1cIndpZHRoOiAzMHB4XCI+XG4gICAgICAgICAgICA8bWF0LWljb24gW25hdHVyYWxJY29uXT1cImljb25cIiAvPlxuICAgICAgICA8L2Rpdj5cbiAgICB9XG4gICAgQGlmICghbW9kZWwuaWQpIHtcbiAgICAgICAgPGRpdiBjbGFzcz1cIm1hdC1oZWFkbGluZS01IG5vLW1hcmdpbiBuZXdMYWJlbFwiPnt7IG5ld0xhYmVsIH19PC9kaXY+XG4gICAgfVxuICAgIEBpZiAobW9kZWwuaWQpIHtcbiAgICAgICAgPGRpdiBjbGFzcz1cIm1hdC1oZWFkbGluZS01IG5vLW1hcmdpbiBsYWJlbFwiPnt7IG1vZGVsLm5hbWUgfHwgbW9kZWwuZnVsbE5hbWUgfHwgbGFiZWwgfX08L2Rpdj5cbiAgICB9XG4gICAgPGRpdj5cbiAgICAgICAgPG5nLWNvbnRlbnQgLz5cbiAgICA8L2Rpdj5cbjwvZGl2PlxuIl19
79
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLWhlYWRlci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9kZXRhaWwtaGVhZGVyL2RldGFpbC1oZWFkZXIuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZGV0YWlsLWhlYWRlci9kZXRhaWwtaGVhZGVyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsS0FBSyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRS9DLE9BQU8sRUFBQyxvQkFBb0IsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQzVELE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSx3QkFBd0IsQ0FBQztBQUNyRCxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0saUJBQWlCLENBQUM7QUFDM0MsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLDBCQUEwQixDQUFDOzs7O0FBU3pELE1BQU0sT0FBTyw0QkFBNEI7SUFQekM7UUFhSTs7V0FFRztRQUNhLFlBQU8sR0FBRyxLQUFLLENBQUM7UUFFaEM7O1dBRUc7UUFDYSxTQUFJLEdBQUcsRUFBRSxDQUFDO1FBQzFCOzs7O1dBSUc7UUFDYSxVQUFLLEdBQUcsRUFBRSxDQUFDO1FBRTNCOzs7O1dBSUc7UUFDYSxjQUFTLEdBQUcsRUFBRSxDQUFDO1FBRS9COzs7O1dBSUc7UUFDYSxhQUFRLEdBQUcsRUFBRSxDQUFDO1FBRWQsZ0JBQVcsR0FBYyxFQUFFLENBQUM7UUFDNUIsY0FBUyxHQUFVLEVBQUUsQ0FBQztLQWV6QztJQVhVLFdBQVc7UUFDZCxPQUFPLENBQUMsSUFBSSxDQUFDLGNBQWMsSUFBSSxHQUFHLENBQUMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQy9ELENBQUM7SUFFTSxPQUFPLENBQUMsRUFBVTtRQUNyQixJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNaLE9BQU8sSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDcEQsQ0FBQztRQUVELE9BQU8sSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDM0MsQ0FBQzs4R0FuRFEsNEJBQTRCO2tHQUE1Qiw0QkFBNEIsK1VDZHpDLCs1QkE4QkEsMGhCRGxCYyxlQUFlLDRNQUFFLFVBQVUsbU9BQUUsYUFBYSxvTEFBRSxvQkFBb0I7OzJGQUVqRSw0QkFBNEI7a0JBUHhDLFNBQVM7K0JBQ0ksdUJBQXVCLGNBR3JCLElBQUksV0FDUCxDQUFDLGVBQWUsRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLG9CQUFvQixDQUFDOzhCQU0zRCxjQUFjO3NCQUE3QixLQUFLO2dCQUtVLE9BQU87c0JBQXRCLEtBQUs7Z0JBS1UsSUFBSTtzQkFBbkIsS0FBSztnQkFNVSxLQUFLO3NCQUFwQixLQUFLO2dCQU9VLFNBQVM7c0JBQXhCLEtBQUs7Z0JBT1UsUUFBUTtzQkFBdkIsS0FBSztnQkFDMEIsS0FBSztzQkFBcEMsS0FBSzt1QkFBQyxFQUFDLFFBQVEsRUFBRSxJQUFJLEVBQUM7Z0JBQ1AsV0FBVztzQkFBMUIsS0FBSztnQkFDVSxTQUFTO3NCQUF4QixLQUFLO2dCQUNVLFlBQVk7c0JBQTNCLEtBQUs7Z0JBQ1UsSUFBSTtzQkFBbkIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29tcG9uZW50LCBJbnB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0xpdGVyYWx9IGZyb20gJy4uLy4uL3R5cGVzL3R5cGVzJztcbmltcG9ydCB7TmF0dXJhbEljb25EaXJlY3RpdmV9IGZyb20gJy4uL2ljb24vaWNvbi5kaXJlY3RpdmUnO1xuaW1wb3J0IHtNYXRJY29uTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9pY29uJztcbmltcG9ydCB7Um91dGVyTGlua30gZnJvbSAnQGFuZ3VsYXIvcm91dGVyJztcbmltcG9ydCB7TWF0QnV0dG9uTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9idXR0b24nO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ25hdHVyYWwtZGV0YWlsLWhlYWRlcicsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2RldGFpbC1oZWFkZXIuY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsOiAnLi9kZXRhaWwtaGVhZGVyLmNvbXBvbmVudC5zY3NzJyxcbiAgICBzdGFuZGFsb25lOiB0cnVlLFxuICAgIGltcG9ydHM6IFtNYXRCdXR0b25Nb2R1bGUsIFJvdXRlckxpbmssIE1hdEljb25Nb2R1bGUsIE5hdHVyYWxJY29uRGlyZWN0aXZlXSxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbERldGFpbEhlYWRlckNvbXBvbmVudCB7XG4gICAgLyoqXG4gICAgICogQmFzZSBVUkwgdXNlZCB0byBidWlsZCBsaW5rcywgZGVmYXVsdHMgdG8gJy8nXG4gICAgICovXG4gICAgQElucHV0KCkgcHVibGljIGN1cnJlbnRCYXNlVXJsPzogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogTXVzdCBiZSBzZXQgdG8gZ2V0IHByb3BlciBsaW5rcyB3aGVuIHVzZWQgaW4gcGFuZWxzXG4gICAgICovXG4gICAgQElucHV0KCkgcHVibGljIGlzUGFuZWwgPSBmYWxzZTtcblxuICAgIC8qKlxuICAgICAqIElmIGdpdmVuIHdpbGwgc2hvdyBpY29uIGJlZm9yZSB0aXRsZVxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBpY29uID0gJyc7XG4gICAgLyoqXG4gICAgICogVGl0bGUgc2hvd24gaWYgbW9kZWwgaGFzIG5vIG5hbWUsIG9yIGVtcHR5IG5hbWUuXG4gICAgICpcbiAgICAgKiBUeXBpY2FsbHkgc2hvdWxkIGJlIHRoZSBodW1hbiBuYW1lIGZvciB0aGUgb2JqZWN0IHR5cGUsIGVnOiAnUHJvZHVjdCdcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbGFiZWwgPSAnJztcblxuICAgIC8qKlxuICAgICAqIExhYmVsIG9mIHRoZSByb290IG9mIHRoZSBicmVhZGNydW1iLCBkZWZhdWx0cyB0byB0aGUgdmFsdWUgb2YgYGxhYmVsYC5cbiAgICAgKlxuICAgICAqIFR5cGljYWxseSBzaG91bGQgYmUgdGhlIHBsdXJhbCBmb3JtIG9mIHRoZSBvYmplY3QgdHlwZSwgZWc6ICdQcm9kdWN0cydcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgcm9vdExhYmVsID0gJyc7XG5cbiAgICAvKipcbiAgICAgKiBUaXRsZSBzaG93biBpZiBtb2RlbCBoYXMgbm8gaWQuXG4gICAgICpcbiAgICAgKiBUeXBpY2FsbHkgc2hvdWxkIGJlIHNpbWlsYXIgdG8gJ05ldyBwcm9kdWN0Jy5cbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbmV3TGFiZWwgPSAnJztcbiAgICBASW5wdXQoe3JlcXVpcmVkOiB0cnVlfSkgcHVibGljIG1vZGVsITogTGl0ZXJhbDtcbiAgICBASW5wdXQoKSBwdWJsaWMgYnJlYWRjcnVtYnM6IExpdGVyYWxbXSA9IFtdO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBsaXN0Um91dGU6IGFueVtdID0gW107XG4gICAgQElucHV0KCkgcHVibGljIGxpc3RGcmFnbWVudDogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBsaW5rPzogKGlkOiBzdHJpbmcpID0+IGFueVtdO1xuXG4gICAgcHVibGljIGdldFJvb3RMaW5rKCk6IHN0cmluZ1tdIHtcbiAgICAgICAgcmV0dXJuIFt0aGlzLmN1cnJlbnRCYXNlVXJsIHx8ICcvJ10uY29uY2F0KHRoaXMubGlzdFJvdXRlKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0TGluayhpZDogc3RyaW5nKTogYW55W10ge1xuICAgICAgICBpZiAodGhpcy5saW5rKSB7XG4gICAgICAgICAgICByZXR1cm4gdGhpcy5nZXRSb290TGluaygpLmNvbmNhdCh0aGlzLmxpbmsoaWQpKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiB0aGlzLmdldFJvb3RMaW5rKCkuY29uY2F0KFtpZF0pO1xuICAgIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJicmVhZGNydW1iXCI+XG4gICAgQGlmIChyb290TGFiZWwgfHwgbGFiZWwpIHtcbiAgICAgICAgPGEgW3JvdXRlckxpbmtdPVwiaXNQYW5lbCA/IFtdIDogZ2V0Um9vdExpbmsoKVwiIFtmcmFnbWVudF09XCJsaXN0RnJhZ21lbnRcIiBjb2xvcj1cInByaW1hcnlcIiBtYXQtYnV0dG9uPnt7XG4gICAgICAgICAgICByb290TGFiZWwgfHwgbGFiZWxcbiAgICAgICAgfX08L2E+XG4gICAgfVxuICAgIEBmb3IgKHBhcmVudCBvZiBicmVhZGNydW1iczsgdHJhY2sgcGFyZW50KSB7XG4gICAgICAgIC9cbiAgICAgICAgPGEgW3JvdXRlckxpbmtdPVwiaXNQYW5lbCA/IFtdIDogZ2V0TGluayhwYXJlbnQuaWQpXCIgY29sb3I9XCJwcmltYXJ5XCIgbWF0LWJ1dHRvbj5cbiAgICAgICAgICAgIHt7IHBhcmVudD8uZnVsbE5hbWUgfHwgcGFyZW50Py5uYW1lIH19PC9hXG4gICAgICAgID5cbiAgICB9XG48L2Rpdj5cblxuPGRpdiBjbGFzcz1cImJvZHlcIj5cbiAgICBAaWYgKGljb24pIHtcbiAgICAgICAgPGRpdiBzdHlsZT1cIndpZHRoOiAzMHB4XCI+XG4gICAgICAgICAgICA8bWF0LWljb24gW25hdHVyYWxJY29uXT1cImljb25cIiAvPlxuICAgICAgICA8L2Rpdj5cbiAgICB9XG4gICAgQGlmICghbW9kZWwuaWQpIHtcbiAgICAgICAgPGRpdiBjbGFzcz1cIm1hdC1oZWFkbGluZS01IG5hdC1uby1tYXJnaW4gbmV3TGFiZWxcIj57eyBuZXdMYWJlbCB9fTwvZGl2PlxuICAgIH1cbiAgICBAaWYgKG1vZGVsLmlkKSB7XG4gICAgICAgIDxkaXYgY2xhc3M9XCJtYXQtaGVhZGxpbmUtNSBuYXQtbm8tbWFyZ2luIGxhYmVsXCI+e3sgbW9kZWwubmFtZSB8fCBtb2RlbC5mdWxsTmFtZSB8fCBsYWJlbCB9fTwvZGl2PlxuICAgIH1cbiAgICA8ZGl2PlxuICAgICAgICA8bmctY29udGVudCAvPlxuICAgIDwvZGl2PlxuPC9kaXY+XG4iXX0=