ngx-material-entity 16.0.6 → 16.0.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.d.ts +10 -5
- package/components/input/array/array-string-chips-input/array-string-chips-input.component.d.ts +4 -2
- package/components/input/input.component.d.ts +28 -8
- package/components/input/number/number-dropdown-input/number-dropdown-input.component.d.ts +12 -2
- package/components/input/relations/references-many-input/references-many-input.component.d.ts +10 -2
- package/components/input/string/string-autocomplete-input/string-autocomplete-input.component.d.ts +4 -2
- package/components/input/string/string-dropdown-input/string-dropdown-input.component.d.ts +12 -2
- package/components/input/string/string-password-input/string-password-input.component.d.ts +12 -0
- package/components/table/create-dialog/create-entity-dialog.component.d.ts +1 -1
- package/components/table/edit-dialog/edit-entity-dialog.component.d.ts +1 -1
- package/components/table/table-data.builder.d.ts +2 -1
- package/components/table/table-data.d.ts +10 -0
- package/decorators/array/array-decorator-internal.data.d.ts +2 -1
- package/decorators/array/array-decorator.data.d.ts +3 -1
- package/decorators/string/string-decorator-internal.data.d.ts +3 -1
- package/decorators/string/string-decorator.data.d.ts +7 -1
- package/directives/dynamic-style-class.directive.d.ts +25 -0
- package/directives/included-in.directive.d.ts +1 -1
- package/esm2022/components/create-page/create-page.component.mjs +8 -5
- package/esm2022/components/edit-page/edit-page.component.mjs +8 -5
- package/esm2022/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.mjs +28 -15
- package/esm2022/components/input/array/array-string-chips-input/array-string-chips-input.component.mjs +14 -19
- package/esm2022/components/input/base-input.component.mjs +7 -2
- package/esm2022/components/input/input.component.mjs +73 -27
- package/esm2022/components/input/number/number-dropdown-input/number-dropdown-input.component.mjs +36 -8
- package/esm2022/components/input/relations/references-many-input/references-many-input.component.mjs +29 -5
- package/esm2022/components/input/string/string-autocomplete-input/string-autocomplete-input.component.mjs +17 -10
- package/esm2022/components/input/string/string-dropdown-input/string-dropdown-input.component.mjs +36 -8
- package/esm2022/components/input/string/string-password-input/string-password-input.component.mjs +57 -3
- package/esm2022/components/table/create-dialog/create-entity-dialog.component.mjs +4 -4
- package/esm2022/components/table/edit-dialog/edit-entity-dialog.component.mjs +9 -5
- package/esm2022/components/table/table-data.builder.mjs +5 -1
- package/esm2022/components/table/table-data.mjs +1 -1
- package/esm2022/components/table/table.component.mjs +6 -4
- package/esm2022/decorators/array/array-decorator-internal.data.mjs +11 -3
- package/esm2022/decorators/array/array-decorator.data.mjs +1 -1
- package/esm2022/decorators/string/string-decorator-internal.data.mjs +14 -3
- package/esm2022/decorators/string/string-decorator.data.mjs +1 -1
- package/esm2022/directives/dynamic-style-class.directive.mjs +51 -0
- package/esm2022/directives/included-in.directive.mjs +2 -3
- package/esm2022/encapsulation/reflect.utilities.mjs +6 -2
- package/esm2022/functions/default-style-classes.function.mjs +7 -0
- package/esm2022/public-api.mjs +6 -1
- package/esm2022/services/entity.service.mjs +3 -1
- package/esm2022/utilities/entity.utilities.mjs +37 -33
- package/esm2022/utilities/validation.utilities.mjs +75 -10
- package/fesm2022/ngx-material-entity.mjs +514 -156
- package/fesm2022/ngx-material-entity.mjs.map +1 -1
- package/functions/default-style-classes.function.d.ts +6 -0
- package/package.json +1 -1
- package/public-api.d.ts +4 -0
- package/utilities/entity.utilities.d.ts +4 -3
- package/utilities/validation.utilities.d.ts +8 -3
|
@@ -122,7 +122,7 @@ export class NgxMatEntityCreatePageComponent {
|
|
|
122
122
|
this.entity = new this.EntityClass();
|
|
123
123
|
EntityUtilities.setDefaultValues(this.entity);
|
|
124
124
|
this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);
|
|
125
|
-
this.checkIsEntityValid();
|
|
125
|
+
void this.checkIsEntityValid();
|
|
126
126
|
this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, true, false);
|
|
127
127
|
setTimeout(() => this.checkOffset(), 1);
|
|
128
128
|
// setTimeout(() => this.checkEntity(), 1);
|
|
@@ -156,11 +156,11 @@ export class NgxMatEntityCreatePageComponent {
|
|
|
156
156
|
* Checks if the entity has become invalid or dirty.
|
|
157
157
|
*/
|
|
158
158
|
async checkEntity() {
|
|
159
|
-
this.checkIsEntityValid();
|
|
159
|
+
await this.checkIsEntityValid();
|
|
160
160
|
this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);
|
|
161
161
|
}
|
|
162
|
-
checkIsEntityValid() {
|
|
163
|
-
this.validationErrors = ValidationUtilities.getEntityValidationErrors(this.entity, 'create');
|
|
162
|
+
async checkIsEntityValid() {
|
|
163
|
+
this.validationErrors = await ValidationUtilities.getEntityValidationErrors(this.entity, this.injector, 'create');
|
|
164
164
|
this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));
|
|
165
165
|
this.isEntityValid = this.validationErrors.length === 0;
|
|
166
166
|
}
|
|
@@ -235,6 +235,9 @@ export class NgxMatEntityCreatePageComponent {
|
|
|
235
235
|
isReadOnly(key) {
|
|
236
236
|
return runInInjectionContext(this.injector, () => {
|
|
237
237
|
const metadata = EntityUtilities.getPropertyMetadata(this.entity, key);
|
|
238
|
+
if (!metadata) {
|
|
239
|
+
throw new Error(`No metadata was found for the key "${String(key)}"`);
|
|
240
|
+
}
|
|
238
241
|
return metadata.isReadOnly(this.entity);
|
|
239
242
|
});
|
|
240
243
|
}
|
|
@@ -272,4 +275,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImpor
|
|
|
272
275
|
type: HostListener,
|
|
273
276
|
args: ['window:scroll']
|
|
274
277
|
}] } });
|
|
275
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-page.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/create-page/create-page.component.ts","../../../../../projects/ngx-material-entity/src/components/create-page/create-page.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Location, NgFor, NgIf } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EnvironmentInjector, HostListener, Inject, InjectionToken, OnInit, Renderer2, runInInjectionContext } from '@angular/core';\nimport { MatBadgeModule } from '@angular/material/badge';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { MatTabsModule } from '@angular/material/tabs';\nimport { Observable, first, map } from 'rxjs';\nimport { BaseEntityType, EntityClassNewable } from '../../classes/entity.model';\nimport { PropertyDecoratorConfigInternal } from '../../decorators/base/property-decorator-internal.data';\nimport { NGX_INTERNAL_GLOBAL_DEFAULT_VALUES } from '../../default-global-configuration-values';\nimport { LodashUtilities } from '../../encapsulation/lodash.utilities';\nimport { getValidationErrorsTooltipContent } from '../../functions/get-validation-errors-tooltip-content.function.ts';\nimport { NgxGlobalDefaultValues } from '../../global-configuration-values';\nimport { EntityService } from '../../services/entity.service';\nimport { EntityTab, EntityUtilities } from '../../utilities/entity.utilities';\nimport { ValidationError, ValidationUtilities } from '../../utilities/validation.utilities';\nimport { ConfirmDialogData } from '../confirm-dialog/confirm-dialog-data';\nimport { ConfirmDialogDataBuilder, ConfirmDialogDataInternal } from '../confirm-dialog/confirm-dialog-data.builder';\nimport { NgxMatEntityConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component';\nimport { NgxMatEntityInputModule } from '../input/input.module';\nimport { CreateEntityData } from '../table/create-dialog/create-entity-dialog-data';\nimport { CreateData } from '../table/table-data';\nimport { TooltipComponent } from '../tooltip/tooltip.component';\nimport { PageCreateDataBuilder, PageCreateDataInternal } from './page-create-data.builder';\n\n/**\n * The data that needs to be provided for a route to be able to create a entity.\n */\n// eslint-disable-next-line max-len\nexport type PageCreateData<EntityType extends BaseEntityType<EntityType>> = Omit<CreateEntityData<EntityType>, 'entity' | 'EntityServiceClass'> & {\n    /**\n     * Whether or not to display a loading spinner while the data for the page is loaded.\n     * @default true\n     */\n    displayLoadingSpinner?: boolean,\n    /**\n     * The data of the default create page.\n     */\n    createData?: CreateData & {\n        /**\n         * The data for the dialog when the user tries to leave the site with unsaved changes.\n         */\n        confirmUnsavedChangesDialogData?: ConfirmDialogData,\n        /**\n         * Whether or not leaving with unsaved changes should require a confirm dialog.\n         * @default true\n         */\n        unsavedChangesRequireConfirmDialog?: boolean\n    }\n};\n\n/**\n * The entity service that needs to be provided in the providers array of the create page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_CREATE_DATA_ENTITY_SERVICE: InjectionToken<EntityService<any>> = new InjectionToken<EntityService<any>>('NGX_CREATE_DATA_ENTITY_SERVICE');\n/**\n * The entity class that needs to be provided in the providers array of the create page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_CREATE_DATA_ENTITY: InjectionToken<EntityClassNewable<any>> = new InjectionToken<EntityClassNewable<any>>('NGX_CREATE_DATA_ENTITY');\n/**\n * The configuration that needs to be provided in the providers array of the create page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any, constCase/uppercase\nexport const NGX_CREATE_DATA: InjectionToken<PageCreateData<any>> = new InjectionToken<PageCreateData<any>>('NGX_CREATE_DATA');\n\n/**\n * A generic page that allows you to create a specific entity.\n * For this to work you need to provide some data for the route.\n */\n@Component({\n    selector: 'ngx-mat-entity-create-page',\n    templateUrl: './create-page.component.html',\n    styleUrls: ['./create-page.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatButtonModule,\n        MatTabsModule,\n        NgxMatEntityInputModule,\n        MatProgressSpinnerModule,\n        MatBadgeModule,\n        TooltipComponent\n    ]\n})\nexport class NgxMatEntityCreatePageComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    /**\n     * Contains HelperMethods around handling Entities and their property-metadata.\n     */\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The tabs to display.\n     */\n    entityTabs!: EntityTab<EntityType>[];\n\n    /**\n     * The entity to create.\n     */\n    entity!: EntityType;\n    /**\n     * The entity before any changes have been made.\n     * This is used to determine if leaving the page is possible without interruption.\n     */\n    entityPriorChanges!: EntityType;\n\n    /**\n     * Configuration data for this component.\n     */\n    data!: PageCreateDataInternal<EntityType>;\n\n    /**\n     * All validation errors of the entity.\n     */\n    validationErrors!: ValidationError[];\n    /**\n     * Whether or not the entity is valid.\n     */\n    isEntityValid: boolean = false;\n    /**\n     * Whether or not the entity is dirty.\n     */\n    isEntityDirty: boolean = false;\n    /**\n     * What to display inside the tooltip.\n     */\n    tooltipContent: string = '';\n\n    private inConfirmNavigation: boolean = false;\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    get hasUnsavedChanges(): boolean {\n        return this.isEntityDirty && this.data.createData.unsavedChangesRequireConfirmDialog;\n    }\n\n    constructor(\n        private readonly dialog: MatDialog,\n        private readonly location: Location,\n        private readonly injector: EnvironmentInjector,\n        @Inject(NGX_CREATE_DATA_ENTITY_SERVICE)\n        readonly entityService: EntityService<EntityType>,\n        @Inject(NGX_CREATE_DATA_ENTITY)\n        private readonly EntityClass: EntityClassNewable<EntityType>,\n        @Inject(NGX_CREATE_DATA)\n        private readonly inputData: PageCreateData<EntityType>,\n        private readonly http: HttpClient,\n        private readonly el: ElementRef,\n        private readonly renderer: Renderer2,\n        @Inject(NGX_INTERNAL_GLOBAL_DEFAULT_VALUES)\n        protected readonly globalConfig: NgxGlobalDefaultValues\n    ) { }\n\n    ngOnInit(): void {\n        this.data = new PageCreateDataBuilder(this.inputData, this.globalConfig).getResult();\n        if (this.data == null) {\n            this.confirmNavigateBack();\n            // eslint-disable-next-line max-len\n            throw new Error('No create data was provided for \"NGX_CREATE_DATA\". You need to provide a value in your routes providers array.');\n        }\n\n        this.entity = new this.EntityClass();\n        EntityUtilities.setDefaultValues(this.entity);\n        this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);\n        this.checkIsEntityValid();\n\n        this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, true, false);\n        setTimeout(() => this.checkOffset(), 1);\n        // setTimeout(() => this.checkEntity(), 1);\n    }\n\n    /**\n     * Whether the page can be left without confirmation (of unsaved changes).\n     * @returns Whether or not the page can be left without confirmation.\n     */\n    @HostListener('window:beforeunload')\n    canDeactivate(): boolean {\n        return !this.hasUnsavedChanges || this.inConfirmNavigation;\n    }\n\n    /**\n     * Checks if the bottom row should be displayed as fixed.\n     */\n    @HostListener('window:scroll')\n    checkOffset(): void {\n        const scrollY: number = window.scrollY;\n        const bottomRow: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row');\n        const bottomRowContainer: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row-container');\n\n        if (bottomRow && bottomRowContainer) {\n            const bottomRowContainerOffset: number = bottomRowContainer.offsetTop;\n            const windowHeight: number = window.innerHeight;\n\n            if (scrollY + windowHeight >= bottomRowContainerOffset) {\n                this.renderer.removeClass(bottomRow, 'fixed');\n            }\n            else {\n                this.renderer.addClass(bottomRow, 'fixed');\n            }\n        }\n    }\n\n    /**\n     * Checks if the entity has become invalid or dirty.\n     */\n    async checkEntity(): Promise<void> {\n        this.checkIsEntityValid();\n        this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);\n    }\n\n    private checkIsEntityValid(): void {\n        this.validationErrors = ValidationUtilities.getEntityValidationErrors(this.entity, 'create');\n        this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));\n        this.isEntityValid = this.validationErrors.length === 0;\n    }\n\n    /**\n     * Tries create the entity and navigate back afterwards.\n     * Also handles the confirmation if required.\n     */\n    create(): void {\n        if (!this.isEntityValid) {\n            return;\n        }\n        if (!this.data.createData.createRequiresConfirmDialog) {\n            this.confirmCreate();\n            return;\n        }\n        // eslint-disable-next-line max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.createData.confirmCreateDialogData)\n            .withDefault('text', this.globalConfig.confirmCreateText)\n            .withDefault('confirmButtonLabel', this.globalConfig.createLabel)\n            .withDefault('title', this.globalConfig.createLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmCreate();\n            }\n        });\n    }\n\n    private confirmCreate(): void {\n        void this.entityService.create(this.entity).then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to navigate back.\n     */\n    navigateBack(): void {\n        if (!this.hasUnsavedChanges) {\n            this.confirmNavigateBack();\n            return;\n        }\n\n        this.openConfirmNavigationDialog().subscribe(res => {\n            if (res) {\n                this.confirmNavigateBack();\n            }\n        });\n    }\n\n    /**\n     * Opens the confirm dialog for navigating with unsaved changes.\n     * This is exposed because the UnsavedChangesGuard needs to access this.\n     * @returns The first observable result of the confirm dialog.\n     */\n    openConfirmNavigationDialog(): Observable<boolean> {\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: this.data.createData.confirmUnsavedChangesDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        return dialogRef.afterClosed().pipe(first(), map(p => (p ?? false)));\n    }\n\n    private confirmNavigateBack(): void {\n        this.inConfirmNavigation = true;\n        this.location.back();\n    }\n\n    /**\n     * Checks if the input with the given key is readonly.\n     * @param key - The key for the input to check.\n     * @returns Whether or not the input for the key is read only.\n     */\n    isReadOnly(key: keyof EntityType): boolean {\n        return runInInjectionContext(this.injector, () => {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = EntityUtilities.getPropertyMetadata(this.entity, key);\n            return metadata.isReadOnly(this.entity);\n        });\n    }\n}","<!-- eslint-disable angular/no-call-expression -->\n<!-- eslint-disable angular/cyclomatic-complexity -->\n<div *ngIf=\"!entityTabs && data.displayLoadingSpinner\" class=\"container\">\n    <br>\n    <mat-spinner></mat-spinner>\n    <br>\n</div>\n\n<div *ngIf=\"entityTabs\" class=\"container\">\n    <br>\n\n    <!------------>\n    <!-- Header -->\n    <!------------>\n    <div class=\"header\">\n        <div class=\"cancel-container\">\n            <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n                <i class=\"fas fa-chevron-left\"></i>\n                {{data.createData.cancelButtonLabel}}\n                <i *ngIf=\"hasUnsavedChanges\" class=\"fas fa-warning\"></i>\n            </button>\n        </div>\n    </div>\n\n    <h1>{{data.createData.title}}</h1>\n\n    <!----------->\n    <!-- Input -->\n    <!----------->\n    <form>\n        <mat-tab-group *ngIf=\"entityTabs.length > 1\" preserveContent (selectedTabChange)=\"checkOffset()\">\n            <mat-tab *ngFor=\"let tab of entityTabs\" [label]=\"tab.tabName\">\n                <div *ngFor=\"let row of tab.rows\" class=\"row\">\n                    <ngx-mat-entity-input\n                        *ngFor=\"let key of row.keys\"\n                        [entity]=\"entity\"\n                        [propertyKey]=\"key\"\n                        [hideOmitForEdit]=\"true\"\n                        [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                        [isReadOnly]=\"isReadOnly(key)\"\n                        (inputChangeEvent)=\"checkEntity()\"\n                    >\n                    </ngx-mat-entity-input>\n                </div>\n            </mat-tab>\n        </mat-tab-group>\n    \n        <div *ngIf=\"entityTabs.length <= 1\">\n            <span *ngIf=\"!entityTabs.length\" class=\"no-entity-tabs\">\n                ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n            </span>\n            <div *ngFor=\"let row of entityTabs[0]?.rows\" class=\"row\">\n                <ngx-mat-entity-input\n                    *ngFor=\"let key of row.keys\"\n                    [entity]=\"entity\"\n                    [propertyKey]=\"key\"\n                    [hideOmitForEdit]=\"true\"\n                    [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                    [isReadOnly]=\"isReadOnly(key)\"\n                    (inputChangeEvent)=\"checkEntity()\"\n                >\n                </ngx-mat-entity-input>\n            </div>\n        </div>\n\n        <div class=\"bottom-row-container\">\n            <div class=\"bottom-row container\" style=\"margin-top: 10px;\">\n                <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n                    [disabled]=\"!isEntityValid || !isEntityDirty\"\n                    [matBadge]=\"validationErrors.length\"\n                    [matBadgeHidden]=\"!validationErrors.length\"\n                    (click)=\"create()\"\n                >\n                    {{data.createData.createButtonLabel}}\n                </button>\n                <ngx-mat-entity-tooltip *ngIf=\"validationErrors.length\" [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n            </div>\n        </div>\n    </form>\n\n    <br>\n</div>"]}
|
|
278
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-page.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/create-page/create-page.component.ts","../../../../../projects/ngx-material-entity/src/components/create-page/create-page.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Location, NgFor, NgIf } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EnvironmentInjector, HostListener, Inject, InjectionToken, OnInit, Renderer2, runInInjectionContext } from '@angular/core';\nimport { MatBadgeModule } from '@angular/material/badge';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { MatTabsModule } from '@angular/material/tabs';\nimport { Observable, first, map } from 'rxjs';\nimport { BaseEntityType, EntityClassNewable } from '../../classes/entity.model';\nimport { PropertyDecoratorConfigInternal } from '../../decorators/base/property-decorator-internal.data';\nimport { NGX_INTERNAL_GLOBAL_DEFAULT_VALUES } from '../../default-global-configuration-values';\nimport { LodashUtilities } from '../../encapsulation/lodash.utilities';\nimport { getValidationErrorsTooltipContent } from '../../functions/get-validation-errors-tooltip-content.function.ts';\nimport { NgxGlobalDefaultValues } from '../../global-configuration-values';\nimport { EntityService } from '../../services/entity.service';\nimport { EntityTab, EntityUtilities } from '../../utilities/entity.utilities';\nimport { ValidationError, ValidationUtilities } from '../../utilities/validation.utilities';\nimport { ConfirmDialogData } from '../confirm-dialog/confirm-dialog-data';\nimport { ConfirmDialogDataBuilder, ConfirmDialogDataInternal } from '../confirm-dialog/confirm-dialog-data.builder';\nimport { NgxMatEntityConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component';\nimport { NgxMatEntityInputModule } from '../input/input.module';\nimport { CreateEntityData } from '../table/create-dialog/create-entity-dialog-data';\nimport { CreateData } from '../table/table-data';\nimport { TooltipComponent } from '../tooltip/tooltip.component';\nimport { PageCreateDataBuilder, PageCreateDataInternal } from './page-create-data.builder';\n\n/**\n * The data that needs to be provided for a route to be able to create a entity.\n */\n// eslint-disable-next-line max-len\nexport type PageCreateData<EntityType extends BaseEntityType<EntityType>> = Omit<CreateEntityData<EntityType>, 'entity' | 'EntityServiceClass'> & {\n    /**\n     * Whether or not to display a loading spinner while the data for the page is loaded.\n     * @default true\n     */\n    displayLoadingSpinner?: boolean,\n    /**\n     * The data of the default create page.\n     */\n    createData?: CreateData & {\n        /**\n         * The data for the dialog when the user tries to leave the site with unsaved changes.\n         */\n        confirmUnsavedChangesDialogData?: ConfirmDialogData,\n        /**\n         * Whether or not leaving with unsaved changes should require a confirm dialog.\n         * @default true\n         */\n        unsavedChangesRequireConfirmDialog?: boolean\n    }\n};\n\n/**\n * The entity service that needs to be provided in the providers array of the create page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_CREATE_DATA_ENTITY_SERVICE: InjectionToken<EntityService<any>> = new InjectionToken<EntityService<any>>('NGX_CREATE_DATA_ENTITY_SERVICE');\n/**\n * The entity class that needs to be provided in the providers array of the create page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_CREATE_DATA_ENTITY: InjectionToken<EntityClassNewable<any>> = new InjectionToken<EntityClassNewable<any>>('NGX_CREATE_DATA_ENTITY');\n/**\n * The configuration that needs to be provided in the providers array of the create page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any, constCase/uppercase\nexport const NGX_CREATE_DATA: InjectionToken<PageCreateData<any>> = new InjectionToken<PageCreateData<any>>('NGX_CREATE_DATA');\n\n/**\n * A generic page that allows you to create a specific entity.\n * For this to work you need to provide some data for the route.\n */\n@Component({\n    selector: 'ngx-mat-entity-create-page',\n    templateUrl: './create-page.component.html',\n    styleUrls: ['./create-page.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatButtonModule,\n        MatTabsModule,\n        NgxMatEntityInputModule,\n        MatProgressSpinnerModule,\n        MatBadgeModule,\n        TooltipComponent\n    ]\n})\nexport class NgxMatEntityCreatePageComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    /**\n     * Contains HelperMethods around handling Entities and their property-metadata.\n     */\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The tabs to display.\n     */\n    entityTabs!: EntityTab<EntityType>[];\n\n    /**\n     * The entity to create.\n     */\n    entity!: EntityType;\n    /**\n     * The entity before any changes have been made.\n     * This is used to determine if leaving the page is possible without interruption.\n     */\n    entityPriorChanges!: EntityType;\n\n    /**\n     * Configuration data for this component.\n     */\n    data!: PageCreateDataInternal<EntityType>;\n\n    /**\n     * All validation errors of the entity.\n     */\n    validationErrors!: ValidationError[];\n    /**\n     * Whether or not the entity is valid.\n     */\n    isEntityValid: boolean = false;\n    /**\n     * Whether or not the entity is dirty.\n     */\n    isEntityDirty: boolean = false;\n    /**\n     * What to display inside the tooltip.\n     */\n    tooltipContent: string = '';\n\n    private inConfirmNavigation: boolean = false;\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    get hasUnsavedChanges(): boolean {\n        return this.isEntityDirty && this.data.createData.unsavedChangesRequireConfirmDialog;\n    }\n\n    constructor(\n        private readonly dialog: MatDialog,\n        private readonly location: Location,\n        private readonly injector: EnvironmentInjector,\n        @Inject(NGX_CREATE_DATA_ENTITY_SERVICE)\n        readonly entityService: EntityService<EntityType>,\n        @Inject(NGX_CREATE_DATA_ENTITY)\n        private readonly EntityClass: EntityClassNewable<EntityType>,\n        @Inject(NGX_CREATE_DATA)\n        private readonly inputData: PageCreateData<EntityType>,\n        private readonly http: HttpClient,\n        private readonly el: ElementRef,\n        private readonly renderer: Renderer2,\n        @Inject(NGX_INTERNAL_GLOBAL_DEFAULT_VALUES)\n        protected readonly globalConfig: NgxGlobalDefaultValues\n    ) { }\n\n    ngOnInit(): void {\n        this.data = new PageCreateDataBuilder(this.inputData, this.globalConfig).getResult();\n        if (this.data == null) {\n            this.confirmNavigateBack();\n            // eslint-disable-next-line max-len\n            throw new Error('No create data was provided for \"NGX_CREATE_DATA\". You need to provide a value in your routes providers array.');\n        }\n\n        this.entity = new this.EntityClass();\n        EntityUtilities.setDefaultValues(this.entity);\n        this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);\n        void this.checkIsEntityValid();\n\n        this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, true, false);\n        setTimeout(() => this.checkOffset(), 1);\n        // setTimeout(() => this.checkEntity(), 1);\n    }\n\n    /**\n     * Whether the page can be left without confirmation (of unsaved changes).\n     * @returns Whether or not the page can be left without confirmation.\n     */\n    @HostListener('window:beforeunload')\n    canDeactivate(): boolean {\n        return !this.hasUnsavedChanges || this.inConfirmNavigation;\n    }\n\n    /**\n     * Checks if the bottom row should be displayed as fixed.\n     */\n    @HostListener('window:scroll')\n    checkOffset(): void {\n        const scrollY: number = window.scrollY;\n        const bottomRow: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row');\n        const bottomRowContainer: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row-container');\n\n        if (bottomRow && bottomRowContainer) {\n            const bottomRowContainerOffset: number = bottomRowContainer.offsetTop;\n            const windowHeight: number = window.innerHeight;\n\n            if (scrollY + windowHeight >= bottomRowContainerOffset) {\n                this.renderer.removeClass(bottomRow, 'fixed');\n            }\n            else {\n                this.renderer.addClass(bottomRow, 'fixed');\n            }\n        }\n    }\n\n    /**\n     * Checks if the entity has become invalid or dirty.\n     */\n    async checkEntity(): Promise<void> {\n        await this.checkIsEntityValid();\n        this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);\n    }\n\n    private async checkIsEntityValid(): Promise<void> {\n        this.validationErrors = await ValidationUtilities.getEntityValidationErrors(this.entity, this.injector, 'create');\n        this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));\n        this.isEntityValid = this.validationErrors.length === 0;\n    }\n\n    /**\n     * Tries create the entity and navigate back afterwards.\n     * Also handles the confirmation if required.\n     */\n    create(): void {\n        if (!this.isEntityValid) {\n            return;\n        }\n        if (!this.data.createData.createRequiresConfirmDialog) {\n            this.confirmCreate();\n            return;\n        }\n        // eslint-disable-next-line max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.createData.confirmCreateDialogData)\n            .withDefault('text', this.globalConfig.confirmCreateText)\n            .withDefault('confirmButtonLabel', this.globalConfig.createLabel)\n            .withDefault('title', this.globalConfig.createLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmCreate();\n            }\n        });\n    }\n\n    private confirmCreate(): void {\n        void this.entityService.create(this.entity).then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to navigate back.\n     */\n    navigateBack(): void {\n        if (!this.hasUnsavedChanges) {\n            this.confirmNavigateBack();\n            return;\n        }\n\n        this.openConfirmNavigationDialog().subscribe(res => {\n            if (res) {\n                this.confirmNavigateBack();\n            }\n        });\n    }\n\n    /**\n     * Opens the confirm dialog for navigating with unsaved changes.\n     * This is exposed because the UnsavedChangesGuard needs to access this.\n     * @returns The first observable result of the confirm dialog.\n     */\n    openConfirmNavigationDialog(): Observable<boolean> {\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: this.data.createData.confirmUnsavedChangesDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        return dialogRef.afterClosed().pipe(first(), map(p => (p ?? false)));\n    }\n\n    private confirmNavigateBack(): void {\n        this.inConfirmNavigation = true;\n        this.location.back();\n    }\n\n    /**\n     * Checks if the input with the given key is readonly.\n     * @param key - The key for the input to check.\n     * @returns Whether or not the input for the key is read only.\n     */\n    isReadOnly(key: keyof EntityType): boolean {\n        return runInInjectionContext(this.injector, () => {\n            const metadata: PropertyDecoratorConfigInternal<unknown> | undefined = EntityUtilities.getPropertyMetadata(this.entity, key);\n            if (!metadata) {\n                throw new Error(`No metadata was found for the key \"${String(key)}\"`);\n            }\n            return metadata.isReadOnly(this.entity);\n        });\n    }\n}","<!-- eslint-disable angular/no-call-expression -->\n<!-- eslint-disable angular/cyclomatic-complexity -->\n<div *ngIf=\"!entityTabs && data.displayLoadingSpinner\" class=\"container\">\n    <br>\n    <mat-spinner></mat-spinner>\n    <br>\n</div>\n\n<div *ngIf=\"entityTabs\" class=\"container\">\n    <br>\n\n    <!------------>\n    <!-- Header -->\n    <!------------>\n    <div class=\"header\">\n        <div class=\"cancel-container\">\n            <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n                <i class=\"fas fa-chevron-left\"></i>\n                {{data.createData.cancelButtonLabel}}\n                <i *ngIf=\"hasUnsavedChanges\" class=\"fas fa-warning\"></i>\n            </button>\n        </div>\n    </div>\n\n    <h1>{{data.createData.title}}</h1>\n\n    <!----------->\n    <!-- Input -->\n    <!----------->\n    <form>\n        <mat-tab-group *ngIf=\"entityTabs.length > 1\" preserveContent (selectedTabChange)=\"checkOffset()\">\n            <mat-tab *ngFor=\"let tab of entityTabs\" [label]=\"tab.tabName\">\n                <div *ngFor=\"let row of tab.rows\" class=\"row\">\n                    <ngx-mat-entity-input\n                        *ngFor=\"let key of row.keys\"\n                        [entity]=\"entity\"\n                        [propertyKey]=\"key\"\n                        [hideOmitForEdit]=\"true\"\n                        [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                        [isReadOnly]=\"isReadOnly(key)\"\n                        (inputChangeEvent)=\"checkEntity()\"\n                    >\n                    </ngx-mat-entity-input>\n                </div>\n            </mat-tab>\n        </mat-tab-group>\n    \n        <div *ngIf=\"entityTabs.length <= 1\">\n            <span *ngIf=\"!entityTabs.length\" class=\"no-entity-tabs\">\n                ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n            </span>\n            <div *ngFor=\"let row of entityTabs[0]?.rows\" class=\"row\">\n                <ngx-mat-entity-input\n                    *ngFor=\"let key of row.keys\"\n                    [entity]=\"entity\"\n                    [propertyKey]=\"key\"\n                    [hideOmitForEdit]=\"true\"\n                    [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                    [isReadOnly]=\"isReadOnly(key)\"\n                    (inputChangeEvent)=\"checkEntity()\"\n                >\n                </ngx-mat-entity-input>\n            </div>\n        </div>\n\n        <div class=\"bottom-row-container\">\n            <div class=\"bottom-row container\" style=\"margin-top: 10px;\">\n                <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n                    [disabled]=\"!isEntityValid || !isEntityDirty\"\n                    [matBadge]=\"validationErrors.length\"\n                    [matBadgeHidden]=\"!validationErrors.length\"\n                    (click)=\"create()\"\n                >\n                    {{data.createData.createButtonLabel}}\n                </button>\n                <ngx-mat-entity-tooltip *ngIf=\"validationErrors.length\" [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n            </div>\n        </div>\n    </form>\n\n    <br>\n</div>"]}
|
|
@@ -133,6 +133,9 @@ export class NgxMatEntityEditPageComponent {
|
|
|
133
133
|
isReadOnly(key) {
|
|
134
134
|
return runInInjectionContext(this.injector, () => {
|
|
135
135
|
const metadata = EntityUtilities.getPropertyMetadata(this.entity, key);
|
|
136
|
+
if (!metadata) {
|
|
137
|
+
throw new Error(`No metadata was found for the key "${String(key)}"`);
|
|
138
|
+
}
|
|
136
139
|
return this.isEntityReadOnly || metadata.isReadOnly(this.entity);
|
|
137
140
|
});
|
|
138
141
|
}
|
|
@@ -156,7 +159,7 @@ export class NgxMatEntityEditPageComponent {
|
|
|
156
159
|
});
|
|
157
160
|
this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, false, true);
|
|
158
161
|
setTimeout(() => this.checkOffset(), 1);
|
|
159
|
-
setTimeout(() => this.checkIsEntityValid(), 1);
|
|
162
|
+
setTimeout(() => void this.checkIsEntityValid(), 1);
|
|
160
163
|
}
|
|
161
164
|
/**
|
|
162
165
|
* Checks if the bottom row should be displayed as fixed.
|
|
@@ -187,11 +190,11 @@ export class NgxMatEntityEditPageComponent {
|
|
|
187
190
|
* Checks if the entity has become invalid or dirty.
|
|
188
191
|
*/
|
|
189
192
|
async checkEntity() {
|
|
190
|
-
this.checkIsEntityValid();
|
|
193
|
+
await this.checkIsEntityValid();
|
|
191
194
|
this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);
|
|
192
195
|
}
|
|
193
|
-
checkIsEntityValid() {
|
|
194
|
-
this.validationErrors = ValidationUtilities.getEntityValidationErrors(this.entity, 'update');
|
|
196
|
+
async checkIsEntityValid() {
|
|
197
|
+
this.validationErrors = await ValidationUtilities.getEntityValidationErrors(this.entity, this.injector, 'update');
|
|
195
198
|
this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));
|
|
196
199
|
this.isEntityValid = this.validationErrors.length === 0;
|
|
197
200
|
}
|
|
@@ -363,4 +366,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImpor
|
|
|
363
366
|
type: HostListener,
|
|
364
367
|
args: ['window:beforeunload']
|
|
365
368
|
}] } });
|
|
366
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edit-page.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.ts","../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Location, NgFor, NgIf } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EnvironmentInjector, HostListener, Inject, InjectionToken, OnInit, Renderer2, runInInjectionContext } from '@angular/core';\nimport { MatBadgeModule } from '@angular/material/badge';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { MatTabsModule } from '@angular/material/tabs';\nimport { ActivatedRoute } from '@angular/router';\nimport { Observable, first, map } from 'rxjs';\nimport { BaseEntityType, EntityClassNewable } from '../../classes/entity.model';\nimport { PropertyDecoratorConfigInternal } from '../../decorators/base/property-decorator-internal.data';\nimport { NGX_INTERNAL_GLOBAL_DEFAULT_VALUES } from '../../default-global-configuration-values';\nimport { LodashUtilities } from '../../encapsulation/lodash.utilities';\nimport { getValidationErrorsTooltipContent } from '../../functions/get-validation-errors-tooltip-content.function.ts';\nimport { NgxGlobalDefaultValues } from '../../global-configuration-values';\nimport { EntityService } from '../../services/entity.service';\nimport { EntityTab, EntityUtilities } from '../../utilities/entity.utilities';\nimport { ValidationError, ValidationUtilities } from '../../utilities/validation.utilities';\nimport { ConfirmDialogData } from '../confirm-dialog/confirm-dialog-data';\nimport { ConfirmDialogDataBuilder, ConfirmDialogDataInternal } from '../confirm-dialog/confirm-dialog-data.builder';\nimport { NgxMatEntityConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component';\nimport { NgxMatEntityInputModule } from '../input/input.module';\nimport { EditActionInternal } from '../table/edit-dialog/edit-data.builder';\nimport { EditEntityData } from '../table/edit-dialog/edit-entity-data';\nimport { EditData } from '../table/table-data';\nimport { TooltipComponent } from '../tooltip/tooltip.component';\nimport { PageEditDataBuilder, PageEditDataInternal } from './page-edit-data.builder';\n\n/**\n * The data that needs to be provided for a route to be able to edit a entity.\n */\n// eslint-disable-next-line max-len\nexport type PageEditData<EntityType extends BaseEntityType<EntityType>> = Omit<EditEntityData<EntityType>, 'entity' | 'EntityServiceClass'> & {\n    /**\n     * Whether or not to display a loading spinner while the entity for the page is loaded.\n     * @default true\n     */\n    displayLoadingSpinner?: boolean,\n    /**\n     * The data of the default edit page.\n     */\n    editData?: EditData<EntityType> & {\n        /**\n         * The data for the dialog when the user tries to leave the site with unsaved changes.\n         */\n        confirmUnsavedChangesDialogData?: ConfirmDialogData,\n        /**\n         * Whether or not leaving with unsaved changes should require a confirm dialog.\n         * @default true\n         */\n        unsavedChangesRequireConfirmDialog?: boolean\n    }\n};\n\n/**\n * The entity service that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_EDIT_DATA_ENTITY_SERVICE: InjectionToken<EntityService<any>> = new InjectionToken<EntityService<any>>('NGX_EDIT_DATA_ENTITY_SERVICE');\n/**\n * The entity class that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_EDIT_DATA_ENTITY: InjectionToken<EntityClassNewable<any>> = new InjectionToken<EntityClassNewable<any>>('NGX_EDIT_DATA_ENTITY');\n/**\n * The configuration that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any, constCase/uppercase\nexport const NGX_EDIT_DATA: InjectionToken<PageEditData<any>> = new InjectionToken<PageEditData<any>>('NGX_EDIT_DATA');\n\n/**\n * A generic page that allows you to edit a specific entity.\n * For this to work you need to provide some data for the route.\n */\n@Component({\n    selector: 'ngx-mat-entity-edit-page',\n    templateUrl: './edit-page.component.html',\n    styleUrls: ['./edit-page.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatButtonModule,\n        MatTabsModule,\n        NgxMatEntityInputModule,\n        MatProgressSpinnerModule,\n        MatMenuModule,\n        MatBadgeModule,\n        TooltipComponent\n    ]\n})\nexport class NgxMatEntityEditPageComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    /**\n     * Contains HelperMethods around handling Entities and their property-metadata.\n     */\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The tabs to display.\n     */\n    entityTabs!: EntityTab<EntityType>[];\n\n    /**\n     * The entity that is being edited.\n     */\n    entity!: EntityType;\n    /**\n     * The entity before any changes have been made.\n     */\n    entityPriorChanges!: EntityType;\n\n    /**\n     * Configuration data for the component.\n     */\n    data!: PageEditDataInternal<EntityType>;\n\n    /**\n     * All validation errors of the entity.\n     */\n    validationErrors: ValidationError[] = [];\n    /**\n     * Whether or not the entity is valid.\n     */\n    isEntityValid: boolean = true;\n    /**\n     * Whether or not the entity is dirty.\n     */\n    isEntityDirty: boolean = false;\n    /**\n     * What to display inside the tooltip.\n     */\n    tooltipContent: string = '';\n\n    /**\n     * Whether or not the entity is readonly.\n     */\n    isEntityReadOnly!: boolean;\n    /**\n     * Whether or not the current user is allowed to delete the entity.\n     */\n    allowDelete!: boolean;\n\n    private inConfirmNavigation: boolean = false;\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    get hasUnsavedChanges(): boolean {\n        return this.isEntityDirty && this.data.editData.unsavedChangesRequireConfirmDialog;\n    }\n\n    constructor(\n        private readonly dialog: MatDialog,\n        private readonly location: Location,\n        private readonly route: ActivatedRoute,\n        private readonly injector: EnvironmentInjector,\n        @Inject(NGX_EDIT_DATA_ENTITY_SERVICE)\n        readonly entityService: EntityService<EntityType>,\n        @Inject(NGX_EDIT_DATA_ENTITY)\n        private readonly EntityClass: EntityClassNewable<EntityType>,\n        @Inject(NGX_EDIT_DATA)\n        private readonly inputData: PageEditData<EntityType>,\n        private readonly http: HttpClient,\n        private readonly el: ElementRef,\n        private readonly renderer: Renderer2,\n        @Inject(NGX_INTERNAL_GLOBAL_DEFAULT_VALUES)\n        protected readonly globalConfig: NgxGlobalDefaultValues\n    ) { }\n\n    /**\n     * Checks if the input with the given key is readonly.\n     * @param key - The key for the input to check.\n     * @returns Whether or not the input for the key is read only.\n     */\n    isReadOnly(key: keyof EntityType): boolean {\n        return runInInjectionContext(this.injector, () => {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = EntityUtilities.getPropertyMetadata(this.entity, key);\n            return this.isEntityReadOnly || metadata.isReadOnly(this.entity);\n        });\n    }\n\n    async ngOnInit(): Promise<void> {\n        this.data = new PageEditDataBuilder(this.inputData, this.globalConfig).getResult();\n        if (this.data == null) {\n            this.confirmNavigateBack();\n            throw new Error('No edit data was provided for \"NGX_EDIT_DATA\". You need to provide a value in your routes providers array.');\n        }\n\n        const id: EntityType[keyof EntityType] = this.route.snapshot.paramMap.get('id') as EntityType[keyof EntityType];\n        const foundEntity: EntityType | undefined = await this.entityService.findById(id);\n\n        if (foundEntity == null) {\n            this.confirmNavigateBack();\n            throw new Error(`Could not find entity with id ${id}`);\n        }\n\n        this.entity = new this.EntityClass(foundEntity);\n        this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);\n\n        runInInjectionContext(this.injector, () => {\n            this.isEntityReadOnly = !this.data.allowUpdate(this.entityPriorChanges);\n            this.allowDelete = this.data.allowDelete(this.entityPriorChanges);\n        });\n        this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, false, true);\n        setTimeout(() => this.checkOffset(), 1);\n        setTimeout(() => this.checkIsEntityValid(), 1);\n    }\n\n    /**\n     * Checks if the bottom row should be displayed as fixed.\n     */\n    @HostListener('window:scroll')\n    checkOffset(): void {\n        const scrollY: number = window.scrollY;\n        const bottomRow: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row');\n        const bottomRowContainer: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row-container');\n\n        if (bottomRow && bottomRowContainer) {\n            const bottomRowContainerOffset: number = bottomRowContainer.offsetTop;\n            const windowHeight: number = window.innerHeight;\n\n            if (scrollY + windowHeight >= bottomRowContainerOffset) {\n                this.renderer.removeClass(bottomRow, 'fixed');\n            }\n            else {\n                this.renderer.addClass(bottomRow, 'fixed');\n            }\n        }\n    }\n\n    /**\n     * Whether the page can be left without confirmation (of unsaved changes).\n     * @returns Whether or not the page can be left without confirmation.\n     */\n    @HostListener('window:beforeunload')\n    canDeactivate(): boolean {\n        return !this.hasUnsavedChanges || this.inConfirmNavigation;\n    }\n\n    /**\n     * Checks if the entity has become invalid or dirty.\n     */\n    async checkEntity(): Promise<void> {\n        this.checkIsEntityValid();\n        this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);\n    }\n\n    private checkIsEntityValid(): void {\n        this.validationErrors = ValidationUtilities.getEntityValidationErrors(this.entity, 'update');\n        this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));\n        this.isEntityValid = this.validationErrors.length === 0;\n    }\n\n    /**\n     * Tries to save the changes and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    edit(): void {\n        if (this.isEntityReadOnly || !this.isEntityValid || !this.isEntityDirty) {\n            return;\n        }\n        if (!this.data.editData.editRequiresConfirmDialog) {\n            this.confirmEdit();\n            return;\n        }\n        // eslint-disable-next-line max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmEditDialogData)\n            .withDefault('text', this.globalConfig.confirmSaveText)\n            .withDefault('confirmButtonLabel', this.globalConfig.saveLabel)\n            .withDefault('title', this.globalConfig.editLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmEdit();\n            }\n        });\n    }\n\n    private confirmEdit(): void {\n        void this.entityService.update(this.entity, this.entityPriorChanges).then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to delete the entity and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    delete(): void {\n        if (!this.data.editData.deleteRequiresConfirmDialog) {\n            this.confirmDelete();\n            return;\n        }\n        // eslint-disable-next-line max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmDeleteDialogData)\n            .withDefault('text', this.globalConfig.confirmDeleteText)\n            .withDefault('type', 'delete')\n            .withDefault('confirmButtonLabel', this.globalConfig.deleteLabel)\n            .withDefault('title', this.globalConfig.deleteLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmDelete();\n            }\n        });\n    }\n\n    private confirmDelete(): void {\n        void this.entityService\n            .delete(this.entityPriorChanges)\n            .then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to navigate back.\n     */\n    navigateBack(): void {\n        if (!this.hasUnsavedChanges) {\n            this.confirmNavigateBack();\n            return;\n        }\n\n        this.openConfirmNavigationDialog().subscribe(res => {\n            if (res) {\n                this.confirmNavigateBack();\n            }\n        });\n    }\n\n    /**\n     * Opens the confirm dialog for navigating with unsaved changes.\n     * This is exposed because the UnsavedChangesGuard needs to access this.\n     * @returns The first observable result of the confirm dialog.\n     */\n    openConfirmNavigationDialog(): Observable<boolean> {\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: this.data.editData.confirmUnsavedChangesDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        return dialogRef.afterClosed().pipe(first(), map(p => (p ?? false)));\n    }\n\n    private confirmNavigateBack(): void {\n        this.inConfirmNavigation = true;\n        EntityUtilities.resetChangesOnEntity(this.entity, this.entityPriorChanges);\n        this.location.back();\n    }\n\n    /**\n     * Runs the edit action on the entity.\n     * @param action - The action to run.\n     */\n    runEditAction(action: EditActionInternal<EntityType>): void {\n        const requireConfirmDialog: boolean = runInInjectionContext(this.injector, () => {\n            return action.requireConfirmDialog(this.entityPriorChanges);\n        });\n\n        if (!requireConfirmDialog) {\n            this.confirmRunEditAction(action);\n            return;\n        }\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: action.confirmDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmRunEditAction(action);\n            }\n        });\n    }\n\n    private confirmRunEditAction(action: EditActionInternal<EntityType>): void {\n        void runInInjectionContext(this.injector, async () => {\n            await action.action(this.entity, this.entityPriorChanges);\n            await this.checkEntity();\n        });\n    }\n\n    /**\n     * Checks if an EditAction is disabled (e.g. Because the current entry doesn't fullfil the requirements).\n     * @param action - The EditAction to check.\n     * @returns Whether or not the Action can be used.\n     */\n    editActionDisabled(action: EditActionInternal<EntityType>): boolean {\n        return runInInjectionContext(this.injector, () => !action.enabled(this.entityPriorChanges));\n    }\n}"," \n<!-- eslint-disable angular/no-call-expression -->\n<!-- eslint-disable angular/cyclomatic-complexity -->\n<div *ngIf=\"!entityTabs && data.displayLoadingSpinner\" class=\"container\">\n    <br>\n    <mat-spinner></mat-spinner>\n    <br>\n</div>\n\n<div *ngIf=\"entityTabs\" class=\"container\">\n    <br>\n\n    <!------------>\n    <!-- Header -->\n    <!------------>\n    <div class=\"header\">\n        <div class=\"cancel-container\">\n            <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n                <i class=\"fas fa-chevron-left\"></i>\n                {{data.editData.cancelButtonLabel}}\n                <i *ngIf=\"hasUnsavedChanges\" class=\"fas fa-warning\"></i>\n            </button>\n        </div>\n        <div class=\"actions-container\">\n            <button *ngIf=\"data.editData.actions.length\" type=\"button\" mat-raised-button [matMenuTriggerFor]=\"menu\">\n                {{data.editData.actionsLabel}}\n            </button>\n            <mat-menu #menu=\"matMenu\">\n                <button *ngFor=\"let action of data.editData.actions\" type=\"button\" mat-menu-item [disabled]=\"editActionDisabled(action)\" (click)=\"runEditAction(action)\">\n                    {{action.displayName}}\n                </button>\n            </mat-menu>\n        \n            <button *ngIf=\"allowDelete\" type=\"button\" mat-raised-button color=\"warn\" class=\"delete-button\" tabindex=\"-1\" (click)=\"delete()\">\n                {{data.editData.deleteButtonLabel}}\n            </button>\n        </div>\n    </div>\n\n    <h1>{{data.editData.title(entityPriorChanges)}}</h1>\n\n    <!----------->\n    <!-- Input -->\n    <!----------->\n    <form>\n        <mat-tab-group *ngIf=\"entityTabs.length > 1\" preserveContent (selectedTabChange)=\"checkOffset()\">\n            <mat-tab *ngFor=\"let tab of entityTabs\" [label]=\"tab.tabName\">\n                <div *ngFor=\"let row of tab.rows\" class=\"row\">\n                    <ngx-mat-entity-input\n                        *ngFor=\"let key of row.keys\"\n                        [entity]=\"entity\"\n                        [propertyKey]=\"key\"\n                        [hideOmitForEdit]=\"true\"\n                        [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                        [isReadOnly]=\"isReadOnly(key)\"\n                        (inputChangeEvent)=\"checkEntity()\"\n                    >\n                    </ngx-mat-entity-input>\n                </div>\n            </mat-tab>\n        </mat-tab-group>\n    \n        <div *ngIf=\"entityTabs.length <= 1\">\n            <span *ngIf=\"!entityTabs.length\" class=\"no-entity-tabs\">\n                ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n            </span>\n            <div *ngFor=\"let row of entityTabs[0]?.rows\" class=\"row\">\n                <ngx-mat-entity-input\n                    *ngFor=\"let key of row.keys\"\n                    [entity]=\"entity\"\n                    [propertyKey]=\"key\"\n                    [hideOmitForEdit]=\"true\"\n                    [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                    [isReadOnly]=\"isReadOnly(key)\"\n                    (inputChangeEvent)=\"checkEntity()\"\n                >\n                </ngx-mat-entity-input>\n            </div>\n        </div>\n\n        <div class=\"bottom-row-container\">\n            <div class=\"bottom-row container\" style=\"margin-top: 10px;\">\n                <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n                    [disabled]=\"isEntityReadOnly || !isEntityValid || !isEntityDirty\"\n                    [matBadge]=\"validationErrors.length\"\n                    [matBadgeHidden]=\"!validationErrors.length\"\n                    (click)=\"edit()\"\n                >\n                    {{data.editData.confirmButtonLabel}}\n                </button>\n                <ngx-mat-entity-tooltip *ngIf=\"validationErrors.length\" [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n            </div>\n        </div>\n    </form>\n\n    <br>\n</div>"]}
|
|
369
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edit-page.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.ts","../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Location, NgFor, NgIf } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EnvironmentInjector, HostListener, Inject, InjectionToken, OnInit, Renderer2, runInInjectionContext } from '@angular/core';\nimport { MatBadgeModule } from '@angular/material/badge';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { MatTabsModule } from '@angular/material/tabs';\nimport { ActivatedRoute } from '@angular/router';\nimport { Observable, first, map } from 'rxjs';\nimport { BaseEntityType, EntityClassNewable } from '../../classes/entity.model';\nimport { PropertyDecoratorConfigInternal } from '../../decorators/base/property-decorator-internal.data';\nimport { NGX_INTERNAL_GLOBAL_DEFAULT_VALUES } from '../../default-global-configuration-values';\nimport { LodashUtilities } from '../../encapsulation/lodash.utilities';\nimport { getValidationErrorsTooltipContent } from '../../functions/get-validation-errors-tooltip-content.function.ts';\nimport { NgxGlobalDefaultValues } from '../../global-configuration-values';\nimport { EntityService } from '../../services/entity.service';\nimport { EntityTab, EntityUtilities } from '../../utilities/entity.utilities';\nimport { ValidationError, ValidationUtilities } from '../../utilities/validation.utilities';\nimport { ConfirmDialogData } from '../confirm-dialog/confirm-dialog-data';\nimport { ConfirmDialogDataBuilder, ConfirmDialogDataInternal } from '../confirm-dialog/confirm-dialog-data.builder';\nimport { NgxMatEntityConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component';\nimport { NgxMatEntityInputModule } from '../input/input.module';\nimport { EditActionInternal } from '../table/edit-dialog/edit-data.builder';\nimport { EditEntityData } from '../table/edit-dialog/edit-entity-data';\nimport { EditData } from '../table/table-data';\nimport { TooltipComponent } from '../tooltip/tooltip.component';\nimport { PageEditDataBuilder, PageEditDataInternal } from './page-edit-data.builder';\n\n/**\n * The data that needs to be provided for a route to be able to edit a entity.\n */\n// eslint-disable-next-line max-len\nexport type PageEditData<EntityType extends BaseEntityType<EntityType>> = Omit<EditEntityData<EntityType>, 'entity' | 'EntityServiceClass'> & {\n    /**\n     * Whether or not to display a loading spinner while the entity for the page is loaded.\n     * @default true\n     */\n    displayLoadingSpinner?: boolean,\n    /**\n     * The data of the default edit page.\n     */\n    editData?: EditData<EntityType> & {\n        /**\n         * The data for the dialog when the user tries to leave the site with unsaved changes.\n         */\n        confirmUnsavedChangesDialogData?: ConfirmDialogData,\n        /**\n         * Whether or not leaving with unsaved changes should require a confirm dialog.\n         * @default true\n         */\n        unsavedChangesRequireConfirmDialog?: boolean\n    }\n};\n\n/**\n * The entity service that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_EDIT_DATA_ENTITY_SERVICE: InjectionToken<EntityService<any>> = new InjectionToken<EntityService<any>>('NGX_EDIT_DATA_ENTITY_SERVICE');\n/**\n * The entity class that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line max-len, typescript/no-explicit-any, constCase/uppercase\nexport const NGX_EDIT_DATA_ENTITY: InjectionToken<EntityClassNewable<any>> = new InjectionToken<EntityClassNewable<any>>('NGX_EDIT_DATA_ENTITY');\n/**\n * The configuration that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any, constCase/uppercase\nexport const NGX_EDIT_DATA: InjectionToken<PageEditData<any>> = new InjectionToken<PageEditData<any>>('NGX_EDIT_DATA');\n\n/**\n * A generic page that allows you to edit a specific entity.\n * For this to work you need to provide some data for the route.\n */\n@Component({\n    selector: 'ngx-mat-entity-edit-page',\n    templateUrl: './edit-page.component.html',\n    styleUrls: ['./edit-page.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatButtonModule,\n        MatTabsModule,\n        NgxMatEntityInputModule,\n        MatProgressSpinnerModule,\n        MatMenuModule,\n        MatBadgeModule,\n        TooltipComponent\n    ]\n})\nexport class NgxMatEntityEditPageComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    /**\n     * Contains HelperMethods around handling Entities and their property-metadata.\n     */\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The tabs to display.\n     */\n    entityTabs!: EntityTab<EntityType>[];\n\n    /**\n     * The entity that is being edited.\n     */\n    entity!: EntityType;\n    /**\n     * The entity before any changes have been made.\n     */\n    entityPriorChanges!: EntityType;\n\n    /**\n     * Configuration data for the component.\n     */\n    data!: PageEditDataInternal<EntityType>;\n\n    /**\n     * All validation errors of the entity.\n     */\n    validationErrors: ValidationError[] = [];\n    /**\n     * Whether or not the entity is valid.\n     */\n    isEntityValid: boolean = true;\n    /**\n     * Whether or not the entity is dirty.\n     */\n    isEntityDirty: boolean = false;\n    /**\n     * What to display inside the tooltip.\n     */\n    tooltipContent: string = '';\n\n    /**\n     * Whether or not the entity is readonly.\n     */\n    isEntityReadOnly!: boolean;\n    /**\n     * Whether or not the current user is allowed to delete the entity.\n     */\n    allowDelete!: boolean;\n\n    private inConfirmNavigation: boolean = false;\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    get hasUnsavedChanges(): boolean {\n        return this.isEntityDirty && this.data.editData.unsavedChangesRequireConfirmDialog;\n    }\n\n    constructor(\n        private readonly dialog: MatDialog,\n        private readonly location: Location,\n        private readonly route: ActivatedRoute,\n        private readonly injector: EnvironmentInjector,\n        @Inject(NGX_EDIT_DATA_ENTITY_SERVICE)\n        readonly entityService: EntityService<EntityType>,\n        @Inject(NGX_EDIT_DATA_ENTITY)\n        private readonly EntityClass: EntityClassNewable<EntityType>,\n        @Inject(NGX_EDIT_DATA)\n        private readonly inputData: PageEditData<EntityType>,\n        private readonly http: HttpClient,\n        private readonly el: ElementRef,\n        private readonly renderer: Renderer2,\n        @Inject(NGX_INTERNAL_GLOBAL_DEFAULT_VALUES)\n        protected readonly globalConfig: NgxGlobalDefaultValues\n    ) { }\n\n    /**\n     * Checks if the input with the given key is readonly.\n     * @param key - The key for the input to check.\n     * @returns Whether or not the input for the key is read only.\n     */\n    isReadOnly(key: keyof EntityType): boolean {\n        return runInInjectionContext(this.injector, () => {\n            const metadata: PropertyDecoratorConfigInternal<unknown> | undefined = EntityUtilities.getPropertyMetadata(this.entity, key);\n            if (!metadata) {\n                throw new Error(`No metadata was found for the key \"${String(key)}\"`);\n            }\n            return this.isEntityReadOnly || metadata.isReadOnly(this.entity);\n        });\n    }\n\n    async ngOnInit(): Promise<void> {\n        this.data = new PageEditDataBuilder(this.inputData, this.globalConfig).getResult();\n        if (this.data == null) {\n            this.confirmNavigateBack();\n            throw new Error('No edit data was provided for \"NGX_EDIT_DATA\". You need to provide a value in your routes providers array.');\n        }\n\n        const id: EntityType[keyof EntityType] = this.route.snapshot.paramMap.get('id') as EntityType[keyof EntityType];\n        const foundEntity: EntityType | undefined = await this.entityService.findById(id);\n\n        if (foundEntity == null) {\n            this.confirmNavigateBack();\n            throw new Error(`Could not find entity with id ${id}`);\n        }\n\n        this.entity = new this.EntityClass(foundEntity);\n        this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);\n\n        runInInjectionContext(this.injector, () => {\n            this.isEntityReadOnly = !this.data.allowUpdate(this.entityPriorChanges);\n            this.allowDelete = this.data.allowDelete(this.entityPriorChanges);\n        });\n        this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, false, true);\n        setTimeout(() => this.checkOffset(), 1);\n        setTimeout(() => void this.checkIsEntityValid(), 1);\n    }\n\n    /**\n     * Checks if the bottom row should be displayed as fixed.\n     */\n    @HostListener('window:scroll')\n    checkOffset(): void {\n        const scrollY: number = window.scrollY;\n        const bottomRow: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row');\n        const bottomRowContainer: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row-container');\n\n        if (bottomRow && bottomRowContainer) {\n            const bottomRowContainerOffset: number = bottomRowContainer.offsetTop;\n            const windowHeight: number = window.innerHeight;\n\n            if (scrollY + windowHeight >= bottomRowContainerOffset) {\n                this.renderer.removeClass(bottomRow, 'fixed');\n            }\n            else {\n                this.renderer.addClass(bottomRow, 'fixed');\n            }\n        }\n    }\n\n    /**\n     * Whether the page can be left without confirmation (of unsaved changes).\n     * @returns Whether or not the page can be left without confirmation.\n     */\n    @HostListener('window:beforeunload')\n    canDeactivate(): boolean {\n        return !this.hasUnsavedChanges || this.inConfirmNavigation;\n    }\n\n    /**\n     * Checks if the entity has become invalid or dirty.\n     */\n    async checkEntity(): Promise<void> {\n        await this.checkIsEntityValid();\n        this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);\n    }\n\n    private async checkIsEntityValid(): Promise<void> {\n        this.validationErrors = await ValidationUtilities.getEntityValidationErrors(this.entity, this.injector, 'update');\n        this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));\n        this.isEntityValid = this.validationErrors.length === 0;\n    }\n\n    /**\n     * Tries to save the changes and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    edit(): void {\n        if (this.isEntityReadOnly || !this.isEntityValid || !this.isEntityDirty) {\n            return;\n        }\n        if (!this.data.editData.editRequiresConfirmDialog) {\n            this.confirmEdit();\n            return;\n        }\n        // eslint-disable-next-line max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmEditDialogData)\n            .withDefault('text', this.globalConfig.confirmSaveText)\n            .withDefault('confirmButtonLabel', this.globalConfig.saveLabel)\n            .withDefault('title', this.globalConfig.editLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmEdit();\n            }\n        });\n    }\n\n    private confirmEdit(): void {\n        void this.entityService.update(this.entity, this.entityPriorChanges).then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to delete the entity and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    delete(): void {\n        if (!this.data.editData.deleteRequiresConfirmDialog) {\n            this.confirmDelete();\n            return;\n        }\n        // eslint-disable-next-line max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmDeleteDialogData)\n            .withDefault('text', this.globalConfig.confirmDeleteText)\n            .withDefault('type', 'delete')\n            .withDefault('confirmButtonLabel', this.globalConfig.deleteLabel)\n            .withDefault('title', this.globalConfig.deleteLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmDelete();\n            }\n        });\n    }\n\n    private confirmDelete(): void {\n        void this.entityService\n            .delete(this.entityPriorChanges)\n            .then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to navigate back.\n     */\n    navigateBack(): void {\n        if (!this.hasUnsavedChanges) {\n            this.confirmNavigateBack();\n            return;\n        }\n\n        this.openConfirmNavigationDialog().subscribe(res => {\n            if (res) {\n                this.confirmNavigateBack();\n            }\n        });\n    }\n\n    /**\n     * Opens the confirm dialog for navigating with unsaved changes.\n     * This is exposed because the UnsavedChangesGuard needs to access this.\n     * @returns The first observable result of the confirm dialog.\n     */\n    openConfirmNavigationDialog(): Observable<boolean> {\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: this.data.editData.confirmUnsavedChangesDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        return dialogRef.afterClosed().pipe(first(), map(p => (p ?? false)));\n    }\n\n    private confirmNavigateBack(): void {\n        this.inConfirmNavigation = true;\n        EntityUtilities.resetChangesOnEntity(this.entity, this.entityPriorChanges);\n        this.location.back();\n    }\n\n    /**\n     * Runs the edit action on the entity.\n     * @param action - The action to run.\n     */\n    runEditAction(action: EditActionInternal<EntityType>): void {\n        const requireConfirmDialog: boolean = runInInjectionContext(this.injector, () => {\n            return action.requireConfirmDialog(this.entityPriorChanges);\n        });\n\n        if (!requireConfirmDialog) {\n            this.confirmRunEditAction(action);\n            return;\n        }\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: action.confirmDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmRunEditAction(action);\n            }\n        });\n    }\n\n    private confirmRunEditAction(action: EditActionInternal<EntityType>): void {\n        void runInInjectionContext(this.injector, async () => {\n            await action.action(this.entity, this.entityPriorChanges);\n            await this.checkEntity();\n        });\n    }\n\n    /**\n     * Checks if an EditAction is disabled (e.g. Because the current entry doesn't fullfil the requirements).\n     * @param action - The EditAction to check.\n     * @returns Whether or not the Action can be used.\n     */\n    editActionDisabled(action: EditActionInternal<EntityType>): boolean {\n        return runInInjectionContext(this.injector, () => !action.enabled(this.entityPriorChanges));\n    }\n}"," \n<!-- eslint-disable angular/no-call-expression -->\n<!-- eslint-disable angular/cyclomatic-complexity -->\n<div *ngIf=\"!entityTabs && data.displayLoadingSpinner\" class=\"container\">\n    <br>\n    <mat-spinner></mat-spinner>\n    <br>\n</div>\n\n<div *ngIf=\"entityTabs\" class=\"container\">\n    <br>\n\n    <!------------>\n    <!-- Header -->\n    <!------------>\n    <div class=\"header\">\n        <div class=\"cancel-container\">\n            <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n                <i class=\"fas fa-chevron-left\"></i>\n                {{data.editData.cancelButtonLabel}}\n                <i *ngIf=\"hasUnsavedChanges\" class=\"fas fa-warning\"></i>\n            </button>\n        </div>\n        <div class=\"actions-container\">\n            <button *ngIf=\"data.editData.actions.length\" type=\"button\" mat-raised-button [matMenuTriggerFor]=\"menu\">\n                {{data.editData.actionsLabel}}\n            </button>\n            <mat-menu #menu=\"matMenu\">\n                <button *ngFor=\"let action of data.editData.actions\" type=\"button\" mat-menu-item [disabled]=\"editActionDisabled(action)\" (click)=\"runEditAction(action)\">\n                    {{action.displayName}}\n                </button>\n            </mat-menu>\n        \n            <button *ngIf=\"allowDelete\" type=\"button\" mat-raised-button color=\"warn\" class=\"delete-button\" tabindex=\"-1\" (click)=\"delete()\">\n                {{data.editData.deleteButtonLabel}}\n            </button>\n        </div>\n    </div>\n\n    <h1>{{data.editData.title(entityPriorChanges)}}</h1>\n\n    <!----------->\n    <!-- Input -->\n    <!----------->\n    <form>\n        <mat-tab-group *ngIf=\"entityTabs.length > 1\" preserveContent (selectedTabChange)=\"checkOffset()\">\n            <mat-tab *ngFor=\"let tab of entityTabs\" [label]=\"tab.tabName\">\n                <div *ngFor=\"let row of tab.rows\" class=\"row\">\n                    <ngx-mat-entity-input\n                        *ngFor=\"let key of row.keys\"\n                        [entity]=\"entity\"\n                        [propertyKey]=\"key\"\n                        [hideOmitForEdit]=\"true\"\n                        [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                        [isReadOnly]=\"isReadOnly(key)\"\n                        (inputChangeEvent)=\"checkEntity()\"\n                    >\n                    </ngx-mat-entity-input>\n                </div>\n            </mat-tab>\n        </mat-tab-group>\n    \n        <div *ngIf=\"entityTabs.length <= 1\">\n            <span *ngIf=\"!entityTabs.length\" class=\"no-entity-tabs\">\n                ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n            </span>\n            <div *ngFor=\"let row of entityTabs[0]?.rows\" class=\"row\">\n                <ngx-mat-entity-input\n                    *ngFor=\"let key of row.keys\"\n                    [entity]=\"entity\"\n                    [propertyKey]=\"key\"\n                    [hideOmitForEdit]=\"true\"\n                    [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                    [isReadOnly]=\"isReadOnly(key)\"\n                    (inputChangeEvent)=\"checkEntity()\"\n                >\n                </ngx-mat-entity-input>\n            </div>\n        </div>\n\n        <div class=\"bottom-row-container\">\n            <div class=\"bottom-row container\" style=\"margin-top: 10px;\">\n                <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n                    [disabled]=\"isEntityReadOnly || !isEntityValid || !isEntityDirty\"\n                    [matBadge]=\"validationErrors.length\"\n                    [matBadgeHidden]=\"!validationErrors.length\"\n                    (click)=\"edit()\"\n                >\n                    {{data.editData.confirmButtonLabel}}\n                </button>\n                <ngx-mat-entity-tooltip *ngIf=\"validationErrors.length\" [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n            </div>\n        </div>\n    </form>\n\n    <br>\n</div>"]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/* eslint-disable jsdoc/require-jsdoc */
|
|
2
|
-
import { Component } from '@angular/core';
|
|
2
|
+
import { Component, EnvironmentInjector, runInInjectionContext } from '@angular/core';
|
|
3
3
|
import { LodashUtilities } from '../../../../encapsulation/lodash.utilities';
|
|
4
4
|
import { ArrayStringChipsInputComponent } from '../array-string-chips-input/array-string-chips-input.component';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
@@ -13,27 +13,40 @@ import * as i7 from "@angular/material/form-field";
|
|
|
13
13
|
import * as i8 from "@angular/material/input";
|
|
14
14
|
// eslint-disable-next-line angular/prefer-standalone-component
|
|
15
15
|
export class ArrayStringAutocompleteChipsComponent extends ArrayStringChipsInputComponent {
|
|
16
|
+
injector;
|
|
16
17
|
filteredAutocompleteStrings;
|
|
17
|
-
|
|
18
|
-
return this.metadata.autocompleteValues;
|
|
19
|
-
}
|
|
18
|
+
autocompleteStrings = [];
|
|
20
19
|
get autocompleteMetadata() {
|
|
21
20
|
return this.metadata;
|
|
22
21
|
}
|
|
23
|
-
|
|
22
|
+
constructor(injector) {
|
|
23
|
+
super();
|
|
24
|
+
this.injector = injector;
|
|
25
|
+
}
|
|
26
|
+
async ngOnInit() {
|
|
24
27
|
super.ngOnInit();
|
|
25
|
-
this.
|
|
28
|
+
await runInInjectionContext(this.injector, async () => {
|
|
29
|
+
this.autocompleteStrings = await this.autocompleteMetadata.autocompleteValues(this.entity);
|
|
30
|
+
this.filteredAutocompleteStrings = LodashUtilities.cloneDeep(this.autocompleteStrings);
|
|
31
|
+
});
|
|
26
32
|
}
|
|
27
33
|
/**
|
|
28
34
|
* Handles adding a string to the array when an autocomplete value has been selected.
|
|
29
35
|
* @param event - The autocomplete selected event.
|
|
30
36
|
* @param chipsInput - The element where the user typed the value.
|
|
37
|
+
* @param model - The model of the string array.
|
|
38
|
+
* @param chipsModel - The model of the single string input.
|
|
31
39
|
*/
|
|
32
|
-
selected(event, chipsInput) {
|
|
33
|
-
|
|
34
|
-
this.
|
|
40
|
+
selected(event, chipsInput, model, chipsModel) {
|
|
41
|
+
// validation is not needed as selected options are all valid.
|
|
42
|
+
this.propertyValue = this.propertyValue ?? [];
|
|
43
|
+
this.propertyValue.push(event.option.value);
|
|
35
44
|
chipsInput.value = '';
|
|
36
|
-
this.
|
|
45
|
+
this.chipsInput = '';
|
|
46
|
+
this.filterAutocompleteStrings(this.chipsInput);
|
|
47
|
+
chipsModel.control.updateValueAndValidity();
|
|
48
|
+
model.control.updateValueAndValidity();
|
|
49
|
+
this.setValidationErrors(model, chipsModel);
|
|
37
50
|
this.emitChange();
|
|
38
51
|
}
|
|
39
52
|
/**
|
|
@@ -44,11 +57,11 @@ export class ArrayStringAutocompleteChipsComponent extends ArrayStringChipsInput
|
|
|
44
57
|
const filterValue = input.toLowerCase();
|
|
45
58
|
this.filteredAutocompleteStrings = this.autocompleteStrings.filter(s => s.toLowerCase().includes(filterValue));
|
|
46
59
|
}
|
|
47
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: ArrayStringAutocompleteChipsComponent, deps:
|
|
48
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.0", type: ArrayStringAutocompleteChipsComponent, selector: "array-string-autocomplete-chips", usesInheritance: true, ngImport: i0, template: "<!-- eslint-disable angular/no-call-expression -->\n
|
|
60
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: ArrayStringAutocompleteChipsComponent, deps: [{ token: i0.EnvironmentInjector }], target: i0.ɵɵFactoryTarget.Component });
|
|
61
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.0", type: ArrayStringAutocompleteChipsComponent, selector: "array-string-autocomplete-chips", usesInheritance: true, ngImport: i0, template: "<!-- eslint-disable angular/no-call-expression -->\n\n<mat-form-field>\n <mat-label>{{metadata.displayName}}</mat-label>\n <mat-chip-grid #chipGrid\n #model=\"ngModel\" [name]=\"name\"\n [required]=\"metadata.required(entity)\"\n [disabled]=\"isReadOnly\"\n [(ngModel)]=\"propertyValue\"\n (ngModelChange)=\"setValidationErrors(model, chipsModel)\"\n >\n <mat-chip-row *ngFor=\"let value of propertyValue\" (removed)=\"removeStringChipArrayValue(value)\">\n {{value}}\n <button type=\"button\" matChipRemove>\n <i [class]=\"metadata.deleteIcon\"></i>\n </button>\n </mat-chip-row>\n <input #chipsModel=\"ngModel\"\n #chipsElement\n matInput\n [matChipInputFor]=\"chipGrid\"\n [matAutocomplete]=\"auto\"\n [matChipInputAddOnBlur]=\"true\"\n [name]=\"uuid2\"\n [minlength]=\"metadata.minLength ?? null\"\n [maxlength]=\"metadata.maxLength ?? null\"\n [pattern]=\"metadata.regex ?? '[\\\\s\\\\S]*'\"\n [includedIn]=\"autocompleteMetadata.restrictToOptions ? autocompleteStrings : undefined\"\n [(ngModel)]=\"chipsInput\"\n (keyup)=\"filterAutocompleteStrings(chipsInput)\"\n (matChipInputTokenEnd)=\"addStringChipArrayValue($event, !!chipsModel.errors)\"\n />\n </mat-chip-grid>\n <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"selected($event, chipsElement, model, chipsModel)\">\n <mat-option *ngFor=\"let value of filteredAutocompleteStrings\" [value]=\"value\">\n {{value}}\n </mat-option>\n </mat-autocomplete>\n <mat-error>{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>", styles: ["mat-form-field{width:100%}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i2.MinLengthValidator, selector: "[minlength][formControlName],[minlength][formControl],[minlength][ngModel]", inputs: ["minlength"] }, { kind: "directive", type: i2.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i2.PatternValidator, selector: "[pattern][formControlName],[pattern][formControl],[pattern][ngModel]", inputs: ["pattern"] }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i3.IncludedInValidatorDirective, selector: "[includedIn]", inputs: ["includedIn"] }, { kind: "component", type: i4.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple", "hideSingleSelectionIndicator"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i5.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i4.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { kind: "component", type: i6.MatChipGrid, selector: "mat-chip-grid", inputs: ["tabIndex", "disabled", "placeholder", "required", "value", "errorStateMatcher"], outputs: ["change", "valueChange"] }, { kind: "directive", type: i6.MatChipInput, selector: "input[matChipInputFor]", inputs: ["matChipInputFor", "matChipInputAddOnBlur", "matChipInputSeparatorKeyCodes", "placeholder", "id", "disabled"], outputs: ["matChipInputTokenEnd"], exportAs: ["matChipInput", "matChipInputFor"] }, { kind: "directive", type: i6.MatChipRemove, selector: "[matChipRemove]" }, { kind: "component", type: i6.MatChipRow, selector: "mat-chip-row, [mat-chip-row], mat-basic-chip-row, [mat-basic-chip-row]", inputs: ["color", "disabled", "disableRipple", "tabIndex", "editable"], outputs: ["edited"] }, { kind: "component", type: i7.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i7.MatLabel, selector: "mat-label" }, { kind: "directive", type: i7.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i8.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }] });
|
|
49
62
|
}
|
|
50
63
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: ArrayStringAutocompleteChipsComponent, decorators: [{
|
|
51
64
|
type: Component,
|
|
52
|
-
args: [{ selector: 'array-string-autocomplete-chips', template: "<!-- eslint-disable angular/no-call-expression -->\n
|
|
53
|
-
}] });
|
|
54
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"array-string-autocomplete-chips.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-material-entity/src/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.ts","../../../../../../../projects/ngx-material-entity/src/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.html"],"names":[],"mappings":"AAAA,wCAAwC;AACxC,OAAO,EAAE,SAAS,EAAU,MAAM,eAAe,CAAC;AAIlD,OAAO,EAAE,eAAe,EAAE,MAAM,4CAA4C,CAAC;AAC7E,OAAO,EAAE,8BAA8B,EAAE,MAAM,gEAAgE,CAAC;;;;;;;;;;AAEhH,+DAA+D;AAM/D,MAAM,OAAO,qCACT,SAAQ,8BAA0C;IAElD,2BAA2B,CAAY;IACvC,IAAI,mBAAmB;QACnB,OAAQ,IAAI,CAAC,QAA2E,CAAC,kBAAkB,CAAC;IAChH,CAAC;IAED,IAAI,oBAAoB;QACpB,OAAO,IAAI,CAAC,QAA0E,CAAC;IAC3F,CAAC;IAEQ,QAAQ;QACb,KAAK,CAAC,QAAQ,EAAE,CAAC;QACjB,IAAI,CAAC,2BAA2B,GAAG,eAAe,CAAC,SAAS,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;IAC3F,CAAC;IAED;;;;OAIG;IACH,QAAQ,CAAC,KAAmC,EAAE,UAA4B;QACtE,MAAM,KAAK,GAAW,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QAC5D,IAAI,CAAC,2BAA2B,CAAC,KAAK,CAAC,CAAC;QACxC,UAAU,CAAC,KAAK,GAAG,EAAE,CAAC;QACtB,IAAI,CAAC,yBAAyB,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAEjD,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,yBAAyB,CAAC,KAAc;QACpC,MAAM,WAAW,GAAY,KAAgB,CAAC,WAAW,EAAE,CAAC;QAC5D,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC;IACnH,CAAC;uGAtCQ,qCAAqC;2FAArC,qCAAqC,8FCdlD,88GA0EiB;;2FD5DJ,qCAAqC;kBALjD,SAAS;+BACI,iCAAiC","sourcesContent":["/* eslint-disable jsdoc/require-jsdoc */\nimport { Component, OnInit } from '@angular/core';\nimport { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete';\nimport { BaseEntityType } from '../../../../classes/entity.model';\nimport { AutocompleteStringChipsArrayDecoratorConfigInternal } from '../../../../decorators/array/array-decorator-internal.data';\nimport { LodashUtilities } from '../../../../encapsulation/lodash.utilities';\nimport { ArrayStringChipsInputComponent } from '../array-string-chips-input/array-string-chips-input.component';\n\n// eslint-disable-next-line angular/prefer-standalone-component\n@Component({\n    selector: 'array-string-autocomplete-chips',\n    templateUrl: './array-string-autocomplete-chips.component.html',\n    styleUrls: ['./array-string-autocomplete-chips.component.scss']\n})\nexport class ArrayStringAutocompleteChipsComponent<EntityType extends BaseEntityType<EntityType>>\n    extends ArrayStringChipsInputComponent<EntityType> implements OnInit {\n\n    filteredAutocompleteStrings!: string[];\n    get autocompleteStrings(): string[] {\n        return (this.metadata as unknown as AutocompleteStringChipsArrayDecoratorConfigInternal).autocompleteValues;\n    }\n\n    get autocompleteMetadata(): AutocompleteStringChipsArrayDecoratorConfigInternal {\n        return this.metadata as unknown as AutocompleteStringChipsArrayDecoratorConfigInternal;\n    }\n\n    override ngOnInit(): void {\n        super.ngOnInit();\n        this.filteredAutocompleteStrings = LodashUtilities.cloneDeep(this.autocompleteStrings);\n    }\n\n    /**\n     * Handles adding a string to the array when an autocomplete value has been selected.\n     * @param event - The autocomplete selected event.\n     * @param chipsInput - The element where the user typed the value.\n     */\n    selected(event: MatAutocompleteSelectedEvent, chipsInput: HTMLInputElement): void {\n        const value: string = (event.option.viewValue || '').trim();\n        this.validateAndSetPropertyValue(value);\n        chipsInput.value = '';\n        this.filterAutocompleteStrings(chipsInput.value);\n\n        this.emitChange();\n    }\n\n    /**\n     * Dynamically filters the Autocomplete options when the user inputs something.\n     * @param input - The input of the user.\n     */\n    filterAutocompleteStrings(input: unknown): void {\n        const filterValue: string = (input as string).toLowerCase();\n        this.filteredAutocompleteStrings = this.autocompleteStrings.filter(s => s.toLowerCase().includes(filterValue));\n    }\n}","<!-- eslint-disable angular/no-call-expression -->\n<!-- <mat-form-field>\n    <mat-label>{{metadata.displayName}}</mat-label>\n    <mat-chip-grid #chipList\n        (ngModelChange)=\"emitChange()\"\n        [(ngModel)]=\"propertyValue\" [name]=\"key.toString()\" #model=\"ngModel\"\n        [required]=\"metadata.required(entity)\"\n        [disabled]=\"isReadOnly\"\n    >\n        <mat-chip *ngFor=\"let value of propertyValue\" (removed)=\"removeStringChipArrayValue(value)\">\n            {{value}}\n            <button type=\"button\" matChipRemove>\n                <i class=\"{{metadata.deleteIcon}}\"></i>\n            </button>\n        </mat-chip>\n        <input matInput\n            [matChipInputFor]=\"chipList\"\n            [matAutocomplete]=\"auto\"\n            [matChipInputAddOnBlur]=\"true\"\n            (matChipInputTokenEnd)=\"addStringChipArrayValue($event)\"\n            (keyup)=\"filterAutocompleteStrings(chipsInput)\"\n            [(ngModel)]=\"chipsInput\" [name]=\"key.toString()\" #chipsModel=\"ngModel\"\n            #chipsElement\n            [minlength]='metadata.minLength ?? null'\n            [maxlength]='metadata.maxLength ?? null'\n            [pattern]=\"metadata.regex ?? '[\\\\s\\\\S]*'\"\n        >\n        <mat-error *ngIf=\"chipsModel.errors\">{{getValidationErrorMessage(chipsModel)}}</mat-error>\n    </mat-chip-grid>\n    <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"selected($event, chipsElement)\">\n        <mat-option *ngFor=\"let value of metadata.autocompleteValues\" [value]=\"value\">\n          {{value}}\n        </mat-option>\n    </mat-autocomplete>\n    <mat-error *ngIf=\"!chipsModel.errors\">{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field> -->\n\n<mat-form-field>\n    <mat-label>{{metadata.displayName}}</mat-label>\n    <mat-chip-grid #chipGrid\n        #model=\"ngModel\" [name]=\"name\"\n        [required]=\"metadata.required(entity)\"\n        [disabled]=\"isReadOnly\"\n        [(ngModel)]=\"propertyValue\"\n    >\n        <mat-chip-row *ngFor=\"let value of propertyValue\" (removed)=\"removeStringChipArrayValue(value)\">\n            {{value}}\n            <button type=\"button\" matChipRemove>\n                <i [class]=\"metadata.deleteIcon\"></i>\n            </button>\n        </mat-chip-row>\n        <input #chipsModel=\"ngModel\"\n            #chipsElement\n            matInput\n            [matChipInputFor]=\"chipGrid\"\n            [matAutocomplete]=\"auto\"\n            [matChipInputAddOnBlur]=\"true\"\n            [name]=\"uuid2\"\n            [minlength]=\"metadata.minLength ?? null\"\n            [maxlength]=\"metadata.maxLength ?? null\"\n            [pattern]=\"metadata.regex ?? '[\\\\s\\\\S]*'\"\n            [includedIn]=\"autocompleteMetadata.restrictToOptions ? autocompleteStrings : undefined\"\n            [(ngModel)]=\"chipsInput\"\n            (keyup)=\"filterAutocompleteStrings(propertyValue)\"\n            (matChipInputTokenEnd)=\"addStringChipArrayValue($event)\"\n        />\n        <mat-error *ngIf=\"chipsModel.errors\">{{getValidationErrorMessage(chipsModel)}}</mat-error>\n    </mat-chip-grid>\n    <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"selected($event, chipsElement)\">\n        <mat-option *ngFor=\"let value of filteredAutocompleteStrings\" [value]=\"value\">\n          {{value}}\n        </mat-option>\n    </mat-autocomplete>\n    <mat-error *ngIf=\"!chipsModel.errors\">{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>"]}
|
|
65
|
+
args: [{ selector: 'array-string-autocomplete-chips', template: "<!-- eslint-disable angular/no-call-expression -->\n\n<mat-form-field>\n <mat-label>{{metadata.displayName}}</mat-label>\n <mat-chip-grid #chipGrid\n #model=\"ngModel\" [name]=\"name\"\n [required]=\"metadata.required(entity)\"\n [disabled]=\"isReadOnly\"\n [(ngModel)]=\"propertyValue\"\n (ngModelChange)=\"setValidationErrors(model, chipsModel)\"\n >\n <mat-chip-row *ngFor=\"let value of propertyValue\" (removed)=\"removeStringChipArrayValue(value)\">\n {{value}}\n <button type=\"button\" matChipRemove>\n <i [class]=\"metadata.deleteIcon\"></i>\n </button>\n </mat-chip-row>\n <input #chipsModel=\"ngModel\"\n #chipsElement\n matInput\n [matChipInputFor]=\"chipGrid\"\n [matAutocomplete]=\"auto\"\n [matChipInputAddOnBlur]=\"true\"\n [name]=\"uuid2\"\n [minlength]=\"metadata.minLength ?? null\"\n [maxlength]=\"metadata.maxLength ?? null\"\n [pattern]=\"metadata.regex ?? '[\\\\s\\\\S]*'\"\n [includedIn]=\"autocompleteMetadata.restrictToOptions ? autocompleteStrings : undefined\"\n [(ngModel)]=\"chipsInput\"\n (keyup)=\"filterAutocompleteStrings(chipsInput)\"\n (matChipInputTokenEnd)=\"addStringChipArrayValue($event, !!chipsModel.errors)\"\n />\n </mat-chip-grid>\n <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"selected($event, chipsElement, model, chipsModel)\">\n <mat-option *ngFor=\"let value of filteredAutocompleteStrings\" [value]=\"value\">\n {{value}}\n </mat-option>\n </mat-autocomplete>\n <mat-error>{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>", styles: ["mat-form-field{width:100%}\n"] }]
|
|
66
|
+
}], ctorParameters: function () { return [{ type: i0.EnvironmentInjector }]; } });
|
|
67
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"array-string-autocomplete-chips.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-material-entity/src/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.ts","../../../../../../../projects/ngx-material-entity/src/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.html"],"names":[],"mappings":"AAAA,wCAAwC;AACxC,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAU,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAK9F,OAAO,EAAE,eAAe,EAAE,MAAM,4CAA4C,CAAC;AAC7E,OAAO,EAAE,8BAA8B,EAAE,MAAM,gEAAgE,CAAC;;;;;;;;;;AAEhH,+DAA+D;AAM/D,MAAM,OAAO,qCACT,SAAQ,8BAA0C;IASrB;IAP7B,2BAA2B,CAAY;IACvC,mBAAmB,GAAa,EAAE,CAAC;IAEnC,IAAI,oBAAoB;QACpB,OAAO,IAAI,CAAC,QAA0E,CAAC;IAC3F,CAAC;IAED,YAA6B,QAA6B;QACtD,KAAK,EAAE,CAAC;QADiB,aAAQ,GAAR,QAAQ,CAAqB;IAE1D,CAAC;IAEQ,KAAK,CAAC,QAAQ;QACnB,KAAK,CAAC,QAAQ,EAAE,CAAC;QACjB,MAAM,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE;YAClD,IAAI,CAAC,mBAAmB,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,kBAAkB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC3F,IAAI,CAAC,2BAA2B,GAAG,eAAe,CAAC,SAAS,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;QAC3F,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;;;OAMG;IACH,QAAQ,CAAC,KAAmC,EAAE,UAA4B,EAAE,KAAc,EAAE,UAAmB;QAC3G,8DAA8D;QAC9D,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,IAAI,EAAE,CAAC;QAC9C,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAe,CAAC,CAAC;QACtD,UAAU,CAAC,KAAK,GAAG,EAAE,CAAC;QACtB,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;QACrB,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAChD,UAAU,CAAC,OAAO,CAAC,sBAAsB,EAAE,CAAC;QAC5C,KAAK,CAAC,OAAO,CAAC,sBAAsB,EAAE,CAAC;QACvC,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QAC5C,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,yBAAyB,CAAC,KAAa;QACnC,MAAM,WAAW,GAAW,KAAK,CAAC,WAAW,EAAE,CAAC;QAChD,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC;IACnH,CAAC;uGAjDQ,qCAAqC;2FAArC,qCAAqC,8FCflD,ixDAuCiB;;2FDxBJ,qCAAqC;kBALjD,SAAS;+BACI,iCAAiC","sourcesContent":["/* eslint-disable jsdoc/require-jsdoc */\nimport { Component, EnvironmentInjector, OnInit, runInInjectionContext } from '@angular/core';\nimport { NgModel } from '@angular/forms';\nimport { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete';\nimport { BaseEntityType } from '../../../../classes/entity.model';\nimport { AutocompleteStringChipsArrayDecoratorConfigInternal } from '../../../../decorators/array/array-decorator-internal.data';\nimport { LodashUtilities } from '../../../../encapsulation/lodash.utilities';\nimport { ArrayStringChipsInputComponent } from '../array-string-chips-input/array-string-chips-input.component';\n\n// eslint-disable-next-line angular/prefer-standalone-component\n@Component({\n    selector: 'array-string-autocomplete-chips',\n    templateUrl: './array-string-autocomplete-chips.component.html',\n    styleUrls: ['./array-string-autocomplete-chips.component.scss']\n})\nexport class ArrayStringAutocompleteChipsComponent<EntityType extends BaseEntityType<EntityType>>\n    extends ArrayStringChipsInputComponent<EntityType> implements OnInit {\n\n    filteredAutocompleteStrings!: string[];\n    autocompleteStrings: string[] = [];\n\n    get autocompleteMetadata(): AutocompleteStringChipsArrayDecoratorConfigInternal {\n        return this.metadata as unknown as AutocompleteStringChipsArrayDecoratorConfigInternal;\n    }\n\n    constructor(private readonly injector: EnvironmentInjector) {\n        super();\n    }\n\n    override async ngOnInit(): Promise<void> {\n        super.ngOnInit();\n        await runInInjectionContext(this.injector, async () => {\n            this.autocompleteStrings = await this.autocompleteMetadata.autocompleteValues(this.entity);\n            this.filteredAutocompleteStrings = LodashUtilities.cloneDeep(this.autocompleteStrings);\n        });\n    }\n\n    /**\n     * Handles adding a string to the array when an autocomplete value has been selected.\n     * @param event - The autocomplete selected event.\n     * @param chipsInput - The element where the user typed the value.\n     * @param model - The model of the string array.\n     * @param chipsModel - The model of the single string input.\n     */\n    selected(event: MatAutocompleteSelectedEvent, chipsInput: HTMLInputElement, model: NgModel, chipsModel: NgModel): void {\n        // validation is not needed as selected options are all valid.\n        this.propertyValue = this.propertyValue ?? [];\n        this.propertyValue.push(event.option.value as string);\n        chipsInput.value = '';\n        this.chipsInput = '';\n        this.filterAutocompleteStrings(this.chipsInput);\n        chipsModel.control.updateValueAndValidity();\n        model.control.updateValueAndValidity();\n        this.setValidationErrors(model, chipsModel);\n        this.emitChange();\n    }\n\n    /**\n     * Dynamically filters the Autocomplete options when the user inputs something.\n     * @param input - The input of the user.\n     */\n    filterAutocompleteStrings(input: string): void {\n        const filterValue: string = input.toLowerCase();\n        this.filteredAutocompleteStrings = this.autocompleteStrings.filter(s => s.toLowerCase().includes(filterValue));\n    }\n}","<!-- eslint-disable angular/no-call-expression -->\n\n<mat-form-field>\n    <mat-label>{{metadata.displayName}}</mat-label>\n    <mat-chip-grid #chipGrid\n        #model=\"ngModel\" [name]=\"name\"\n        [required]=\"metadata.required(entity)\"\n        [disabled]=\"isReadOnly\"\n        [(ngModel)]=\"propertyValue\"\n        (ngModelChange)=\"setValidationErrors(model, chipsModel)\"\n    >\n        <mat-chip-row *ngFor=\"let value of propertyValue\" (removed)=\"removeStringChipArrayValue(value)\">\n            {{value}}\n            <button type=\"button\" matChipRemove>\n                <i [class]=\"metadata.deleteIcon\"></i>\n            </button>\n        </mat-chip-row>\n        <input #chipsModel=\"ngModel\"\n            #chipsElement\n            matInput\n            [matChipInputFor]=\"chipGrid\"\n            [matAutocomplete]=\"auto\"\n            [matChipInputAddOnBlur]=\"true\"\n            [name]=\"uuid2\"\n            [minlength]=\"metadata.minLength ?? null\"\n            [maxlength]=\"metadata.maxLength ?? null\"\n            [pattern]=\"metadata.regex ?? '[\\\\s\\\\S]*'\"\n            [includedIn]=\"autocompleteMetadata.restrictToOptions ? autocompleteStrings : undefined\"\n            [(ngModel)]=\"chipsInput\"\n            (keyup)=\"filterAutocompleteStrings(chipsInput)\"\n            (matChipInputTokenEnd)=\"addStringChipArrayValue($event, !!chipsModel.errors)\"\n        />\n    </mat-chip-grid>\n    <mat-autocomplete #auto=\"matAutocomplete\" (optionSelected)=\"selected($event, chipsElement, model, chipsModel)\">\n        <mat-option *ngFor=\"let value of filteredAutocompleteStrings\" [value]=\"value\">\n          {{value}}\n        </mat-option>\n    </mat-autocomplete>\n    <mat-error>{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>"]}
|