@ecodev/natural 44.0.4 → 45.0.0

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.
@@ -7,7 +7,7 @@ import { NaturalAbstractPanel } from '../modules/panels/abstract-panel';
7
7
  import { finalize } from 'rxjs/operators';
8
8
  import { ifValid, validateAllFormControls } from './validators';
9
9
  import { mergeOverrideArray } from './utility';
10
- import { EMPTY } from 'rxjs';
10
+ import { EMPTY, endWith, last, switchMap } from 'rxjs';
11
11
  import * as i0 from "@angular/core";
12
12
  // @dynamic
13
13
  export class NaturalAbstractDetail extends NaturalAbstractPanel {
@@ -79,51 +79,50 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
79
79
  this.form.disable();
80
80
  this.service
81
81
  .create(this.data.model)
82
- .pipe(finalize(() => this.form.enable()))
83
- .subscribe(model => {
82
+ .pipe(switchMap(model => {
84
83
  this.alertService.info($localize `Créé`);
85
84
  this.form.patchValue(model);
86
- this.postCreate(model).subscribe({
87
- complete: () => {
88
- if (redirect) {
89
- if (this.isPanel) {
90
- const oldUrl = this.router.url;
91
- const nextUrl = this.panelData?.config.params.nextRoute;
92
- const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');
93
- this.router.navigateByUrl(newUrl); // replace /new by /123
94
- }
95
- else {
96
- this.router.navigate(['..', model.id], { relativeTo: this.route });
97
- }
98
- }
99
- },
100
- });
101
- });
85
+ return this.postCreate(model).pipe(endWith(model), last());
86
+ }), switchMap(model => {
87
+ if (redirect) {
88
+ if (this.isPanel) {
89
+ const oldUrl = this.router.url;
90
+ const nextUrl = this.panelData?.config.params.nextRoute;
91
+ const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');
92
+ return this.router.navigateByUrl(newUrl); // replace /new by /123
93
+ }
94
+ else {
95
+ return this.router.navigate(['..', model.id], { relativeTo: this.route });
96
+ }
97
+ }
98
+ return EMPTY;
99
+ }), finalize(() => this.form.enable()))
100
+ .subscribe();
102
101
  }
103
102
  delete(redirectionRoute) {
103
+ this.form.disable();
104
104
  this.alertService
105
105
  .confirm($localize `Suppression`, $localize `Voulez-vous supprimer définitivement cet élément ?`, $localize `Supprimer définitivement`)
106
- .subscribe(confirmed => {
107
- if (confirmed) {
108
- this.preDelete(this.data.model);
109
- this.form.disable();
110
- this.service
111
- .delete([this.data.model])
112
- .pipe(finalize(() => this.form.enable()))
113
- .subscribe(() => {
114
- this.alertService.info($localize `Supprimé`);
115
- if (!this.isPanel) {
116
- const defaultRoute = ['../../' + kebabCase(this.key)];
117
- this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {
118
- relativeTo: this.route,
119
- });
120
- }
121
- else {
122
- this.panelService?.goToPenultimatePanel();
123
- }
124
- });
106
+ .pipe(switchMap(confirmed => {
107
+ if (!confirmed) {
108
+ return EMPTY;
125
109
  }
126
- });
110
+ this.preDelete(this.data.model);
111
+ return this.service.delete([this.data.model]).pipe(switchMap(() => {
112
+ this.alertService.info($localize `Supprimé`);
113
+ if (this.isPanel) {
114
+ this.panelService?.goToPenultimatePanel();
115
+ return EMPTY;
116
+ }
117
+ else {
118
+ const defaultRoute = ['../../' + kebabCase(this.key)];
119
+ return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {
120
+ relativeTo: this.route,
121
+ });
122
+ }
123
+ }));
124
+ }), finalize(() => this.form.enable()))
125
+ .subscribe();
127
126
  }
128
127
  postUpdate(model) { }
129
128
  /**
@@ -149,4 +148,4 @@ NaturalAbstractDetail.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0",
149
148
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalAbstractDetail, decorators: [{
150
149
  type: Directive
151
150
  }], ctorParameters: function () { return [{ type: undefined }, { type: undefined }, { type: i0.Injector }]; } });
152
- //# 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, Injector, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase, merge, mergeWith, omit} 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 {ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {finalize} from 'rxjs/operators';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {mergeOverrideArray} from './utility';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {EMPTY, Observable} from 'rxjs';\n\n// @dynamic\n@Directive()\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            unknown,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Empty placeholder for data retrieved by the server\n     */\n    public data: any = {\n        model: {},\n    };\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 alertService: NaturalAlertService;\n\n    /**\n     * Injected service\n     */\n    protected router: Router;\n\n    /**\n     * Injected service\n     */\n    protected route: ActivatedRoute;\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n        protected readonly injector: Injector,\n    ) {\n        super();\n\n        this.alertService = injector.get(NaturalAlertService);\n        this.router = injector.get(Router);\n        this.route = injector.get(ActivatedRoute);\n    }\n\n    public ngOnInit(): void {\n        if (!this.isPanel) {\n            this.route.data.subscribe(data => {\n                this.data = merge({model: this.service.getConsolidatedForClient()}, data[this.key]);\n                this.data = merge(this.data, omit(data, [this.key]));\n                this.initForm();\n            });\n        } else {\n            this.initForm();\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    public update(now: boolean = false): void {\n        if (!this.data.model.id) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            this.formToData();\n            const postUpdate = (model: ExtractTupdate<TService>): void => {\n                this.alertService.info($localize`Mis à jour`);\n                this.form.patchValue(model);\n                this.postUpdate(model);\n            };\n\n            if (now) {\n                this.service.updateNow(this.data.model).subscribe(postUpdate);\n            } else {\n                this.service.update(this.data.model).subscribe(postUpdate);\n            }\n        });\n    }\n\n    public create(redirect: boolean = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        this.formToData();\n        this.form.disable();\n\n        this.service\n            .create(this.data.model)\n            .pipe(finalize(() => this.form.enable()))\n            .subscribe(model => {\n                this.alertService.info($localize`Créé`);\n                this.form.patchValue(model);\n\n                this.postCreate(model).subscribe({\n                    complete: () => {\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                                this.router.navigateByUrl(newUrl); // replace /new by /123\n                            } else {\n                                this.router.navigate(['..', model.id], {relativeTo: this.route});\n                            }\n                        }\n                    },\n                });\n            });\n    }\n\n    public delete(redirectionRoute?: unknown[]): void {\n        this.alertService\n            .confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n            .subscribe(confirmed => {\n                if (confirmed) {\n                    this.preDelete(this.data.model);\n                    this.form.disable();\n\n                    this.service\n                        .delete([this.data.model])\n                        .pipe(finalize(() => this.form.enable()))\n                        .subscribe(() => {\n                            this.alertService.info($localize`Supprimé`);\n\n                            if (!this.isPanel) {\n                                const defaultRoute = ['../../' + kebabCase(this.key)];\n                                this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {\n                                    relativeTo: this.route,\n                                });\n                            } else {\n                                this.panelService?.goToPenultimatePanel();\n                            }\n                        });\n                }\n            });\n    }\n\n    protected postUpdate(model: ExtractTupdate<TService>): void {}\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    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    protected preDelete(model: ExtractTone<TService>): void {}\n\n    protected initForm(): void {\n        this.form = this.service.getFormGroup(this.data.model);\n    }\n\n    /**\n     * Merge values of form into `this.data.model`.\n     */\n    protected formToData(): void {\n        mergeWith(this.data.model, this.form.value, mergeOverrideArray);\n    }\n}\n"]}
151
+ //# 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, Injector, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase, merge, mergeWith, omit} 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 {ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {finalize} from 'rxjs/operators';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {mergeOverrideArray} from './utility';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {EMPTY, endWith, last, Observable, switchMap} from 'rxjs';\n\n// @dynamic\n@Directive()\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            unknown,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Empty placeholder for data retrieved by the server\n     */\n    public data: any = {\n        model: {},\n    };\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 alertService: NaturalAlertService;\n\n    /**\n     * Injected service\n     */\n    protected router: Router;\n\n    /**\n     * Injected service\n     */\n    protected route: ActivatedRoute;\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n        protected readonly injector: Injector,\n    ) {\n        super();\n\n        this.alertService = injector.get(NaturalAlertService);\n        this.router = injector.get(Router);\n        this.route = injector.get(ActivatedRoute);\n    }\n\n    public ngOnInit(): void {\n        if (!this.isPanel) {\n            this.route.data.subscribe(data => {\n                this.data = merge({model: this.service.getConsolidatedForClient()}, data[this.key]);\n                this.data = merge(this.data, omit(data, [this.key]));\n                this.initForm();\n            });\n        } else {\n            this.initForm();\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    public update(now: boolean = false): void {\n        if (!this.data.model.id) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            this.formToData();\n            const postUpdate = (model: ExtractTupdate<TService>): void => {\n                this.alertService.info($localize`Mis à jour`);\n                this.form.patchValue(model);\n                this.postUpdate(model);\n            };\n\n            if (now) {\n                this.service.updateNow(this.data.model).subscribe(postUpdate);\n            } else {\n                this.service.update(this.data.model).subscribe(postUpdate);\n            }\n        });\n    }\n\n    public create(redirect: boolean = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        this.formToData();\n        this.form.disable();\n\n        this.service\n            .create(this.data.model)\n            .pipe(\n                switchMap(model => {\n                    this.alertService.info($localize`Créé`);\n                    this.form.patchValue(model);\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    public delete(redirectionRoute?: unknown[]): void {\n        this.form.disable();\n        this.alertService\n            .confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n            .pipe(\n                switchMap(confirmed => {\n                    if (!confirmed) {\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    protected postUpdate(model: ExtractTupdate<TService>): void {}\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    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    protected preDelete(model: ExtractTone<TService>): void {}\n\n    protected initForm(): void {\n        this.form = this.service.getFormGroup(this.data.model);\n    }\n\n    /**\n     * Merge values of form into `this.data.model`.\n     */\n    protected formToData(): void {\n        mergeWith(this.data.model, this.form.value, mergeOverrideArray);\n    }\n}\n"]}
@@ -40,9 +40,10 @@ export function cancellableTimeout(canceller, milliSeconds = 0) {
40
40
  export function debug(debugName) {
41
41
  return tap({
42
42
  subscribe: () => console.log('SUBSCRIBE', debugName),
43
+ unsubscribe: () => console.log('UNSUBSCRIBE', debugName),
43
44
  next: value => console.log('NEXT', debugName, value),
44
45
  error: error => console.log('ERROR', debugName, error),
45
46
  complete: () => console.log('COMPLETE', debugName),
46
47
  });
47
48
  }
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnhqcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9jbGFzc2VzL3J4anMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUF1QyxLQUFLLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDakUsT0FBTyxFQUFDLEdBQUcsRUFBRSxJQUFJLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRXpEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0EyQkc7QUFDSCxNQUFNLFVBQVUsa0JBQWtCLENBQUMsU0FBOEIsRUFBRSxlQUF1QixDQUFDO0lBQ3ZGLE9BQU8sS0FBSyxDQUFDLFlBQVksQ0FBQyxDQUFDLElBQUksQ0FDM0IsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUNQLFNBQVMsQ0FBQyxTQUFTLENBQUMsRUFDcEIsR0FBRyxDQUFDLEdBQUcsRUFBRTtRQUNMLE9BQU87SUFDWCxDQUFDLENBQUMsQ0FDTCxDQUFDO0FBQ04sQ0FBQztBQUVEOzs7R0FHRztBQUNILE1BQU0sVUFBVSxLQUFLLENBQUksU0FBaUI7SUFDdEMsT0FBTyxHQUFHLENBQUk7UUFDVixTQUFTLEVBQUUsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLEVBQUUsU0FBUyxDQUFDO1FBQ3BELElBQUksRUFBRSxLQUFLLENBQUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFNBQVMsRUFBRSxLQUFLLENBQUM7UUFDcEQsS0FBSyxFQUFFLEtBQUssQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssQ0FBQztRQUN0RCxRQUFRLEVBQUUsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLEVBQUUsU0FBUyxDQUFDO0tBQ3JELENBQUMsQ0FBQztBQUNQLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge01vbm9UeXBlT3BlcmF0b3JGdW5jdGlvbiwgT2JzZXJ2YWJsZSwgdGltZXJ9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHttYXAsIHRha2UsIHRha2VVbnRpbCwgdGFwfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbi8qKlxuICogQmVoYXZlIGxpa2Ugc2V0VGltZW91dCgpLCBidXQgd2l0aCBhIG1hbmRhdG9yeSBjYW5jZWwgbWVjaGFuaXNtLlxuICpcbiAqIFRoaXMgaXMgdHlwaWNhbGx5IHVzZWZ1bCB0byByZXBsYWNlIHNldFRpbWVvdXQoKSBpbiBjb21wb25lbnRzIHdoZXJlIHRoZSBjYWxsYmFja1xuICogd291bGQgY3Jhc2ggaWYgZXhlY3V0ZWQgYWZ0ZXIgdGhlIGNvbXBvbmVudCBkZXN0cnVjdGlvbi4gVGhhdCBjYW4gZWFzaWx5IGhhcHBlblxuICogd2hlbiB0aGUgdXNlciBuYXZpZ2F0ZSBxdWlja2x5IGJldHdlZW4gcGFnZXMuXG4gKlxuICogVHlwaWNhbCB1c2FnZSBpbiBhIGNvbXBvbmVudCB3b3VsZCBiZTpcbiAqXG4gKiBgYGB0c1xuICogY2FuY2VsbGFibGVUaW1lb3V0KHRoaXMubmdVbnN1YnNjcmliZSkuc3Vic2NyaWJlKG15Q2FsbGJhY2spO1xuICogYGBgXG4gKlxuICogSW5zdGVhZCBvZiB0aGUgbW9yZSBlcnJvciBwcm9uZTpcbiAqXG4gKiBgYGB0c1xuICogcHVibGljIGZvbygpOiB2b2lkIHtcbiAqICAgICB0aGlzLnRpbWVvdXQgPSBzZXRUaW1lb3V0KG15Q2FsbEJhY2spO1xuICogfVxuICpcbiAqIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAqICAgICBpZiAodGhpcy50aW1lb3V0KSB7XG4gKiAgICAgICAgIGNsZWFyVGltZW91dCh0aGlzLnRpbWVvdXQpO1xuICogICAgICAgICB0aGlzLnRpbWVvdXQgPSBudWxsO1xuICogICAgICB9XG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNhbmNlbGxhYmxlVGltZW91dChjYW5jZWxsZXI6IE9ic2VydmFibGU8dW5rbm93bj4sIG1pbGxpU2Vjb25kczogbnVtYmVyID0gMCk6IE9ic2VydmFibGU8dm9pZD4ge1xuICAgIHJldHVybiB0aW1lcihtaWxsaVNlY29uZHMpLnBpcGUoXG4gICAgICAgIHRha2UoMSksXG4gICAgICAgIHRha2VVbnRpbChjYW5jZWxsZXIpLFxuICAgICAgICBtYXAoKCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9KSxcbiAgICApO1xufVxuXG4vKipcbiAqIEZvciBkZWJ1Z2dpbmcgcHVycG9zZSBvbmx5LCB3aWxsIGR1bXAgaW4gY29uc29sZSBldmVyeXRoaW5nIHRoYXQgaGFwcGVuIHRvXG4gKiB0aGUgb2JzZXJ2YWJsZVxuICovXG5leHBvcnQgZnVuY3Rpb24gZGVidWc8VD4oZGVidWdOYW1lOiBzdHJpbmcpOiBNb25vVHlwZU9wZXJhdG9yRnVuY3Rpb248VD4ge1xuICAgIHJldHVybiB0YXA8VD4oe1xuICAgICAgICBzdWJzY3JpYmU6ICgpID0+IGNvbnNvbGUubG9nKCdTVUJTQ1JJQkUnLCBkZWJ1Z05hbWUpLFxuICAgICAgICBuZXh0OiB2YWx1ZSA9PiBjb25zb2xlLmxvZygnTkVYVCcsIGRlYnVnTmFtZSwgdmFsdWUpLFxuICAgICAgICBlcnJvcjogZXJyb3IgPT4gY29uc29sZS5sb2coJ0VSUk9SJywgZGVidWdOYW1lLCBlcnJvciksXG4gICAgICAgIGNvbXBsZXRlOiAoKSA9PiBjb25zb2xlLmxvZygnQ09NUExFVEUnLCBkZWJ1Z05hbWUpLFxuICAgIH0pO1xufVxuIl19
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnhqcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9jbGFzc2VzL3J4anMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUF1QyxLQUFLLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDakUsT0FBTyxFQUFDLEdBQUcsRUFBRSxJQUFJLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRXpEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0EyQkc7QUFDSCxNQUFNLFVBQVUsa0JBQWtCLENBQUMsU0FBOEIsRUFBRSxlQUF1QixDQUFDO0lBQ3ZGLE9BQU8sS0FBSyxDQUFDLFlBQVksQ0FBQyxDQUFDLElBQUksQ0FDM0IsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUNQLFNBQVMsQ0FBQyxTQUFTLENBQUMsRUFDcEIsR0FBRyxDQUFDLEdBQUcsRUFBRTtRQUNMLE9BQU87SUFDWCxDQUFDLENBQUMsQ0FDTCxDQUFDO0FBQ04sQ0FBQztBQUVEOzs7R0FHRztBQUNILE1BQU0sVUFBVSxLQUFLLENBQUksU0FBaUI7SUFDdEMsT0FBTyxHQUFHLENBQUk7UUFDVixTQUFTLEVBQUUsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLEVBQUUsU0FBUyxDQUFDO1FBQ3BELFdBQVcsRUFBRSxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLGFBQWEsRUFBRSxTQUFTLENBQUM7UUFDeEQsSUFBSSxFQUFFLEtBQUssQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsU0FBUyxFQUFFLEtBQUssQ0FBQztRQUNwRCxLQUFLLEVBQUUsS0FBSyxDQUFDLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxDQUFDO1FBQ3RELFFBQVEsRUFBRSxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLFVBQVUsRUFBRSxTQUFTLENBQUM7S0FDckQsQ0FBQyxDQUFDO0FBQ1AsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7TW9ub1R5cGVPcGVyYXRvckZ1bmN0aW9uLCBPYnNlcnZhYmxlLCB0aW1lcn0gZnJvbSAncnhqcyc7XG5pbXBvcnQge21hcCwgdGFrZSwgdGFrZVVudGlsLCB0YXB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuLyoqXG4gKiBCZWhhdmUgbGlrZSBzZXRUaW1lb3V0KCksIGJ1dCB3aXRoIGEgbWFuZGF0b3J5IGNhbmNlbCBtZWNoYW5pc20uXG4gKlxuICogVGhpcyBpcyB0eXBpY2FsbHkgdXNlZnVsIHRvIHJlcGxhY2Ugc2V0VGltZW91dCgpIGluIGNvbXBvbmVudHMgd2hlcmUgdGhlIGNhbGxiYWNrXG4gKiB3b3VsZCBjcmFzaCBpZiBleGVjdXRlZCBhZnRlciB0aGUgY29tcG9uZW50IGRlc3RydWN0aW9uLiBUaGF0IGNhbiBlYXNpbHkgaGFwcGVuXG4gKiB3aGVuIHRoZSB1c2VyIG5hdmlnYXRlIHF1aWNrbHkgYmV0d2VlbiBwYWdlcy5cbiAqXG4gKiBUeXBpY2FsIHVzYWdlIGluIGEgY29tcG9uZW50IHdvdWxkIGJlOlxuICpcbiAqIGBgYHRzXG4gKiBjYW5jZWxsYWJsZVRpbWVvdXQodGhpcy5uZ1Vuc3Vic2NyaWJlKS5zdWJzY3JpYmUobXlDYWxsYmFjayk7XG4gKiBgYGBcbiAqXG4gKiBJbnN0ZWFkIG9mIHRoZSBtb3JlIGVycm9yIHByb25lOlxuICpcbiAqIGBgYHRzXG4gKiBwdWJsaWMgZm9vKCk6IHZvaWQge1xuICogICAgIHRoaXMudGltZW91dCA9IHNldFRpbWVvdXQobXlDYWxsQmFjayk7XG4gKiB9XG4gKlxuICogcHVibGljIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICogICAgIGlmICh0aGlzLnRpbWVvdXQpIHtcbiAqICAgICAgICAgY2xlYXJUaW1lb3V0KHRoaXMudGltZW91dCk7XG4gKiAgICAgICAgIHRoaXMudGltZW91dCA9IG51bGw7XG4gKiAgICAgIH1cbiAqIH1cbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gY2FuY2VsbGFibGVUaW1lb3V0KGNhbmNlbGxlcjogT2JzZXJ2YWJsZTx1bmtub3duPiwgbWlsbGlTZWNvbmRzOiBudW1iZXIgPSAwKTogT2JzZXJ2YWJsZTx2b2lkPiB7XG4gICAgcmV0dXJuIHRpbWVyKG1pbGxpU2Vjb25kcykucGlwZShcbiAgICAgICAgdGFrZSgxKSxcbiAgICAgICAgdGFrZVVudGlsKGNhbmNlbGxlciksXG4gICAgICAgIG1hcCgoKSA9PiB7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH0pLFxuICAgICk7XG59XG5cbi8qKlxuICogRm9yIGRlYnVnZ2luZyBwdXJwb3NlIG9ubHksIHdpbGwgZHVtcCBpbiBjb25zb2xlIGV2ZXJ5dGhpbmcgdGhhdCBoYXBwZW4gdG9cbiAqIHRoZSBvYnNlcnZhYmxlXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBkZWJ1ZzxUPihkZWJ1Z05hbWU6IHN0cmluZyk6IE1vbm9UeXBlT3BlcmF0b3JGdW5jdGlvbjxUPiB7XG4gICAgcmV0dXJuIHRhcDxUPih7XG4gICAgICAgIHN1YnNjcmliZTogKCkgPT4gY29uc29sZS5sb2coJ1NVQlNDUklCRScsIGRlYnVnTmFtZSksXG4gICAgICAgIHVuc3Vic2NyaWJlOiAoKSA9PiBjb25zb2xlLmxvZygnVU5TVUJTQ1JJQkUnLCBkZWJ1Z05hbWUpLFxuICAgICAgICBuZXh0OiB2YWx1ZSA9PiBjb25zb2xlLmxvZygnTkVYVCcsIGRlYnVnTmFtZSwgdmFsdWUpLFxuICAgICAgICBlcnJvcjogZXJyb3IgPT4gY29uc29sZS5sb2coJ0VSUk9SJywgZGVidWdOYW1lLCBlcnJvciksXG4gICAgICAgIGNvbXBsZXRlOiAoKSA9PiBjb25zb2xlLmxvZygnQ09NUExFVEUnLCBkZWJ1Z05hbWUpLFxuICAgIH0pO1xufVxuIl19
@@ -1,7 +1,8 @@
1
1
  import { Directive, EventEmitter, HostListener, Inject, Input, Output, } from '@angular/core';
2
- import { acceptType, createInvisibleFileInputWrap, detectSwipe, eventToFiles, fileListToArray, isFileInput, stopEvent, } from './utils';
2
+ import { acceptType, createInvisibleFileInputWrap, detectSwipe, eventToFiles, fileListToArray, isDirectory, isFileInput, stopEvent, } from './utils';
3
3
  import { NaturalAbstractController } from '../../classes/abstract-controller';
4
4
  import { DOCUMENT } from '@angular/common';
5
+ import { forkJoin, map, of, tap } from 'rxjs';
5
6
  import * as i0 from "@angular/core";
6
7
  import * as i1 from "./file.service";
7
8
  /**
@@ -19,10 +20,6 @@ export class NaturalAbstractFile extends NaturalAbstractController {
19
20
  this.element = element;
20
21
  this.naturalFileService = naturalFileService;
21
22
  this.document = document;
22
- this.validators = [
23
- { name: 'accept', fn: this.acceptValidator },
24
- { name: 'fileSize', fn: this.fileSizeValidator },
25
- ];
26
23
  /**
27
24
  * Whether we should accept a single file or multiple files
28
25
  */
@@ -119,8 +116,7 @@ export class NaturalAbstractFile extends NaturalAbstractController {
119
116
  valid: [],
120
117
  invalid: [],
121
118
  };
122
- for (const file of files) {
123
- const error = this.validate(file);
119
+ forkJoin(files.map(file => this.validate(file).pipe(tap(error => {
124
120
  if (error) {
125
121
  selection.invalid.push({
126
122
  file: file,
@@ -130,17 +126,18 @@ export class NaturalAbstractFile extends NaturalAbstractController {
130
126
  else {
131
127
  selection.valid.push(file);
132
128
  }
133
- }
134
- if (selection.valid.length) {
135
- this.fileChange.emit(selection.valid[0]);
136
- }
137
- if (selection.valid.length || selection.invalid.length) {
138
- this.filesChange.emit(selection);
139
- if (this.broadcast) {
140
- this.naturalFileService.filesChanged.next(selection);
129
+ })))).subscribe(() => {
130
+ if (selection.valid.length) {
131
+ this.fileChange.emit(selection.valid[0]);
141
132
  }
142
- }
143
- this.getFileElement().value = '';
133
+ if (selection.valid.length || selection.invalid.length) {
134
+ this.filesChange.emit(selection);
135
+ if (this.broadcast) {
136
+ this.naturalFileService.filesChanged.next(selection);
137
+ }
138
+ }
139
+ this.getFileElement().value = '';
140
+ });
144
141
  }
145
142
  /**
146
143
  * Called when input has files
@@ -187,18 +184,18 @@ export class NaturalAbstractFile extends NaturalAbstractController {
187
184
  this.handleFiles(files);
188
185
  }
189
186
  validate(file) {
190
- for (const validator of this.validators) {
191
- if (!validator.fn.call(this, file)) {
192
- return validator.name;
187
+ return forkJoin({
188
+ accept: of(acceptType(this.accept, file.type, file.name)),
189
+ fileSize: of(!(this.maxSize && file.size > this.maxSize)),
190
+ directory: isDirectory(file),
191
+ }).pipe(map(result => {
192
+ for (const [key, value] of Object.entries(result)) {
193
+ if (!value) {
194
+ return key;
195
+ }
193
196
  }
194
- }
195
- return null;
196
- }
197
- acceptValidator(item) {
198
- return acceptType(this.accept, item.type, item.name);
199
- }
200
- fileSizeValidator(item) {
201
- return !(this.maxSize && item.size > this.maxSize);
197
+ return null;
198
+ }));
202
199
  }
203
200
  }
204
201
  NaturalAbstractFile.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalAbstractFile, deps: [{ token: i0.ElementRef }, { token: i1.NaturalFileService }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Directive });
@@ -228,4 +225,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.1", ngImpor
228
225
  type: HostListener,
229
226
  args: ['change', ['$event']]
230
227
  }] } });
231
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-file.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/abstract-file.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    ElementRef,\n    EventEmitter,\n    HostListener,\n    Inject,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport {\n    acceptType,\n    createInvisibleFileInputWrap,\n    detectSwipe,\n    eventToFiles,\n    fileListToArray,\n    isFileInput,\n    stopEvent,\n} from './utils';\nimport {NaturalFileService} from './file.service';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\nimport {DOCUMENT} from '@angular/common';\n\nexport interface InvalidFile {\n    file: File;\n    error: string;\n}\n\nexport interface FileSelection {\n    /**\n     * The list of files that have been selected.\n     */\n    valid: File[];\n\n    /**\n     * The list of files that have been selected but are invalid according to validators.\n     */\n    invalid: InvalidFile[];\n}\n\n/**\n * A master base set of logic intended to support file select/drag/drop operations\n *\n * In most cases you probably want click-to-select and drag-to-select, so you should use:\n *\n *     <div naturalFileDrop [selectable]=\"true\"></div>\n *\n * @dynamic\n */\n@Directive()\nexport abstract class NaturalAbstractFile extends NaturalAbstractController implements OnInit, OnDestroy, OnChanges {\n    private fileElement?: HTMLInputElement;\n    private readonly validators = [\n        {name: 'accept', fn: this.acceptValidator},\n        {name: 'fileSize', fn: this.fileSizeValidator},\n    ];\n\n    /**\n     * Whether we should accept a single file or multiple files\n     */\n    @Input() public multiple = false;\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element\n     * it can be a mixed of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept = '';\n\n    /**\n     * Maximum file size in bytes. 0 means no validation at all.\n     */\n    @Input() public maxSize = 0;\n\n    /**\n     * Disable the file selection entirely\n     */\n    @Input() public fileSelectionDisabled = false;\n\n    /**\n     * Whether the user can click on the element to select something\n     *\n     * This has only effect during initialization. Subsequent changes will have\n     * no effect.\n     */\n    @Input() public selectable = false;\n\n    /**\n     * If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.\n     *\n     * It is useful to set this to false if there is two upload on a page with different purpose\n     * and the second upload should not be confused with the first one.\n     */\n    @Input() public broadcast = true;\n\n    /**\n     * The single valid file that has been selected.\n     *\n     * It is for convenience of use, and will only emit if there is at least one\n     * valid file. See `filesChange` for a more complete output.\n     */\n    @Output() public readonly fileChange = new EventEmitter<File>();\n\n    /**\n     * The list of files that have been selected.\n     */\n    @Output() public readonly filesChange = new EventEmitter<FileSelection>();\n\n    public constructor(\n        private readonly element: ElementRef<HTMLElement>,\n        protected readonly naturalFileService: NaturalFileService,\n        @Inject(DOCUMENT) private readonly document: Document,\n    ) {\n        super();\n    }\n\n    public ngOnDestroy(): void {\n        super.ngOnDestroy();\n        delete this.fileElement; // faster memory release of dom element\n    }\n\n    public ngOnInit(): void {\n        if (this.selectable) {\n            this.enableSelecting();\n        }\n\n        this.getFileElement().multiple = this.multiple;\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.accept) {\n            this.getFileElement().setAttribute('accept', changes.accept.currentValue || '*');\n        }\n    }\n\n    private getFileElement(): HTMLInputElement {\n        if (this.fileElement) {\n            return this.fileElement;\n        }\n\n        // elm is a file input\n        if (isFileInput(this.element.nativeElement)) {\n            this.fileElement = this.element.nativeElement;\n\n            return this.fileElement;\n        }\n\n        // create foo file input\n        const label = createInvisibleFileInputWrap(this.document);\n        this.fileElement = label.getElementsByTagName('input')[0];\n        this.fileElement.addEventListener('change', this.changeFn.bind(this));\n        this.element.nativeElement.appendChild(label);\n\n        return this.fileElement;\n    }\n\n    private enableSelecting(): void {\n        const elm = this.element.nativeElement;\n\n        if (isFileInput(elm)) {\n            const bindedHandler = (): void => this.beforeSelect();\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            return;\n        } else {\n            const bindedHandler = (event: Event): boolean => this.clickHandler(event);\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            elm.addEventListener('touchend', bindedHandler);\n        }\n    }\n\n    protected handleFiles(files: File[]): void {\n        const selection: FileSelection = {\n            valid: [],\n            invalid: [],\n        };\n\n        for (const file of files) {\n            const error = this.validate(file);\n            if (error) {\n                selection.invalid.push({\n                    file: file,\n                    error: error,\n                });\n            } else {\n                selection.valid.push(file);\n            }\n        }\n\n        if (selection.valid.length) {\n            this.fileChange.emit(selection.valid[0]);\n        }\n\n        if (selection.valid.length || selection.invalid.length) {\n            this.filesChange.emit(selection);\n\n            if (this.broadcast) {\n                this.naturalFileService.filesChanged.next(selection);\n            }\n        }\n\n        this.getFileElement().value = '';\n    }\n\n    /**\n     * Called when input has files\n     */\n    private changeFn(event: Event): void {\n        if (!(event.target instanceof HTMLInputElement)) {\n            return;\n        }\n\n        const fileList = event.target.files;\n\n        if (!fileList) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(fileListToArray(fileList));\n    }\n\n    private clickHandler(event: Event): boolean {\n        const elm = this.element.nativeElement;\n        if (elm.getAttribute('disabled') || this.fileSelectionDisabled) {\n            return false;\n        }\n\n        // prevent the click if it is a swipe\n        if (detectSwipe(event)) {\n            return true;\n        }\n\n        const fileElm = this.getFileElement();\n        fileElm.click();\n        this.beforeSelect();\n\n        return false;\n    }\n\n    private beforeSelect(): void {\n        if (!this.fileElement) {\n            return;\n        }\n\n        // if no files in array, be sure browser doesnt prevent reselect of same file (see github issue 27)\n        this.fileElement.value = '';\n    }\n\n    @HostListener('change', ['$event'])\n    public onChange(event: Event): void {\n        const fileList = this.getFileElement().files;\n        const files: File[] = fileList ? fileListToArray(fileList) : eventToFiles(event);\n\n        if (!files.length) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(files);\n    }\n\n    private validate(file: File): string | null {\n        for (const validator of this.validators) {\n            if (!validator.fn.call(this, file)) {\n                return validator.name;\n            }\n        }\n\n        return null;\n    }\n\n    private acceptValidator(item: File): boolean {\n        return acceptType(this.accept, item.type, item.name);\n    }\n\n    private fileSizeValidator(item: File): boolean {\n        return !(this.maxSize && item.size > this.maxSize);\n    }\n}\n"]}
228
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-file.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/abstract-file.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    ElementRef,\n    EventEmitter,\n    HostListener,\n    Inject,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport {\n    acceptType,\n    createInvisibleFileInputWrap,\n    detectSwipe,\n    eventToFiles,\n    fileListToArray,\n    isDirectory,\n    isFileInput,\n    stopEvent,\n} from './utils';\nimport {NaturalFileService} from './file.service';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\nimport {DOCUMENT} from '@angular/common';\nimport {forkJoin, map, Observable, ObservableInput, of, tap} from 'rxjs';\n\nexport interface InvalidFile {\n    file: File;\n    error: string;\n}\n\nexport interface FileSelection {\n    /**\n     * The list of files that have been selected.\n     */\n    valid: File[];\n\n    /**\n     * The list of files that have been selected but are invalid according to validators.\n     */\n    invalid: InvalidFile[];\n}\n\n/**\n * A master base set of logic intended to support file select/drag/drop operations\n *\n * In most cases you probably want click-to-select and drag-to-select, so you should use:\n *\n *     <div naturalFileDrop [selectable]=\"true\"></div>\n *\n * @dynamic\n */\n@Directive()\nexport abstract class NaturalAbstractFile extends NaturalAbstractController implements OnInit, OnDestroy, OnChanges {\n    private fileElement?: HTMLInputElement;\n\n    /**\n     * Whether we should accept a single file or multiple files\n     */\n    @Input() public multiple = false;\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element\n     * it can be a mixed of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept = '';\n\n    /**\n     * Maximum file size in bytes. 0 means no validation at all.\n     */\n    @Input() public maxSize = 0;\n\n    /**\n     * Disable the file selection entirely\n     */\n    @Input() public fileSelectionDisabled = false;\n\n    /**\n     * Whether the user can click on the element to select something\n     *\n     * This has only effect during initialization. Subsequent changes will have\n     * no effect.\n     */\n    @Input() public selectable = false;\n\n    /**\n     * If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.\n     *\n     * It is useful to set this to false if there is two upload on a page with different purpose\n     * and the second upload should not be confused with the first one.\n     */\n    @Input() public broadcast = true;\n\n    /**\n     * The single valid file that has been selected.\n     *\n     * It is for convenience of use, and will only emit if there is at least one\n     * valid file. See `filesChange` for a more complete output.\n     */\n    @Output() public readonly fileChange = new EventEmitter<File>();\n\n    /**\n     * The list of files that have been selected.\n     */\n    @Output() public readonly filesChange = new EventEmitter<FileSelection>();\n\n    public constructor(\n        private readonly element: ElementRef<HTMLElement>,\n        protected readonly naturalFileService: NaturalFileService,\n        @Inject(DOCUMENT) private readonly document: Document,\n    ) {\n        super();\n    }\n\n    public ngOnDestroy(): void {\n        super.ngOnDestroy();\n        delete this.fileElement; // faster memory release of dom element\n    }\n\n    public ngOnInit(): void {\n        if (this.selectable) {\n            this.enableSelecting();\n        }\n\n        this.getFileElement().multiple = this.multiple;\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.accept) {\n            this.getFileElement().setAttribute('accept', changes.accept.currentValue || '*');\n        }\n    }\n\n    private getFileElement(): HTMLInputElement {\n        if (this.fileElement) {\n            return this.fileElement;\n        }\n\n        // elm is a file input\n        if (isFileInput(this.element.nativeElement)) {\n            this.fileElement = this.element.nativeElement;\n\n            return this.fileElement;\n        }\n\n        // create foo file input\n        const label = createInvisibleFileInputWrap(this.document);\n        this.fileElement = label.getElementsByTagName('input')[0];\n        this.fileElement.addEventListener('change', this.changeFn.bind(this));\n        this.element.nativeElement.appendChild(label);\n\n        return this.fileElement;\n    }\n\n    private enableSelecting(): void {\n        const elm = this.element.nativeElement;\n\n        if (isFileInput(elm)) {\n            const bindedHandler = (): void => this.beforeSelect();\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            return;\n        } else {\n            const bindedHandler = (event: Event): boolean => this.clickHandler(event);\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            elm.addEventListener('touchend', bindedHandler);\n        }\n    }\n\n    protected handleFiles(files: File[]): void {\n        const selection: FileSelection = {\n            valid: [],\n            invalid: [],\n        };\n\n        forkJoin(\n            files.map(file =>\n                this.validate(file).pipe(\n                    tap(error => {\n                        if (error) {\n                            selection.invalid.push({\n                                file: file,\n                                error: error,\n                            });\n                        } else {\n                            selection.valid.push(file);\n                        }\n                    }),\n                ),\n            ),\n        ).subscribe(() => {\n            if (selection.valid.length) {\n                this.fileChange.emit(selection.valid[0]);\n            }\n\n            if (selection.valid.length || selection.invalid.length) {\n                this.filesChange.emit(selection);\n\n                if (this.broadcast) {\n                    this.naturalFileService.filesChanged.next(selection);\n                }\n            }\n\n            this.getFileElement().value = '';\n        });\n    }\n\n    /**\n     * Called when input has files\n     */\n    private changeFn(event: Event): void {\n        if (!(event.target instanceof HTMLInputElement)) {\n            return;\n        }\n\n        const fileList = event.target.files;\n\n        if (!fileList) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(fileListToArray(fileList));\n    }\n\n    private clickHandler(event: Event): boolean {\n        const elm = this.element.nativeElement;\n        if (elm.getAttribute('disabled') || this.fileSelectionDisabled) {\n            return false;\n        }\n\n        // prevent the click if it is a swipe\n        if (detectSwipe(event)) {\n            return true;\n        }\n\n        const fileElm = this.getFileElement();\n        fileElm.click();\n        this.beforeSelect();\n\n        return false;\n    }\n\n    private beforeSelect(): void {\n        if (!this.fileElement) {\n            return;\n        }\n\n        // if no files in array, be sure browser doesnt prevent reselect of same file (see github issue 27)\n        this.fileElement.value = '';\n    }\n\n    @HostListener('change', ['$event'])\n    public onChange(event: Event): void {\n        const fileList = this.getFileElement().files;\n        const files: File[] = fileList ? fileListToArray(fileList) : eventToFiles(event);\n\n        if (!files.length) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(files);\n    }\n\n    private validate(file: File): Observable<string | null> {\n        return forkJoin<Record<string, ObservableInput<boolean>>>({\n            accept: of(acceptType(this.accept, file.type, file.name)),\n            fileSize: of(!(this.maxSize && file.size > this.maxSize)),\n            directory: isDirectory(file),\n        }).pipe(\n            map(result => {\n                for (const [key, value] of Object.entries(result)) {\n                    if (!value) {\n                        return key;\n                    }\n                }\n\n                return null;\n            }),\n        );\n    }\n}\n"]}
@@ -69,6 +69,17 @@ function createFileInput(document) {
69
69
  fileElem.type = 'file';
70
70
  return fileElem;
71
71
  }
72
+ export function isDirectory(file) {
73
+ return file
74
+ .slice(0, 1)
75
+ .text()
76
+ .then(text => {
77
+ // Firefox will return empty string for a folder, so we must check that special case.
78
+ // That means that any empty file will incorrectly be interpreted as a folder on all
79
+ // browsers, but that's tolerable because there is no real use-case to upload an empty file.
80
+ return text !== '';
81
+ }, () => false);
82
+ }
72
83
  export function stopEvent(event) {
73
84
  event.preventDefault();
74
85
  event.stopPropagation();
@@ -104,4 +115,4 @@ export function eventToFiles(event) {
104
115
  }
105
116
  return [];
106
117
  }
107
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/utils.ts"],"names":[],"mappings":"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","sourcesContent":["export function acceptType(accept: string, type: string, filename: string): boolean {\n    if (!accept.trim()) {\n        return true;\n    }\n\n    type = type.toLowerCase();\n    filename = filename.toLowerCase();\n\n    return accept.split(',').some(mimeOrExtension => {\n        mimeOrExtension = mimeOrExtension.trim().toLowerCase();\n        if (mimeOrExtension.startsWith('.')) {\n            return filename.endsWith(mimeOrExtension);\n        } else {\n            // Transform `*` into `.*`\n            const pattern = mimeOrExtension.replace(/\\*/g, '.*');\n\n            return type.match(pattern);\n        }\n    });\n}\n\nexport function isFileInput(elm: HTMLElement): elm is HTMLInputElement {\n    const type = elm.getAttribute('type');\n\n    return elm.tagName.toLowerCase() === 'input' && !!type && type.toLowerCase() === 'file';\n}\n\nlet initialTouchStartY = 0;\nlet initialTouchStartX = 0;\n\nexport function detectSwipe(event: Event | TouchEvent): boolean {\n    const touches = 'changedTouches' in event ? event.changedTouches : null;\n    if (!touches) {\n        return false;\n    }\n\n    if (event.type === 'touchstart') {\n        initialTouchStartX = touches[0].clientX;\n        initialTouchStartY = touches[0].clientY;\n\n        return true; // don't block event default\n    } else {\n        // prevent scroll from triggering event\n        if (event.type === 'touchend') {\n            const currentX = touches[0].clientX;\n            const currentY = touches[0].clientY;\n            if (Math.abs(currentX - initialTouchStartX) > 20 || Math.abs(currentY - initialTouchStartY) > 20) {\n                event.stopPropagation();\n                if (event.cancelable) {\n                    event.preventDefault();\n                }\n\n                return false;\n            }\n        }\n\n        return true;\n    }\n}\n\nexport function createInvisibleFileInputWrap(document: Document): HTMLLabelElement {\n    const fileElem = createFileInput(document);\n    const label = document.createElement('label');\n    label.innerHTML = 'upload';\n    label.style.visibility = 'hidden';\n    label.style.position = 'absolute';\n    label.style.overflow = 'hidden';\n    label.style.width = '0px';\n    label.style.height = '0px';\n    label.style.border = 'none';\n    label.style.margin = '0px';\n    label.style.padding = '0px';\n    label.setAttribute('tabindex', '-1');\n\n    label.appendChild(fileElem);\n\n    return label;\n}\n\nfunction createFileInput(document: Document): HTMLInputElement {\n    const fileElem = document.createElement('input');\n    fileElem.type = 'file';\n\n    return fileElem;\n}\n\nexport function stopEvent(event: Event): void {\n    event.preventDefault();\n    event.stopPropagation();\n}\n\nexport function fileListToArray(fileList: FileList): File[] {\n    const result: File[] = [];\n    for (let i = 0; i < fileList.length; i++) {\n        const file = fileList.item(i);\n        if (file) {\n            result.push(file);\n        }\n    }\n\n    return result;\n}\n\nfunction dataTransferItemListToArray(items: DataTransferItemList): File[] {\n    const result: File[] = [];\n\n    // eslint-disable-next-line @typescript-eslint/prefer-for-of\n    for (let i = 0; i < items.length; i++) {\n        const file = items[i].getAsFile();\n        if (file) {\n            result.push(file);\n        }\n    }\n\n    return result;\n}\n\nexport function eventToFiles(event: Event | DragEvent): File[] {\n    const transfer = 'dataTransfer' in event ? event.dataTransfer : null;\n    if (transfer?.files?.length) {\n        return fileListToArray(transfer.files);\n    }\n\n    if (transfer) {\n        return dataTransferItemListToArray(transfer.items);\n    }\n\n    return [];\n}\n"]}
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/utils.ts"],"names":[],"mappings":"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","sourcesContent":["export function acceptType(accept: string, type: string, filename: string): boolean {\n    if (!accept.trim()) {\n        return true;\n    }\n\n    type = type.toLowerCase();\n    filename = filename.toLowerCase();\n\n    return accept.split(',').some(mimeOrExtension => {\n        mimeOrExtension = mimeOrExtension.trim().toLowerCase();\n        if (mimeOrExtension.startsWith('.')) {\n            return filename.endsWith(mimeOrExtension);\n        } else {\n            // Transform `*` into `.*`\n            const pattern = mimeOrExtension.replace(/\\*/g, '.*');\n\n            return type.match(pattern);\n        }\n    });\n}\n\nexport function isFileInput(elm: HTMLElement): elm is HTMLInputElement {\n    const type = elm.getAttribute('type');\n\n    return elm.tagName.toLowerCase() === 'input' && !!type && type.toLowerCase() === 'file';\n}\n\nlet initialTouchStartY = 0;\nlet initialTouchStartX = 0;\n\nexport function detectSwipe(event: Event | TouchEvent): boolean {\n    const touches = 'changedTouches' in event ? event.changedTouches : null;\n    if (!touches) {\n        return false;\n    }\n\n    if (event.type === 'touchstart') {\n        initialTouchStartX = touches[0].clientX;\n        initialTouchStartY = touches[0].clientY;\n\n        return true; // don't block event default\n    } else {\n        // prevent scroll from triggering event\n        if (event.type === 'touchend') {\n            const currentX = touches[0].clientX;\n            const currentY = touches[0].clientY;\n            if (Math.abs(currentX - initialTouchStartX) > 20 || Math.abs(currentY - initialTouchStartY) > 20) {\n                event.stopPropagation();\n                if (event.cancelable) {\n                    event.preventDefault();\n                }\n\n                return false;\n            }\n        }\n\n        return true;\n    }\n}\n\nexport function createInvisibleFileInputWrap(document: Document): HTMLLabelElement {\n    const fileElem = createFileInput(document);\n    const label = document.createElement('label');\n    label.innerHTML = 'upload';\n    label.style.visibility = 'hidden';\n    label.style.position = 'absolute';\n    label.style.overflow = 'hidden';\n    label.style.width = '0px';\n    label.style.height = '0px';\n    label.style.border = 'none';\n    label.style.margin = '0px';\n    label.style.padding = '0px';\n    label.setAttribute('tabindex', '-1');\n\n    label.appendChild(fileElem);\n\n    return label;\n}\n\nfunction createFileInput(document: Document): HTMLInputElement {\n    const fileElem = document.createElement('input');\n    fileElem.type = 'file';\n\n    return fileElem;\n}\n\nexport function isDirectory(file: File): Promise<boolean> {\n    return file\n        .slice(0, 1)\n        .text()\n        .then(\n            text => {\n                // Firefox will return empty string for a folder, so we must check that special case.\n                // That means that any empty file will incorrectly be interpreted as a folder on all\n                // browsers, but that's tolerable because there is no real use-case to upload an empty file.\n                return text !== '';\n            },\n            () => false,\n        );\n}\n\nexport function stopEvent(event: Event): void {\n    event.preventDefault();\n    event.stopPropagation();\n}\n\nexport function fileListToArray(fileList: FileList): File[] {\n    const result: File[] = [];\n    for (let i = 0; i < fileList.length; i++) {\n        const file = fileList.item(i);\n        if (file) {\n            result.push(file);\n        }\n    }\n\n    return result;\n}\n\nfunction dataTransferItemListToArray(items: DataTransferItemList): File[] {\n    const result: File[] = [];\n\n    // eslint-disable-next-line @typescript-eslint/prefer-for-of\n    for (let i = 0; i < items.length; i++) {\n        const file = items[i].getAsFile();\n        if (file) {\n            result.push(file);\n        }\n    }\n\n    return result;\n}\n\nexport function eventToFiles(event: Event | DragEvent): File[] {\n    const transfer = 'dataTransfer' in event ? event.dataTransfer : null;\n    if (transfer?.files?.length) {\n        return fileListToArray(transfer.files);\n    }\n\n    if (transfer) {\n        return dataTransferItemListToArray(transfer.items);\n    }\n\n    return [];\n}\n"]}
@@ -1,12 +1,25 @@
1
1
  import { Component, EventEmitter, Input, Output } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
- import * as i1 from "@angular/common";
4
- import * as i2 from "../fixed-button/fixed-button.component";
5
- import * as i3 from "@angular/material/tooltip";
3
+ import * as i1 from "@angular/router";
4
+ import * as i2 from "@angular/common";
5
+ import * as i3 from "../fixed-button/fixed-button.component";
6
+ import * as i4 from "@angular/material/tooltip";
6
7
  export class NaturalFixedButtonDetailComponent {
7
- constructor() {
8
+ constructor(route) {
9
+ this.canChange = true;
10
+ this.isCreation = false;
8
11
  this.create = new EventEmitter();
9
12
  this.delete = new EventEmitter();
13
+ route.params.subscribe(() => (this.canChange = true));
14
+ }
15
+ get model() {
16
+ return this._model;
17
+ }
18
+ set model(value) {
19
+ this._model = value;
20
+ if (this.canChange) {
21
+ this.isCreation = !this._model.id;
22
+ }
10
23
  }
11
24
  clickCreate() {
12
25
  if (this.form.enabled) {
@@ -19,12 +32,12 @@ export class NaturalFixedButtonDetailComponent {
19
32
  }
20
33
  }
21
34
  }
22
- NaturalFixedButtonDetailComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalFixedButtonDetailComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
23
- NaturalFixedButtonDetailComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.1", type: NaturalFixedButtonDetailComponent, selector: "natural-fixed-button-detail", inputs: { model: "model", form: "form" }, outputs: { create: "create", delete: "delete" }, ngImport: i0, template: "<natural-fixed-button\n (click)=\"clickCreate()\"\n *ngIf=\"!model.id\"\n [disabled]=\"form.disabled\"\n [color]=\"form.valid ? 'accent' : 'warn'\"\n class=\"detail-speed-dial\"\n icon=\"save\"\n></natural-fixed-button>\n\n<natural-fixed-button\n (click)=\"clickDelete()\"\n *ngIf=\"model.id && (!model.permissions || model.permissions.delete)\"\n [disabled]=\"form.disabled\"\n class=\"detail-speed-dial\"\n color=\"warn\"\n icon=\"delete_forever\"\n i18n-matTooltip\n matTooltip=\"Supprimer d\u00E9finitivement\"\n matTooltipPosition=\"left\"\n></natural-fixed-button>\n", styles: [""], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.NaturalFixedButtonComponent, selector: "natural-fixed-button", inputs: ["icon", "link", "color", "disabled"] }, { kind: "directive", type: i3.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }] });
35
+ NaturalFixedButtonDetailComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalFixedButtonDetailComponent, deps: [{ token: i1.ActivatedRoute }], target: i0.ɵɵFactoryTarget.Component });
36
+ NaturalFixedButtonDetailComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.1", type: NaturalFixedButtonDetailComponent, selector: "natural-fixed-button-detail", inputs: { model: "model", form: "form" }, outputs: { create: "create", delete: "delete" }, ngImport: i0, template: "<natural-fixed-button\n (click)=\"clickCreate()\"\n *ngIf=\"isCreation\"\n [disabled]=\"form.disabled\"\n [color]=\"form.valid ? 'accent' : 'warn'\"\n class=\"detail-speed-dial\"\n icon=\"save\"\n></natural-fixed-button>\n\n<natural-fixed-button\n (click)=\"clickDelete()\"\n *ngIf=\"!isCreation && (!model.permissions || model.permissions.delete)\"\n [disabled]=\"form.disabled\"\n class=\"detail-speed-dial\"\n color=\"warn\"\n icon=\"delete_forever\"\n i18n-matTooltip\n matTooltip=\"Supprimer d\u00E9finitivement\"\n matTooltipPosition=\"left\"\n></natural-fixed-button>\n", styles: [""], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.NaturalFixedButtonComponent, selector: "natural-fixed-button", inputs: ["icon", "link", "color", "disabled"] }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }] });
24
37
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalFixedButtonDetailComponent, decorators: [{
25
38
  type: Component,
26
- args: [{ selector: 'natural-fixed-button-detail', template: "<natural-fixed-button\n (click)=\"clickCreate()\"\n *ngIf=\"!model.id\"\n [disabled]=\"form.disabled\"\n [color]=\"form.valid ? 'accent' : 'warn'\"\n class=\"detail-speed-dial\"\n icon=\"save\"\n></natural-fixed-button>\n\n<natural-fixed-button\n (click)=\"clickDelete()\"\n *ngIf=\"model.id && (!model.permissions || model.permissions.delete)\"\n [disabled]=\"form.disabled\"\n class=\"detail-speed-dial\"\n color=\"warn\"\n icon=\"delete_forever\"\n i18n-matTooltip\n matTooltip=\"Supprimer d\u00E9finitivement\"\n matTooltipPosition=\"left\"\n></natural-fixed-button>\n" }]
27
- }], propDecorators: { model: [{
39
+ args: [{ selector: 'natural-fixed-button-detail', template: "<natural-fixed-button\n (click)=\"clickCreate()\"\n *ngIf=\"isCreation\"\n [disabled]=\"form.disabled\"\n [color]=\"form.valid ? 'accent' : 'warn'\"\n class=\"detail-speed-dial\"\n icon=\"save\"\n></natural-fixed-button>\n\n<natural-fixed-button\n (click)=\"clickDelete()\"\n *ngIf=\"!isCreation && (!model.permissions || model.permissions.delete)\"\n [disabled]=\"form.disabled\"\n class=\"detail-speed-dial\"\n color=\"warn\"\n icon=\"delete_forever\"\n i18n-matTooltip\n matTooltip=\"Supprimer d\u00E9finitivement\"\n matTooltipPosition=\"left\"\n></natural-fixed-button>\n" }]
40
+ }], ctorParameters: function () { return [{ type: i1.ActivatedRoute }]; }, propDecorators: { model: [{
28
41
  type: Input
29
42
  }], form: [{
30
43
  type: Input
@@ -33,4 +46,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.1", ngImpor
33
46
  }], delete: [{
34
47
  type: Output
35
48
  }] } });
36
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZml4ZWQtYnV0dG9uLWRldGFpbC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9maXhlZC1idXR0b24tZGV0YWlsL2ZpeGVkLWJ1dHRvbi1kZXRhaWwuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZml4ZWQtYnV0dG9uLWRldGFpbC9maXhlZC1idXR0b24tZGV0YWlsLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUMsTUFBTSxlQUFlLENBQUM7Ozs7O0FBUXJFLE1BQU0sT0FBTyxpQ0FBaUM7SUFMOUM7UUFlOEIsV0FBTSxHQUFHLElBQUksWUFBWSxFQUFRLENBQUM7UUFDbEMsV0FBTSxHQUFHLElBQUksWUFBWSxFQUFRLENBQUM7S0FhL0Q7SUFYVSxXQUFXO1FBQ2QsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRTtZQUNuQixJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxDQUFDO1NBQ3RCO0lBQ0wsQ0FBQztJQUVNLFdBQVc7UUFDZCxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFO1lBQ25CLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7U0FDdEI7SUFDTCxDQUFDOzs4SEF2QlEsaUNBQWlDO2tIQUFqQyxpQ0FBaUMsOEpDUjlDLDJtQkFvQkE7MkZEWmEsaUNBQWlDO2tCQUw3QyxTQUFTOytCQUNJLDZCQUE2Qjs4QkFLdkIsS0FBSztzQkFBcEIsS0FBSztnQkFPVSxJQUFJO3NCQUFuQixLQUFLO2dCQUVvQixNQUFNO3NCQUEvQixNQUFNO2dCQUNtQixNQUFNO3NCQUEvQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgSW5wdXQsIE91dHB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0Zvcm1Hcm91cH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ25hdHVyYWwtZml4ZWQtYnV0dG9uLWRldGFpbCcsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2ZpeGVkLWJ1dHRvbi1kZXRhaWwuY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsczogWycuL2ZpeGVkLWJ1dHRvbi1kZXRhaWwuY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbEZpeGVkQnV0dG9uRGV0YWlsQ29tcG9uZW50IHtcbiAgICBASW5wdXQoKSBwdWJsaWMgbW9kZWwhOiB7XG4gICAgICAgIGlkPzogc3RyaW5nO1xuICAgICAgICBwZXJtaXNzaW9ucz86IHtcbiAgICAgICAgICAgIGRlbGV0ZTogYm9vbGVhbjtcbiAgICAgICAgfTtcbiAgICB9O1xuXG4gICAgQElucHV0KCkgcHVibGljIGZvcm0hOiBGb3JtR3JvdXA7XG5cbiAgICBAT3V0cHV0KCkgcHVibGljIHJlYWRvbmx5IGNyZWF0ZSA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcbiAgICBAT3V0cHV0KCkgcHVibGljIHJlYWRvbmx5IGRlbGV0ZSA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcblxuICAgIHB1YmxpYyBjbGlja0NyZWF0ZSgpOiB2b2lkIHtcbiAgICAgICAgaWYgKHRoaXMuZm9ybS5lbmFibGVkKSB7XG4gICAgICAgICAgICB0aGlzLmNyZWF0ZS5lbWl0KCk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgY2xpY2tEZWxldGUoKTogdm9pZCB7XG4gICAgICAgIGlmICh0aGlzLmZvcm0uZW5hYmxlZCkge1xuICAgICAgICAgICAgdGhpcy5kZWxldGUuZW1pdCgpO1xuICAgICAgICB9XG4gICAgfVxufVxuIiwiPG5hdHVyYWwtZml4ZWQtYnV0dG9uXG4gICAgKGNsaWNrKT1cImNsaWNrQ3JlYXRlKClcIlxuICAgICpuZ0lmPVwiIW1vZGVsLmlkXCJcbiAgICBbZGlzYWJsZWRdPVwiZm9ybS5kaXNhYmxlZFwiXG4gICAgW2NvbG9yXT1cImZvcm0udmFsaWQgPyAnYWNjZW50JyA6ICd3YXJuJ1wiXG4gICAgY2xhc3M9XCJkZXRhaWwtc3BlZWQtZGlhbFwiXG4gICAgaWNvbj1cInNhdmVcIlxuPjwvbmF0dXJhbC1maXhlZC1idXR0b24+XG5cbjxuYXR1cmFsLWZpeGVkLWJ1dHRvblxuICAgIChjbGljayk9XCJjbGlja0RlbGV0ZSgpXCJcbiAgICAqbmdJZj1cIm1vZGVsLmlkICYmICghbW9kZWwucGVybWlzc2lvbnMgfHwgbW9kZWwucGVybWlzc2lvbnMuZGVsZXRlKVwiXG4gICAgW2Rpc2FibGVkXT1cImZvcm0uZGlzYWJsZWRcIlxuICAgIGNsYXNzPVwiZGV0YWlsLXNwZWVkLWRpYWxcIlxuICAgIGNvbG9yPVwid2FyblwiXG4gICAgaWNvbj1cImRlbGV0ZV9mb3JldmVyXCJcbiAgICBpMThuLW1hdFRvb2x0aXBcbiAgICBtYXRUb29sdGlwPVwiU3VwcHJpbWVyIGTDqWZpbml0aXZlbWVudFwiXG4gICAgbWF0VG9vbHRpcFBvc2l0aW9uPVwibGVmdFwiXG4+PC9uYXR1cmFsLWZpeGVkLWJ1dHRvbj5cbiJdfQ==
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZml4ZWQtYnV0dG9uLWRldGFpbC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9maXhlZC1idXR0b24tZGV0YWlsL2ZpeGVkLWJ1dHRvbi1kZXRhaWwuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZml4ZWQtYnV0dG9uLWRldGFpbC9maXhlZC1idXR0b24tZGV0YWlsLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUMsTUFBTSxlQUFlLENBQUM7Ozs7OztBQWdCckUsTUFBTSxPQUFPLGlDQUFpQztJQXVCMUMsWUFBbUIsS0FBcUI7UUF0QmhDLGNBQVMsR0FBRyxJQUFJLENBQUM7UUFDbEIsZUFBVSxHQUFHLEtBQUssQ0FBQztRQWtCQSxXQUFNLEdBQUcsSUFBSSxZQUFZLEVBQVEsQ0FBQztRQUNsQyxXQUFNLEdBQUcsSUFBSSxZQUFZLEVBQVEsQ0FBQztRQUd4RCxLQUFLLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUMxRCxDQUFDO0lBckJELElBQVcsS0FBSztRQUNaLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQztJQUN2QixDQUFDO0lBRUQsSUFDVyxLQUFLLENBQUMsS0FBWTtRQUN6QixJQUFJLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQztRQUNwQixJQUFJLElBQUksQ0FBQyxTQUFTLEVBQUU7WUFDaEIsSUFBSSxDQUFDLFVBQVUsR0FBRyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDO1NBQ3JDO0lBQ0wsQ0FBQztJQWFNLFdBQVc7UUFDZCxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFO1lBQ25CLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7U0FDdEI7SUFDTCxDQUFDO0lBRU0sV0FBVztRQUNkLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUU7WUFDbkIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztTQUN0QjtJQUNMLENBQUM7OzhIQXJDUSxpQ0FBaUM7a0hBQWpDLGlDQUFpQyw4SkNoQjlDLCttQkFvQkE7MkZESmEsaUNBQWlDO2tCQUw3QyxTQUFTOytCQUNJLDZCQUE2QjtxR0FhNUIsS0FBSztzQkFEZixLQUFLO2dCQVVVLElBQUk7c0JBQW5CLEtBQUs7Z0JBRW9CLE1BQU07c0JBQS9CLE1BQU07Z0JBQ21CLE1BQU07c0JBQS9CLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0NvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgT3V0cHV0fSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7Rm9ybUdyb3VwfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQge0FjdGl2YXRlZFJvdXRlfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuXG50eXBlIE1vZGVsID0ge1xuICAgIGlkPzogc3RyaW5nO1xuICAgIHBlcm1pc3Npb25zPzoge1xuICAgICAgICBkZWxldGU6IGJvb2xlYW47XG4gICAgfTtcbn07XG5cbkBDb21wb25lbnQoe1xuICAgIHNlbGVjdG9yOiAnbmF0dXJhbC1maXhlZC1idXR0b24tZGV0YWlsJyxcbiAgICB0ZW1wbGF0ZVVybDogJy4vZml4ZWQtYnV0dG9uLWRldGFpbC5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmxzOiBbJy4vZml4ZWQtYnV0dG9uLWRldGFpbC5jb21wb25lbnQuc2NzcyddLFxufSlcbmV4cG9ydCBjbGFzcyBOYXR1cmFsRml4ZWRCdXR0b25EZXRhaWxDb21wb25lbnQge1xuICAgIHByaXZhdGUgY2FuQ2hhbmdlID0gdHJ1ZTtcbiAgICBwdWJsaWMgaXNDcmVhdGlvbiA9IGZhbHNlO1xuXG4gICAgcHVibGljIGdldCBtb2RlbCgpOiBNb2RlbCB7XG4gICAgICAgIHJldHVybiB0aGlzLl9tb2RlbDtcbiAgICB9XG5cbiAgICBASW5wdXQoKVxuICAgIHB1YmxpYyBzZXQgbW9kZWwodmFsdWU6IE1vZGVsKSB7XG4gICAgICAgIHRoaXMuX21vZGVsID0gdmFsdWU7XG4gICAgICAgIGlmICh0aGlzLmNhbkNoYW5nZSkge1xuICAgICAgICAgICAgdGhpcy5pc0NyZWF0aW9uID0gIXRoaXMuX21vZGVsLmlkO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBfbW9kZWwhOiBNb2RlbDtcblxuICAgIEBJbnB1dCgpIHB1YmxpYyBmb3JtITogRm9ybUdyb3VwO1xuXG4gICAgQE91dHB1dCgpIHB1YmxpYyByZWFkb25seSBjcmVhdGUgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG4gICAgQE91dHB1dCgpIHB1YmxpYyByZWFkb25seSBkZWxldGUgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG5cbiAgICBwdWJsaWMgY29uc3RydWN0b3Iocm91dGU6IEFjdGl2YXRlZFJvdXRlKSB7XG4gICAgICAgIHJvdXRlLnBhcmFtcy5zdWJzY3JpYmUoKCkgPT4gKHRoaXMuY2FuQ2hhbmdlID0gdHJ1ZSkpO1xuICAgIH1cblxuICAgIHB1YmxpYyBjbGlja0NyZWF0ZSgpOiB2b2lkIHtcbiAgICAgICAgaWYgKHRoaXMuZm9ybS5lbmFibGVkKSB7XG4gICAgICAgICAgICB0aGlzLmNyZWF0ZS5lbWl0KCk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgY2xpY2tEZWxldGUoKTogdm9pZCB7XG4gICAgICAgIGlmICh0aGlzLmZvcm0uZW5hYmxlZCkge1xuICAgICAgICAgICAgdGhpcy5kZWxldGUuZW1pdCgpO1xuICAgICAgICB9XG4gICAgfVxufVxuIiwiPG5hdHVyYWwtZml4ZWQtYnV0dG9uXG4gICAgKGNsaWNrKT1cImNsaWNrQ3JlYXRlKClcIlxuICAgICpuZ0lmPVwiaXNDcmVhdGlvblwiXG4gICAgW2Rpc2FibGVkXT1cImZvcm0uZGlzYWJsZWRcIlxuICAgIFtjb2xvcl09XCJmb3JtLnZhbGlkID8gJ2FjY2VudCcgOiAnd2FybidcIlxuICAgIGNsYXNzPVwiZGV0YWlsLXNwZWVkLWRpYWxcIlxuICAgIGljb249XCJzYXZlXCJcbj48L25hdHVyYWwtZml4ZWQtYnV0dG9uPlxuXG48bmF0dXJhbC1maXhlZC1idXR0b25cbiAgICAoY2xpY2spPVwiY2xpY2tEZWxldGUoKVwiXG4gICAgKm5nSWY9XCIhaXNDcmVhdGlvbiAmJiAoIW1vZGVsLnBlcm1pc3Npb25zIHx8IG1vZGVsLnBlcm1pc3Npb25zLmRlbGV0ZSlcIlxuICAgIFtkaXNhYmxlZF09XCJmb3JtLmRpc2FibGVkXCJcbiAgICBjbGFzcz1cImRldGFpbC1zcGVlZC1kaWFsXCJcbiAgICBjb2xvcj1cIndhcm5cIlxuICAgIGljb249XCJkZWxldGVfZm9yZXZlclwiXG4gICAgaTE4bi1tYXRUb29sdGlwXG4gICAgbWF0VG9vbHRpcD1cIlN1cHByaW1lciBkw6lmaW5pdGl2ZW1lbnRcIlxuICAgIG1hdFRvb2x0aXBQb3NpdGlvbj1cImxlZnRcIlxuPjwvbmF0dXJhbC1maXhlZC1idXR0b24+XG4iXX0=
@@ -26,10 +26,10 @@ export class NaturalGroupComponent {
26
26
  }
27
27
  }
28
28
  NaturalGroupComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalGroupComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
29
- NaturalGroupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.1", type: NaturalGroupComponent, selector: "natural-group", inputs: { placeholder: "placeholder", facets: "facets", selections: "selections" }, outputs: { selectionChange: "selectionChange" }, viewQueries: [{ propertyName: "newValueInput", first: true, predicate: ["newValueInput"], descendants: true }], ngImport: i0, template: "<natural-input\n (cleared)=\"removeInput(i)\"\n (selectionChange)=\"updateInput($event, i)\"\n *ngFor=\"let selection of innerSelections; let i = index\"\n [facets]=\"facets\"\n [selection]=\"selection\"\n></natural-input>\n\n<natural-input\n #newValueInput\n (selectionChange)=\"addInput($event)\"\n [facets]=\"facets\"\n [placeholder]=\"placeholder\"\n tabIndex=\"10\"\n cdkFocusInitial\n></natural-input>\n", styles: [":host{display:flex;flex-direction:row;flex-wrap:wrap}:host natural-input{flex:none;display:inline-flex;margin-right:10px}:host natural-input:last-of-type{flex:1;margin-right:0;min-width:250px}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.NaturalInputComponent, selector: "natural-input", inputs: ["placeholder", "searchFieldName", "selection", "facets"], outputs: ["selectionChange", "cleared"] }] });
29
+ NaturalGroupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.1", type: NaturalGroupComponent, selector: "natural-group", inputs: { placeholder: "placeholder", facets: "facets", selections: "selections" }, outputs: { selectionChange: "selectionChange" }, viewQueries: [{ propertyName: "newValueInput", first: true, predicate: ["newValueInput"], descendants: true }], ngImport: i0, template: "<natural-input\n (cleared)=\"removeInput(i)\"\n (selectionChange)=\"updateInput($event, i)\"\n *ngFor=\"let selection of innerSelections; let i = index\"\n [facets]=\"facets\"\n [selection]=\"selection\"\n></natural-input>\n\n<natural-input\n #newValueInput\n (selectionChange)=\"addInput($event)\"\n [facets]=\"facets\"\n [placeholder]=\"placeholder\"\n tabIndex=\"10\"\n cdkFocusInitial\n></natural-input>\n", styles: [":host{display:flex;flex-direction:row;flex-wrap:wrap}:host natural-input{flex:auto;display:inline-flex;margin-right:10px;margin-bottom:10px}:host natural-input:last-of-type{flex:1;min-width:250px}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.NaturalInputComponent, selector: "natural-input", inputs: ["placeholder", "searchFieldName", "selection", "facets"], outputs: ["selectionChange", "cleared"] }] });
30
30
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: NaturalGroupComponent, decorators: [{
31
31
  type: Component,
32
- args: [{ selector: 'natural-group', template: "<natural-input\n (cleared)=\"removeInput(i)\"\n (selectionChange)=\"updateInput($event, i)\"\n *ngFor=\"let selection of innerSelections; let i = index\"\n [facets]=\"facets\"\n [selection]=\"selection\"\n></natural-input>\n\n<natural-input\n #newValueInput\n (selectionChange)=\"addInput($event)\"\n [facets]=\"facets\"\n [placeholder]=\"placeholder\"\n tabIndex=\"10\"\n cdkFocusInitial\n></natural-input>\n", styles: [":host{display:flex;flex-direction:row;flex-wrap:wrap}:host natural-input{flex:none;display:inline-flex;margin-right:10px}:host natural-input:last-of-type{flex:1;margin-right:0;min-width:250px}\n"] }]
32
+ args: [{ selector: 'natural-group', template: "<natural-input\n (cleared)=\"removeInput(i)\"\n (selectionChange)=\"updateInput($event, i)\"\n *ngFor=\"let selection of innerSelections; let i = index\"\n [facets]=\"facets\"\n [selection]=\"selection\"\n></natural-input>\n\n<natural-input\n #newValueInput\n (selectionChange)=\"addInput($event)\"\n [facets]=\"facets\"\n [placeholder]=\"placeholder\"\n tabIndex=\"10\"\n cdkFocusInitial\n></natural-input>\n", styles: [":host{display:flex;flex-direction:row;flex-wrap:wrap}:host natural-input{flex:auto;display:inline-flex;margin-right:10px;margin-bottom:10px}:host natural-input:last-of-type{flex:1;min-width:250px}\n"] }]
33
33
  }], propDecorators: { newValueInput: [{
34
34
  type: ViewChild,
35
35
  args: ['newValueInput']