@smallpearl/ngx-helper 0.33.29 → 0.33.30
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.
|
@@ -180,10 +180,12 @@ function convertHttpContextInputToHttpContext(context, reqContext) {
|
|
|
180
180
|
* respectively.
|
|
181
181
|
*/
|
|
182
182
|
class SPMatEntityCrudFormBase {
|
|
183
|
-
|
|
184
|
-
|
|
183
|
+
// bridge mode inputs
|
|
184
|
+
entity = input();
|
|
185
|
+
bridge = input();
|
|
185
186
|
params = input();
|
|
186
|
-
//
|
|
187
|
+
// END bridge mode inputs
|
|
188
|
+
// standalone mode inputs
|
|
187
189
|
// Entity name, which is used to parse sideloaded entity responses
|
|
188
190
|
entityName = input();
|
|
189
191
|
// Base CRUD URL, which is the GET-list-of-entities/POST-to-create
|
|
@@ -193,7 +195,7 @@ class SPMatEntityCrudFormBase {
|
|
|
193
195
|
httpReqContext = input();
|
|
194
196
|
// ID key, defaults to 'id'
|
|
195
197
|
idKey = input('id');
|
|
196
|
-
//
|
|
198
|
+
// END standalone mode inputs
|
|
197
199
|
// IMPLEMENTATION
|
|
198
200
|
loadEntity$;
|
|
199
201
|
_entity = signal(undefined);
|
|
@@ -209,6 +211,10 @@ class SPMatEntityCrudFormBase {
|
|
|
209
211
|
transloco = inject(TranslocoService);
|
|
210
212
|
cdr = inject(ChangeDetectorRef);
|
|
211
213
|
http = inject(HttpClient);
|
|
214
|
+
// This is really not necessary. We can check for this.bridge() directly.
|
|
215
|
+
mode = computed(() => {
|
|
216
|
+
return this.bridge() ? 'bridge' : 'standalone';
|
|
217
|
+
});
|
|
212
218
|
canCancelEdit = () => {
|
|
213
219
|
return this._canCancelEdit();
|
|
214
220
|
};
|
|
@@ -220,9 +226,10 @@ class SPMatEntityCrudFormBase {
|
|
|
220
226
|
return true;
|
|
221
227
|
}
|
|
222
228
|
ngOnInit() {
|
|
223
|
-
//
|
|
229
|
+
// Validate inputs. Either bridge or (baseUrl and entityName) must be
|
|
224
230
|
// defined.
|
|
225
|
-
if (
|
|
231
|
+
if (this.mode() === 'standalone' &&
|
|
232
|
+
(!this.getBaseUrl() || !this.getEntityName())) {
|
|
226
233
|
throw new Error('SPMatEntityCrudFormBase: baseUrl and entityName inputs must be defined in standalone mode.');
|
|
227
234
|
}
|
|
228
235
|
this.loadEntity$ = (typeof this.entity() === 'object' || this.entity() === undefined
|
|
@@ -271,13 +278,13 @@ class SPMatEntityCrudFormBase {
|
|
|
271
278
|
if (!resp || typeof resp !== 'object') {
|
|
272
279
|
return undefined;
|
|
273
280
|
}
|
|
274
|
-
const entityName = this.
|
|
281
|
+
const entityName = this.getEntityName();
|
|
275
282
|
if (resp.hasOwnProperty(this.getIdKey())) {
|
|
276
283
|
return resp;
|
|
277
284
|
}
|
|
278
285
|
else if (entityName && resp.hasOwnProperty(entityName)) {
|
|
279
286
|
// const sideloadDataMap = this.sideloadDataMap();
|
|
280
|
-
return sideloadToComposite(resp,
|
|
287
|
+
return sideloadToComposite(resp, entityName, this.getIdKey());
|
|
281
288
|
}
|
|
282
289
|
return undefined;
|
|
283
290
|
}
|
|
@@ -308,7 +315,9 @@ class SPMatEntityCrudFormBase {
|
|
|
308
315
|
? this.create(value)
|
|
309
316
|
: this.update(this._entity()[this.getIdKey()], value);
|
|
310
317
|
this.sub$.add(obs
|
|
311
|
-
?.pipe(tap(entity => this._entity()
|
|
318
|
+
?.pipe(tap((entity) => this._entity()
|
|
319
|
+
? this.onPostUpdate(entity)
|
|
320
|
+
: this.onPostCreate(entity)), setServerErrorsAsFormErrors(this._form(), this.cdr))
|
|
312
321
|
.subscribe());
|
|
313
322
|
}
|
|
314
323
|
onPostCreate(entity) {
|
|
@@ -333,13 +342,6 @@ class SPMatEntityCrudFormBase {
|
|
|
333
342
|
return bridge.loadEntity(entityId, params);
|
|
334
343
|
}
|
|
335
344
|
// Try to load using baseUrl.
|
|
336
|
-
if (!this.baseUrl()) {
|
|
337
|
-
console.warn(`SPMatEntityCrudFormBase.load: No bridge defined, baseUrl input is undefined. Returning undefined.`);
|
|
338
|
-
return new Observable((subscriber) => {
|
|
339
|
-
subscriber.next(undefined);
|
|
340
|
-
subscriber.complete();
|
|
341
|
-
});
|
|
342
|
-
}
|
|
343
345
|
const url = this.getEntityUrl(entityId);
|
|
344
346
|
return this.http
|
|
345
347
|
.get(this.getEntityUrl(entityId), {
|
|
@@ -361,13 +363,10 @@ class SPMatEntityCrudFormBase {
|
|
|
361
363
|
if (bridge) {
|
|
362
364
|
return bridge.create(values);
|
|
363
365
|
}
|
|
364
|
-
const url = this.baseUrl();
|
|
365
|
-
if (!url) {
|
|
366
|
-
console.warn('SPMatEntityCrudFormBase.create: Cannot create entity as neither bridge nor baseUrl inputs are provided.');
|
|
367
|
-
return of(undefined);
|
|
368
|
-
}
|
|
369
366
|
return this.http
|
|
370
|
-
.post(
|
|
367
|
+
.post(this.getBaseUrl(), values, {
|
|
368
|
+
context: this.getRequestContext(),
|
|
369
|
+
})
|
|
371
370
|
.pipe(map((resp) => this.getEntityFromLoadResponse(resp)));
|
|
372
371
|
}
|
|
373
372
|
/**
|
|
@@ -382,23 +381,47 @@ class SPMatEntityCrudFormBase {
|
|
|
382
381
|
if (bridge) {
|
|
383
382
|
return bridge.update(id, values);
|
|
384
383
|
}
|
|
385
|
-
const url = this.baseUrl();
|
|
386
|
-
if (!url) {
|
|
387
|
-
console.warn('SPMatEntityCrudFormBase.update: Cannot update entity as neither bridge nor baseUrl inputs are provided.');
|
|
388
|
-
return of(undefined);
|
|
389
|
-
}
|
|
390
384
|
return this.http
|
|
391
385
|
.patch(this.getEntityUrl(id), values, {
|
|
392
386
|
context: this.getRequestContext(),
|
|
393
387
|
})
|
|
394
388
|
.pipe(map((resp) => this.getEntityFromLoadResponse(resp)));
|
|
395
389
|
}
|
|
390
|
+
/**
|
|
391
|
+
* Wrapper around entityName input to get the entity name. If `bridge` input
|
|
392
|
+
* is defined, then its `getEntityName()` method is used. This allows
|
|
393
|
+
* derived classes to override this method to provide custom logic to
|
|
394
|
+
* determine the entity name.
|
|
395
|
+
* @returns
|
|
396
|
+
*/
|
|
397
|
+
getEntityName() {
|
|
398
|
+
const bridge = this.bridge();
|
|
399
|
+
if (bridge) {
|
|
400
|
+
return bridge.getEntityName();
|
|
401
|
+
}
|
|
402
|
+
return this.entityName();
|
|
403
|
+
}
|
|
404
|
+
/**
|
|
405
|
+
* Returns the baseUrl. Derived classes can override this to provide custom
|
|
406
|
+
* logic to determine the baseUrl.
|
|
407
|
+
* @returns
|
|
408
|
+
*/
|
|
409
|
+
getBaseUrl() {
|
|
410
|
+
return this.baseUrl();
|
|
411
|
+
}
|
|
412
|
+
/**
|
|
413
|
+
* Returns the entity URL for the given entity id. If `bridge` input is
|
|
414
|
+
* defined, then its `getEntityUrl()` method is used. Otherwise, the URL is
|
|
415
|
+
* derived from `baseUrl` input.
|
|
416
|
+
* @param entityId
|
|
417
|
+
* @returns
|
|
418
|
+
*/
|
|
396
419
|
getEntityUrl(entityId) {
|
|
397
420
|
const bridge = this.bridge();
|
|
398
421
|
if (bridge) {
|
|
399
422
|
return bridge.getEntityUrl(entityId);
|
|
400
423
|
}
|
|
401
|
-
const baseUrl = this.
|
|
424
|
+
const baseUrl = this.getBaseUrl();
|
|
402
425
|
if (baseUrl) {
|
|
403
426
|
const urlParts = baseUrl.split('?');
|
|
404
427
|
return `${urlParts[0]}${String(entityId)}/${urlParts[1] ? '?' + urlParts[1] : ''}`;
|
|
@@ -415,7 +438,7 @@ class SPMatEntityCrudFormBase {
|
|
|
415
438
|
return context;
|
|
416
439
|
}
|
|
417
440
|
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.1.6", ngImport: i0, type: SPMatEntityCrudFormBase, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
418
|
-
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "19.1.6", type: SPMatEntityCrudFormBase, isStandalone: false, selector: "_#_sp-mat-entity-crud-form-base_#_", inputs: { entity: { classPropertyName: "entity", publicName: "entity", isSignal: true, isRequired:
|
|
441
|
+
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "19.1.6", type: SPMatEntityCrudFormBase, isStandalone: false, selector: "_#_sp-mat-entity-crud-form-base_#_", inputs: { entity: { classPropertyName: "entity", publicName: "entity", isSignal: true, isRequired: false, transformFunction: null }, bridge: { classPropertyName: "bridge", publicName: "bridge", isSignal: true, isRequired: false, transformFunction: null }, params: { classPropertyName: "params", publicName: "params", isSignal: true, isRequired: false, transformFunction: null }, entityName: { classPropertyName: "entityName", publicName: "entityName", isSignal: true, isRequired: false, transformFunction: null }, baseUrl: { classPropertyName: "baseUrl", publicName: "baseUrl", isSignal: true, isRequired: false, transformFunction: null }, httpReqContext: { classPropertyName: "httpReqContext", publicName: "httpReqContext", isSignal: true, isRequired: false, transformFunction: null }, idKey: { classPropertyName: "idKey", publicName: "idKey", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: ``, isInline: true });
|
|
419
442
|
}
|
|
420
443
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.1.6", ngImport: i0, type: SPMatEntityCrudFormBase, decorators: [{
|
|
421
444
|
type: Component,
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"smallpearl-ngx-helper-mat-entity-crud.mjs","sources":["../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/convert-context-input-to-http-context.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud-form-base.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud-types.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/providers.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/default-config.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/form-view-host.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/preview-host.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud-internal-types.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/preview-pane.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/smallpearl-ngx-helper-mat-entity-crud.ts"],"sourcesContent":["import { HttpContext, HttpContextToken } from '@angular/common/http';\n\nexport type HttpContextInput =\n | [[HttpContextToken<any>, any]]\n | [HttpContextToken<any>, any]\n | HttpContext;\n\n/**\n * Converts array of HttpContextToken key, value pairs to HttpContext\n * object in argument 'context'.\n * @param context HTTP context to which the key, value pairs are added\n * @param reqContext HttpContextToken key, value pairs array\n * @returns HttpContext object, with the key, value pairs added. This is\n * the same object as the 'context' argument.\n */\nexport function convertHttpContextInputToHttpContext(\n context: HttpContext,\n reqContext: HttpContextInput\n): HttpContext {\n if (reqContext instanceof HttpContext) {\n // reqContext is already an HttpContext object.\n for (const k of reqContext.keys()) {\n context.set(k, reqContext.get(k));\n }\n } else if (Array.isArray(reqContext) && reqContext.length == 2 && !Array.isArray(reqContext[0])) {\n // one dimensional array of a key, value pair.\n context.set(reqContext[0], reqContext[1]);\n } else {\n reqContext.forEach(([k, v]) => context.set(k, v));\n }\n return context;\n}\n","import { HttpClient, HttpContext, HttpParams } from '@angular/common/http';\nimport { ChangeDetectorRef, Component, computed, inject, input, OnDestroy, OnInit, signal } from '@angular/core';\nimport { AbstractControl, UntypedFormGroup } from '@angular/forms';\nimport { TranslocoService } from '@jsverse/transloco';\nimport { setServerErrorsAsFormErrors } from '@smallpearl/ngx-helper/forms';\nimport { map, Observable, of, Subscription, tap } from 'rxjs';\n// import { getEntityCrudConfig } from './default-config';\nimport { sideloadToComposite } from '@smallpearl/ngx-helper/sideload';\nimport { convertHttpContextInputToHttpContext, HttpContextInput } from './convert-context-input-to-http-context';\nimport { SPMatEntityCrudCreateEditBridge } from './mat-entity-crud-types';\n\n/**\n * This is a convenience base class that clients can derive from to implement\n * their CRUD form component. Particularly this class registers the change\n * detection hook which will be called when the user attempts to close the\n * form's parent container pane via the Close button on the top right.\n *\n * This button behaves like a Cancel button in a desktop app and therefore if\n * the user has entered any data in the form's controls, (determined by\n * checking form.touched), then a 'Lose Changes' prompt is displayed allowing\n * the user to cancel the closure.\n *\n * The `@Component` decorator is fake to keep the VSCode angular linter quiet.\n *\n * This class can be used in two modes:\n *\n * I. SPMatEntityCrudComponent mode\n * This mode relies on a bridge interface that implements the\n * SPMatEntityCrudCreateEditBridge interface to perform the entity\n * load/create/update operations. This is the intended mode when the\n * component is used as a part of the SPMatEntityCrudComponent to\n * create/update an entity. This mode requires the following properties\n * to be set:\n * - entity: TEntity | TEntity[IdKey] | undefined (for create)\n * - bridge: SPMatEntityCrudCreateEditBridge\n *\n * II. Standalone mode\n * This mode does not rely on the bridge interface and the component\n * itself performs the entity load/create/update operations.\n * This mode requires the following properties to be set:\n * - entity: TEntity | TEntity[IdKey] | undefined (for create)\n * - baseUrl: string - Base URL for CRUD operations. This URL does not\n * include the entity id. The entity id will be appended to this URL\n * for entity load and update operations. For create operation, this\n * URL is used as is.\n * - entityName: string - Name of the entity, used to parse sideloaded\n * entity responses.\n * - httpReqContext?: HttpContextInput - Optional HTTP context to be\n * passed to the HTTP requests. For instance, if your app has a HTTP\n * interceptor that adds authentication tokens to the requests based\n * on a HttpContextToken, then you can pass that token here.\n *\n * I. SPMatEntityCrudComponent mode:\n *\n * 1. Declare a FormGroup<> type as\n *\n * ```\n * type MyForm = FormGroup<{\n * name: FormControl<string>;\n * type: FormControl<string>;\n * notes: FormControl<string>;\n * }>;\n * ```\n *\n * 2. Derive your form's component class from this and implement the\n * createForm() method returing the FormGroup<> instance that matches\n * the FormGroup concrete type above.\n *\n * ```\n * class MyFormComponent extends SPMatEntityCrudFormBase<MyForm, MyEntity> {\n * constructor() {\n * super()\n * }\n * createForm() {\n * return new FormGroup([...])\n * }\n * }\n * ```\n *\n * 3. If your form's value requires manipulation before being sent to the\n * server, override `getFormValue()` method and do it there before returning\n * the modified values.\n *\n * 4. Wire up the form in the template as below\n *\n * ```html\n * @if (loadEntity$ | async) {\n * <form [formGroup]='form'.. (ngSubmit)=\"onSubmit()\">\n * <button type=\"submit\">Submit</button>\n * </form>\n * } @else {\n * <div>Loading...</div>\n * }\n * ```\n *\n * Here `loadEntity$` is an Observable<boolean> that upon emission of `true`\n * indicates that the entity has been loaded from server (in case of edit)\n * and the form is ready to be displayed. Note that if the full entity was\n * passed in the `entity` input property, then no server load is necessary\n * and the form will be created immediately.\n *\n * 5. In the parent component that hosts the SPMatEntityCrudComponent, set\n * the `entity` and `bridge` input properties of this component to\n * appropriate values. For instance, if your form component has the\n * selector `app-my-entity-form`, then the parent component's template\n * will have:\n *\n * ```html\n * <sp-mat-entity-crud\n * ...\n * createEditFormTemplate=\"entityFormTemplate\"\n * ></sp-mat-entity-crud>\n * <ng-template #entityFormTemplate let-data=\"data\">\n * <app-my-entity-form\n * [entity]=\"data.entity\"\n * [bridge]=\"data.bridge\"\n * ></app-my-entity-form>\n * </ng-template>\n * ```\n *\n * II. Standalone mode\n *\n * 1..4. Same as above, except set the required `bridge` input to `undefined`.\n * 5. Initialize the component's inputs `baseUrl` and `entityName` with the\n * appropriate values. If you would like to pass additional HTTP context to\n * the HTTP requests, then set the `httpReqContext` input as well.\n * If the entity uses an id key other than 'id', then set the `idKey` input\n * to the appropriate id key name.\n * 6. If you want to retrieve the created/updated entity after the create/update\n * operation, override the `onPostCreate()` and/or `onPostUpdate()` methods\n * respectively.\n */\n@Component({\n selector: '_#_sp-mat-entity-crud-form-base_#_',\n template: ``,\n standalone: false,\n})\nexport abstract class SPMatEntityCrudFormBase<\n TFormGroup extends AbstractControl,\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> implements OnInit, OnDestroy\n{\n entity = input.required<TEntity | TEntity[IdKey]>();\n bridge = input.required<SPMatEntityCrudCreateEditBridge>();\n params = input<any>();\n\n // --- BEGIN inputs used when `bridge` input is undefined\n // Entity name, which is used to parse sideloaded entity responses\n entityName = input<string>();\n // Base CRUD URL, which is the GET-list-of-entities/POST-to-create\n // URL. Update URL will be derived from this ias `baseUrl()/${TEntity[IdKey]}`\n baseUrl = input<string>();\n // Additional request context to be passed to the request\n httpReqContext = input<HttpContextInput | undefined>();\n // ID key, defaults to 'id'\n idKey = input<string>('id');\n // -- END inputs used when `bridge` input is undefined\n\n // IMPLEMENTATION\n loadEntity$!: Observable<boolean>;\n _entity = signal<TEntity | undefined>(undefined);\n sub$ = new Subscription();\n\n // Store for internal form signal. form() is computed from this.\n _form = signal<TFormGroup | undefined>(undefined);\n // Force typecast to TFormGroup so that we can use it in the template\n // without having to use the non-nullable operator ! with every reference\n // of form(). In any case the form() signal is always set in ngOnInit()\n // method after the form is created. And if form() is not set, then there\n // will be errors while loading the form in the template.\n form = computed(() => this._form() as TFormGroup);\n\n transloco = inject(TranslocoService);\n cdr = inject(ChangeDetectorRef);\n http = inject(HttpClient);\n\n canCancelEdit = () => {\n return this._canCancelEdit();\n };\n\n _canCancelEdit() {\n const form = this._form();\n if (form && form.touched) {\n return window.confirm(\n this.transloco.translate('spMatEntityCrud.loseChangesConfirm')\n );\n }\n return true;\n }\n\n ngOnInit() {\n // validate inputs. Either bridge or (baseUrl and entityName) must be\n // defined.\n if (!this.bridge() && (!this.baseUrl() || !this.entityName())) {\n throw new Error(\n 'SPMatEntityCrudFormBase: baseUrl and entityName inputs must be defined in standalone mode.'\n );\n }\n this.loadEntity$ = (\n typeof this.entity() === 'object' || this.entity() === undefined\n ? new Observable<TEntity | undefined>((subscriber) => {\n subscriber.next(this.entity() as TEntity | undefined);\n subscriber.complete();\n })\n : this.load(this.entity() as any)\n ).pipe(\n map((resp) => {\n const compositeEntity = this.getEntityFromLoadResponse(resp);\n this._entity.set(compositeEntity);\n this._form.set(this.createForm(compositeEntity));\n const bridge = this.bridge();\n if (bridge && bridge.registerCanCancelEditCallback) {\n bridge.registerCanCancelEditCallback(this.canCancelEdit);\n }\n return true;\n })\n );\n }\n\n ngOnDestroy() {\n this.sub$.unsubscribe();\n }\n\n /**\n * Additional parameters for loading the entity, in case this.entity() value\n * is of type TEntity[IdKey].\n * @returns\n */\n getLoadEntityParams(): string | HttpParams {\n return '';\n }\n\n /**\n * Return the TEntity object from the response returned by the\n * load() method. Typically entity load returns the actual\n * entity object itself. In some cases, where response is sideloaded, the\n * default implementation here uses the `sideloadToComposite()` utility to\n * extract the entity from the response after merging (inplace) the\n * sideloaded data into a composite.\n *\n * If you have a different response shape, or if your sideloaded object\n * response requires custom custom `sideloadDataMap`, override this method\n * and implement your custom logic to extract the TEntity object from the\n * response.\n * @param resp\n * @returns\n */\n getEntityFromLoadResponse(resp: any): TEntity | undefined {\n if (!resp || typeof resp !== 'object') {\n return undefined;\n }\n const entityName = this.entityName();\n if (resp.hasOwnProperty(this.getIdKey())) {\n return resp as TEntity;\n } else if (entityName && resp.hasOwnProperty(entityName)) {\n // const sideloadDataMap = this.sideloadDataMap();\n return sideloadToComposite(\n resp,\n this.entityName()!,\n this.getIdKey()\n ) as TEntity;\n }\n return undefined;\n }\n\n /**\n * Create the TFormGroup FormGroup class that will be used for the reactive\n * form.\n * @param entity\n */\n abstract createForm(entity: TEntity | undefined): TFormGroup;\n\n /**\n * Override to customize the id key name if it's not 'id'\n * @returns The name of the unique identifier key that will be used to\n * extract the entity's id for UPDATE operation.\n */\n getIdKey() {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.getIdKey();\n }\n return this.idKey();\n }\n\n /**\n * Return the form's value to be sent to server as Create/Update CRUD\n * operation data.\n * @returns\n */\n getFormValue() {\n const form = this.form();\n return form ? form.value : undefined;\n }\n\n onSubmit() {\n const value = this.getFormValue();\n const obs = !this._entity()\n ? this.create(value)\n : this.update((this._entity() as any)[this.getIdKey()], value);\n this.sub$.add(\n obs\n ?.pipe(\n tap(entity => this._entity() ? this.onPostUpdate(entity) : this.onPostCreate(entity)),\n setServerErrorsAsFormErrors(\n this._form() as unknown as UntypedFormGroup,\n this.cdr\n )\n )\n .subscribe()\n );\n }\n\n onPostCreate(entity: TEntity) {\n /* empty */\n }\n\n onPostUpdate(entity: TEntity) {\n /* empty */\n }\n\n /**\n * Loads the entity if `this.entity()` is of type TEntity[IdKey]. If `bridge`\n * input is defined, then it's `loadEntity()` method is used to load the\n * entity. Otherwise, then this method attempts to load the entity using\n * HTTP GET from the URL derived from `baseUrl` input.\n * @param entityId\n * @param params\n * @returns\n */\n load(entityId: any): Observable<TEntity> {\n const bridge = this.bridge();\n const params = this.getLoadEntityParams();\n if (bridge) {\n return bridge.loadEntity(entityId, params);\n }\n\n // Try to load using baseUrl.\n if (!this.baseUrl()) {\n console.warn(\n `SPMatEntityCrudFormBase.load: No bridge defined, baseUrl input is undefined. Returning undefined.`\n );\n return new Observable<TEntity>((subscriber) => {\n subscriber.next(undefined as unknown as TEntity);\n subscriber.complete();\n });\n }\n const url = this.getEntityUrl(entityId);\n return this.http\n .get<TEntity>(this.getEntityUrl(entityId), {\n params:\n typeof params === 'string'\n ? new HttpParams({ fromString: params })\n : params,\n context: this.getRequestContext(),\n })\n .pipe(map((resp) => this.getEntityFromLoadResponse(resp) as TEntity));\n }\n\n /**\n * Create a new entity using the bridge if defined, otherwise using HTTP\n * POST to the `baseUrl`.\n * @param values\n * @returns\n */\n protected create(values: any): Observable<TEntity> {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.create(values);\n }\n const url = this.baseUrl();\n if (!url) {\n console.warn(\n 'SPMatEntityCrudFormBase.create: Cannot create entity as neither bridge nor baseUrl inputs are provided.'\n );\n return of(undefined as unknown as TEntity)\n }\n return this.http\n .post<TEntity>(url, values, { context: this.getRequestContext() })\n .pipe(map((resp) => this.getEntityFromLoadResponse(resp) as TEntity));\n }\n\n /**\n * Update an existing entity using the bridge if defined, otherwise using HTTP\n * PATCH to the URL derived from `baseUrl` and the entity id.\n * @param id\n * @param values\n * @returns\n */\n protected update(id: any, values: any): Observable<TEntity> {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.update(id, values);\n }\n const url = this.baseUrl();\n if (!url) {\n console.warn(\n 'SPMatEntityCrudFormBase.update: Cannot update entity as neither bridge nor baseUrl inputs are provided.'\n );\n return of(undefined as unknown as TEntity);\n }\n\n return this.http\n .patch<TEntity>(this.getEntityUrl(id), values, {\n context: this.getRequestContext(),\n })\n .pipe(map((resp) => this.getEntityFromLoadResponse(resp) as TEntity));\n }\n\n protected getEntityUrl(entityId: any): string {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.getEntityUrl(entityId);\n }\n const baseUrl = this.baseUrl();\n if (baseUrl) {\n const urlParts = baseUrl.split('?');\n return `${urlParts[0]}${String(entityId)}/${\n urlParts[1] ? '?' + urlParts[1] : ''\n }`;\n }\n console.warn(\n 'SPMatEntityCrudFormBase.getEntityUrl: Cannot determine entity URL as neither baseUrl nor bridge inputs are provided.'\n );\n return '';\n }\n\n protected getRequestContext(): HttpContext {\n let context = new HttpContext();\n const httpReqContext = this.httpReqContext();\n if (httpReqContext) {\n context = convertHttpContextInputToHttpContext(\n context,\n httpReqContext\n );\n }\n return context;\n }\n}\n","import { HttpContextToken, HttpParams } from \"@angular/common/http\";\nimport { SPContextMenuItem } from \"@smallpearl/ngx-helper/mat-context-menu\";\nimport { Observable } from \"rxjs\";\n\n/**\n * Prototype of the function to parse the CRUD action response.\n */\nexport type SPMatEntityCrudResponseParser = (\n entityName: string,\n idKey: string,\n method: string, // 'create' | 'retrieve' | 'update' | 'delete',\n resp: any\n) => any|undefined;\n\n\n/**\n * Global config for SPMatEntityList component.\n */\nexport interface SPMatEntityCrudConfig {\n /**\n * The item actions that will be shown for each item in the list.\n * This defaults to 'Update' & 'Delete' actions, but can be customized\n * by this property. Note the item actions can be set for individual\n * <sp-mat-entity-crud> component through its itemActions property.\n */\n defaultItemActions?: SPContextMenuItem[];\n /**\n * Global crud response parser.\n */\n crudOpResponseParser?: SPMatEntityCrudResponseParser;\n}\n\n/**\n * This is the interface through which the client provided CRUD form component\n * interacts with the 'host' SPMatEntityCrudComponent. When the form wants to\n * submit an entity to the server (for create or update), it should call the\n * one of the create or update methods. The interface also provides other\n * methods for the form component to interact with SPMatEntityCrudComponent\n * such as refresh its entities list, close the form pane, etc.\n *\n * The interface name has a 'Bridge' as the interface acts as a bridge between\n * the client provided form handler component and the host\n * SPMatEntityCrudComponent.\n */\nexport interface SPMatEntityCrudCreateEditBridge {\n /**\n * Returns the entity name as provided to the host SPMatEntityCrudComponent.\n * @returns The entity name string.\n */\n getEntityName(): string;\n\n /**\n * Returns the entity id key as provided to the host SPMatEntityCrudComponent.\n * @returns The entity id key string.\n */\n getIdKey(): string;\n\n /**\n * Get Entity url\n * @param cancel\n * @returns\n */\n getEntityUrl(entityId: any): string;\n\n /**\n * Close the edit/update form pane. This WON'T call the 'cancelEditCallback'\n * even if one is registered.\n */\n close: (cancel: boolean) => void;\n /**\n * Client form view can register a callback that will be invoked by the\n * framework when user cancels the create/edit operation by clicking on the\n * close button on the top right.\n * @param callback\n * @returns None\n */\n registerCanCancelEditCallback: (callback: () => boolean) => void;\n // Parameters of type any are entity values are typically the output of\n // form.value and therefore their types would not necessarily match TEntity.\n // id can be typed as TEntity[Idkey], but TSC doesn't allow that yet.\n /**\n * Create a new instance of TEntity, by sending a POST request to remote.\n * @param entityValue This is the typically the output of Reactive form's\n * form.value. Since this value's shape may be different from TEntity and is\n * known only to client form, we use 'any'.\n * @returns None\n * @inner Implementation will show a busy wheel centered on the form\n * view while the async function to update the object remains active.\n */\n create: (entityValue: any) => Observable<any>;\n /**\n * Update the entity with id `id` with new values in entityValue.\n * @param id TEntity id\n * @param entityValue Entity values to be updated.\n * @returns None\n * @inner Implementation will show a busy wheel centered on the form\n * view while the async function to update the object remains active.\n */\n update: (id: any, entityValue: any) => Observable<any>;\n\n /**\n * Load the entity with the given id from server.\n * @param id The id of the entity to load.\n * @param params Additional parameters for loading the entity. This will\n * be passed to the underlying data service's 'get' method. This can be a\n * query parameters string or HttpParams object.\n * @returns Observable of the loaded entity.\n */\n loadEntity: (id: any, params: string|HttpParams) => Observable<any>;\n}\n\n/**\n * Prototype of the function that will be used instead of HttpClient for\n * CRUD operations.\n * @param op - the CRUD operation being requested\n * @param entityValue - The entity or entity value upon which the operation\n * is being requested. for 'create' & 'update' this will be the value\n * of the reactive form. This is typically form.value or the 2nd arg to create\n * & update methods of SPMatEntityCrudCreateEditBridge.\n */\nexport type CRUD_OP_FN<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> = (\n op: string,\n id: TEntity[IdKey] | undefined, // valid only for 'get', 'update' & 'delete'\n entityValue: any, // valid only for 'create' & 'update'\n entityCrudComponent: any,\n) => Observable<TEntity | null>;\n\n\nexport type ALLOW_ITEM_ACTION_FN<TEntity> = (entity: TEntity, action: string) => boolean;\n\n/**\n * This interface is used to define sub types for the \"New {{item}}\" button on\n * top of the CRUD UI. An array of these is provided as the value to the\n * component property 'newItemSubTypes'.\n */\nexport interface NewItemSubType {\n /**\n * A role string that will be passed as argument to the (action) event\n * handler. This string allows the event handler to distinguish the selected\n * sub-type.\n *\n * The special keyword '_new_' can be used to activate the\n * `createEditTemplate` template if one is provided. In this case the params\n * element value (see below) can be used in the template to distinguish\n * between different menu items.\n */\n role: string;\n /**\n * Label displayed in the menu representing this role.\n */\n label: string;\n /**\n * Arbitrary value that will be passed to the 'createEditTemplate' in the\n * $implicit template context as 'params'. You can access this in the\n * template like below (see `data.params`):-\n ```\n <ng-template #createEdit let-data>\n <app-create-edit-entity-demo\n [bridge]=\"data.bridge\"\n [entity]=\"data.entity\"\n [params]=\"data.params\"\n ></app-create-edit-entity-demo>\n </ng-template>\n ```\n\n If params is an object and it includes the key 'title', its value will be\n used as the title for the edit form.\n */\n params?: any;\n}\n\nexport type CrudOp = 'create'|'retrieve'|'update'|'delete'|undefined;\n\nexport interface SPMatEntityCrudHttpContext {\n entityName: string;\n entityNamePlural: string;\n endpoint: string;\n op: CrudOp;\n}\n\nexport const SP_MAT_ENTITY_CRUD_HTTP_CONTEXT =\n new HttpContextToken<SPMatEntityCrudHttpContext>(() => ({\n entityName: '',\n entityNamePlural: '',\n endpoint: '',\n op: undefined,\n }));\n","import { InjectionToken } from '@angular/core';\nimport { SPMatEntityCrudConfig } from './mat-entity-crud-types';\n\nexport const SP_MAT_ENTITY_CRUD_CONFIG = new InjectionToken<SPMatEntityCrudConfig>('SPMatEntityCrudConfig');\n","import { inject } from \"@angular/core\";\nimport { SPMatEntityCrudConfig } from \"./mat-entity-crud-types\";\nimport { SP_MAT_ENTITY_CRUD_CONFIG } from \"./providers\";\n\nfunction defaultCrudResponseParser(\n entityName: string,\n idKey: string,\n method: string, // 'create' | 'retrieve' | 'update' | 'delete',\n resp: any\n) {\n // If the response is an object with a property '<idKey>', return it as\n // TEntity.\n if (resp.hasOwnProperty(idKey)) {\n return resp;\n }\n // If the response has an object indexed at '<entityName>' and it has\n // the property '<idKey>', return it as TEntity.\n if (resp.hasOwnProperty(entityName)) {\n const obj = resp[entityName];\n if (obj.hasOwnProperty(idKey)) {\n return obj;\n }\n }\n // Return undefined, indicating that we could't parse the response.\n return undefined;\n}\n\nexport const DefaultSPMatEntityCrudConfig: SPMatEntityCrudConfig = {\n crudOpResponseParser: defaultCrudResponseParser\n};\n\n/**\n * To be called from an object constructor as it internally calls Angular's\n * inject() API.\n * @param userConfig\n * @returns\n */\nexport function getEntityCrudConfig(): SPMatEntityCrudConfig {\n const userCrudConfig = inject(SP_MAT_ENTITY_CRUD_CONFIG, {\n optional: true,\n });\n return {\n ...DefaultSPMatEntityCrudConfig,\n ...(userCrudConfig ?? {}),\n };\n}\n","import { CommonModule } from '@angular/common';\nimport { HttpParams } from '@angular/common/http';\nimport {\n ChangeDetectionStrategy,\n Component,\n computed,\n EmbeddedViewRef,\n inject,\n input,\n OnDestroy,\n OnInit,\n signal,\n TemplateRef,\n viewChild,\n ViewContainerRef\n} from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport { TranslocoModule, TranslocoService } from '@jsverse/transloco';\nimport { SPMatHostBusyWheelDirective } from '@smallpearl/ngx-helper/mat-busy-wheel';\nimport { Observable, of, Subscription, tap } from 'rxjs';\nimport { getEntityCrudConfig } from './default-config';\nimport { SPMatEntityCrudComponentBase } from './mat-entity-crud-internal-types';\nimport { SPMatEntityCrudConfig, SPMatEntityCrudCreateEditBridge } from './mat-entity-crud-types';\n\n@Component({\n imports: [\n CommonModule,\n MatButtonModule,\n MatIconModule,\n TranslocoModule,\n SPMatHostBusyWheelDirective,\n ],\n selector: 'sp-create-edit-entity-host',\n template: `\n <div\n [class]=\"\n 'sp-mat-crud-form-wrapper ' +\n entityCrudComponentBase().getFormPaneWrapperClass()\n \"\n spHostBusyWheel=\"formBusyWheel\"\n *transloco=\"let t\"\n >\n <div\n [class]=\"\n 'sp-mat-crud-form-content ' +\n entityCrudComponentBase().getFormPaneContentClass()\n \"\n >\n <div class=\"create-edit-topbar\">\n <div class=\"title\">\n @if (title()) {\n {{ title() | async }}\n } @else {\n {{\n t('spMatEntityCrud.' + (entity() ? 'editItem' : 'newItem'), {\n item: (this._itemLabel() | async)\n })\n }}\n }\n </div>\n <div class=\"spacer\"></div>\n <div class=\"close\">\n <button mat-icon-button (click)=\"onClose()\">\n <mat-icon>cancel</mat-icon>\n </button>\n </div>\n </div>\n <div class=\"form-container\">\n <ng-container #clientFormContainer></ng-container>\n </div>\n </div>\n </div>\n `,\n styles: `\n .sp-mat-crud-form-wrapper {\n width: 100% !important;\n height: 100% !important;\n }\n .sp-mat-crud-form-content {\n height: 100%;\n width: 100%;\n display: flex;\n flex-direction: column;\n padding: 0.4em;\n }\n .create-edit-topbar {\n display: flex;\n flex-direction: row;\n align-items: center;\n min-height: 48px;\n padding: 0.4em;\n }\n .form-container {\n padding-top: 0.4em;\n flex-grow: 1;\n overflow: auto;\n }\n .create-edit-topbar .title {\n font-size: 1.5em;\n font-weight: 500;\n }\n .create-edit-topbar .spacer {\n flex-grow: 1;\n }\n .create-edit-topbar .close {\n }\n `,\n changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FormViewHostComponent<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> implements SPMatEntityCrudCreateEditBridge, OnInit, OnDestroy\n{\n entityCrudComponentBase =\n input.required<SPMatEntityCrudComponentBase<TEntity, IdKey>>();\n clientViewTemplate = input<TemplateRef<any> | null>(null);\n\n _itemLabel = computed<Observable<string>>(() => {\n const label = this.entityCrudComponentBase().getItemLabel();\n return label instanceof Observable ? label : of(label);\n });\n _itemLabelPlural = computed<Observable<string>>(() => {\n const label = this.entityCrudComponentBase().getItemLabelPlural();\n return label instanceof Observable ? label : of(label);\n });\n\n entity = signal<TEntity | undefined>(undefined);\n title = signal<Observable<string> | undefined>(undefined);\n params = signal<any>(undefined);\n clientFormView!: EmbeddedViewRef<any> | null;\n vc = viewChild('clientFormContainer', { read: ViewContainerRef });\n config!: SPMatEntityCrudConfig;\n sub$ = new Subscription();\n transloco = inject(TranslocoService);\n\n constructor() {\n this.config = getEntityCrudConfig();\n }\n\n ngOnInit() {}\n\n ngOnDestroy(): void {\n this.sub$.unsubscribe();\n }\n\n show(entity: TEntity | undefined, params?: any) {\n this.entity.set(entity);\n if (params && params?.title) {\n this.title.set(\n params.title instanceof Observable ? params.title : of(params.title)\n );\n } else {\n // this.title.set(entity ? this.config.i18n.editItemLabel(this.itemLabel()) : this.config.i18n.newItemLabel(this.itemLabel()));\n // this.title.set(\n // this.transloco.translate(entity ? 'editItem' : 'newItem', {\n // item: this.itemLabel(),\n // })\n // );\n }\n this.params.set(params);\n this.createClientView();\n }\n\n // BEGIN SPMatEntityCrudCreateEditBridge METHODS //\n getEntityName(): string {\n return this.entityCrudComponentBase().getEntityName();\n }\n\n getIdKey(): string {\n return this.entityCrudComponentBase().getIdKey();\n }\n\n getEntityUrl(entityId: any): string {\n return this.entityCrudComponentBase().getEntityUrl(entityId);\n }\n\n close(cancel: boolean) {\n this.entityCrudComponentBase().closeCreateEdit(cancel);\n // destroy the client's form component\n this.destroyClientView();\n }\n\n registerCanCancelEditCallback(callback: () => boolean) {\n this.entityCrudComponentBase().registerCanCancelEditCallback(callback);\n }\n\n create(entityValue: any) {\n // console.log(\n // `SPCreateEditEntityHostComponent.create - entity: ${JSON.stringify(\n // entityValue\n // )}`\n // );\n const crudComponent = this.entityCrudComponentBase();\n return crudComponent\n ?.create(entityValue)\n .pipe(tap(() => this.close(false)));\n }\n\n update(id: any, entityValue: any) {\n // console.log(\n // `SPCreateEditEntityHostComponent.update - id: ${String(\n // id\n // )}, entity: ${entityValue}`\n // );\n const crudComponent = this.entityCrudComponentBase();\n return crudComponent\n ?.update(id, entityValue)\n .pipe(tap(() => this.close(false)));\n }\n\n loadEntity(\n id: string | number,\n params: string | HttpParams\n ): Observable<TEntity> {\n return this.entityCrudComponentBase().loadEntity(id, params);\n }\n // END SPMatEntityCrudCreateEditBridge METHODS //\n\n /**\n * Creates the client view provided via template\n */\n createClientView() {\n /** Render preview component if one was provided */\n const ft = this.clientViewTemplate();\n const vc = this.vc();\n if (ft && vc) {\n this.clientFormView = vc.createEmbeddedView(ft, {\n $implicit: {\n bridge: this,\n entity: this.entity(),\n params: this.params(),\n },\n });\n this.clientFormView.detectChanges();\n }\n }\n\n destroyClientView() {\n if (this.clientFormView) {\n this.clientFormView.destroy();\n this.clientFormView = null;\n }\n }\n\n onClose() {\n // Can we give the client form component a chance to intercept this\n // and cancel the closure?\n if (this.entityCrudComponentBase().canCancelEdit()) {\n this.entityCrudComponentBase().closeCreateEdit(true);\n this.destroyClientView();\n }\n }\n}\n","import {\n ChangeDetectionStrategy,\n Component,\n EmbeddedViewRef,\n input,\n OnDestroy,\n OnInit,\n signal,\n TemplateRef,\n viewChild,\n ViewContainerRef\n} from '@angular/core';\nimport { SPMatEntityCrudComponentBase } from './mat-entity-crud-internal-types';\n\n@Component({\n imports: [],\n selector: 'sp-entity-crud-preview-host',\n template: ` <ng-container #previewComponent></ng-container> `,\n changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PreviewHostComponent<TEntity extends { [P in IdKey]: PropertyKey }, IdKey extends string = 'id'> implements OnInit, OnDestroy {\n vc = viewChild('previewComponent', { read: ViewContainerRef });\n\n entityCrudComponentBase = input.required<SPMatEntityCrudComponentBase<TEntity, IdKey>>();\n clientViewTemplate = input<TemplateRef<any> | null>(null);\n entity = signal<TEntity|undefined>(undefined);\n clientView!: EmbeddedViewRef<any> | null;\n\n constructor() {\n // effect(() => {\n // const tmpl = this.clientViewTemplate();\n // this.createClientView(tmpl);\n // });\n }\n\n ngOnInit(): void {}\n\n ngOnDestroy(): void {}\n\n show(entity: TEntity|undefined, params?: any) {\n this.entity.set(entity);\n // if (params && params?.title) {\n // this.title.set(params.title);\n // } else {\n // this.title.set(entity ? this.config.i18n.editItemLabel(this.itemLabel()) : this.config.i18n.newItemLabel(this.itemLabel()));\n // }\n // this.params.set(params);\n this.createClientView();\n }\n\n close() {\n // this.entityCrudComponentBase().closeCreateEdit(cancel);\n // destroy the client's form component\n this.destroyClientView();\n }\n\n private createClientView() {\n if (this.clientView) {\n // We have only one view in the ng-container. So we might as well\n // call clear() to remove all views contained in it.\n this.vc()!.clear();\n this.clientView.destroy();\n }\n /** Render preview component if one was provided */\n const ft = this.clientViewTemplate();\n const vc = this.vc();\n if (ft && vc) {\n this.clientView = this.vc()!.createEmbeddedView(\n ft,\n {\n $implicit: {\n entity: this.entity(),\n entityCrudComponent: this.entityCrudComponentBase(),\n },\n }\n );\n this.clientView.detectChanges();\n }\n }\n\n\n destroyClientView() {\n if (this.clientView) {\n this.clientView.destroy();\n this.clientView = null;\n }\n }\n\n // close() {\n // this.closePreview.emit();\n // }\n}\n","import { CommonModule } from '@angular/common';\nimport { HttpClient, HttpContext, HttpContextToken, HttpParams } from '@angular/common/http';\nimport {\n AfterViewInit,\n ChangeDetectionStrategy,\n Component,\n computed,\n ContentChildren,\n effect,\n EventEmitter,\n inject,\n Injector,\n input,\n Output,\n QueryList,\n signal,\n TemplateRef,\n viewChild,\n viewChildren\n} from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\nimport { MatSortModule } from '@angular/material/sort';\nimport { MatColumnDef, MatTableModule } from '@angular/material/table';\nimport { RouterModule } from '@angular/router';\nimport {\n showBusyWheelUntilComplete,\n SPMatHostBusyWheelDirective\n} from '@smallpearl/ngx-helper/mat-busy-wheel';\nimport {\n SPMatContextMenuComponent\n} from '@smallpearl/ngx-helper/mat-context-menu';\nimport { SPMatEntityListComponent } from '@smallpearl/ngx-helper/mat-entity-list';\n\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { provideTranslocoScope, TranslocoModule, TranslocoService } from '@jsverse/transloco';\nimport { AngularSplitModule } from 'angular-split';\nimport { clone, startCase } from 'lodash';\nimport { plural } from 'pluralize';\nimport { catchError, EMPTY, firstValueFrom, map, Observable, of, Subscription, switchMap, tap, throwError } from 'rxjs';\nimport { convertHttpContextInputToHttpContext } from './convert-context-input-to-http-context';\nimport { getEntityCrudConfig } from './default-config';\nimport { FormViewHostComponent } from './form-view-host.component';\nimport { SPMatEntityCrudComponentBase } from './mat-entity-crud-internal-types';\nimport { MatEntityCrudItemAction } from './mat-entity-crud-item-action';\nimport {\n ALLOW_ITEM_ACTION_FN,\n CRUD_OP_FN,\n CrudOp,\n NewItemSubType,\n SP_MAT_ENTITY_CRUD_HTTP_CONTEXT,\n SPMatEntityCrudConfig,\n SPMatEntityCrudResponseParser,\n} from './mat-entity-crud-types';\nimport { PreviewHostComponent } from './preview-host.component';\n\n@Component({\n imports: [\n CommonModule,\n RouterModule,\n MatButtonModule,\n MatTableModule,\n MatSortModule,\n MatMenuModule,\n MatSnackBarModule,\n MatIconModule,\n TranslocoModule,\n AngularSplitModule,\n SPMatEntityListComponent,\n SPMatContextMenuComponent,\n FormViewHostComponent,\n SPMatHostBusyWheelDirective,\n PreviewHostComponent,\n ],\n providers: [provideTranslocoScope('sp-mat-entity-crud')],\n selector: 'sp-mat-entity-crud',\n template: `\n <as-split direction=\"horizontal\" [gutterSize]=\"6\" *transloco=\"let t\">\n <as-split-area\n [size]=\"entitiesPaneWidth()\"\n [visible]=\"!entitiesPaneHidden()\"\n >\n <div [class]=\"listPaneWrapperClass()\">\n <ng-content select=\"[breadCrumbs]\"></ng-content>\n\n <ng-template #defaultActionButtons>\n <div class=\"action-bar-actions\">\n @if (!disableCreate()) { @if (newItemSubTypes()) {\n <!-- New {{item}} displays a dropdown menu from which the subtype can be selected -->\n <button\n type=\"button\"\n mat-raised-button\n color=\"primary\"\n [matMenuTriggerFor]=\"newSubTypesMenu\"\n >\n {{\n newItemLabel() ??\n t('spMatEntityCrud.newItem', {\n item: _itemLabel() | async\n })\n }}\n <mat-icon>expand_circle_down</mat-icon>\n </button>\n <mat-menu #newSubTypesMenu=\"matMenu\">\n @for (subtype of newItemSubTypes(); track $index) { @if\n (subtype.role) {\n <button mat-menu-item (click)=\"handleNewItemSubType(subtype)\">\n {{ subtype.label }}\n </button>\n } @else {\n <div style=\"padding: .2em 0.5em;\">\n <strong>{{ subtype.label }}</strong>\n </div>\n } }\n </mat-menu>\n } @else {\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"onCreate($event)\"\n [routerLink]=\"newItemLink()\"\n >\n {{\n newItemLabel() ??\n t('spMatEntityCrud.newItem', {\n item: _itemLabel() | async\n })\n }}\n <mat-icon>add_circle</mat-icon>\n </button>\n } }\n </div>\n </ng-template>\n\n <ng-template #defaultHeaderTemplate>\n <div class=\"action-bar\">\n <div class=\"action-bar-title\">\n {{ _title() | async }}\n </div>\n <span class=\"spacer\"></span>\n <!-- Hide the action buttons when Preview/Edit pane is active -->\n @if (!entityPaneActive()) {\n <ng-container\n [ngTemplateOutlet]=\"actionsTemplate() || defaultActionButtons\"\n ></ng-container>\n }\n </div>\n </ng-template>\n <ng-container\n [ngTemplateOutlet]=\"headerTemplate() || defaultHeaderTemplate\"\n ></ng-container>\n <sp-mat-entity-list\n #entitiesList\n [entityName]=\"entityName()\"\n [entityNamePlural]=\"entityNamePlural()\"\n [deferViewInit]=\"true\"\n [endpoint]=\"endpoint()\"\n [entityLoaderFn]=\"entityLoaderFn()\"\n [columns]=\"columnsWithAction()\"\n [displayedColumns]=\"visibleColumns()\"\n [idKey]=\"idKey()\"\n [pagination]=\"pagination()\"\n [paginator]=\"paginator()\"\n [pageSize]=\"pageSize()\"\n [sorter]=\"sorter()\"\n [disableSort]=\"disableSort()\"\n (selectEntity)=\"handleSelectEntity($event)\"\n [httpReqContext]=\"httpReqContext()\"\n >\n </sp-mat-entity-list>\n </div>\n\n <!--\n We'll be initializing the contentColumnDefs separately and not\n relying on <sp-mat-entity-list>'s internal @ContentChildre() querylist\n for building this. So we define this independenly and not as\n <sp-mat-entity-list> content.\n -->\n <ng-container matColumnDef=\"action\">\n <th mat-header-cell *matHeaderCellDef></th>\n <td mat-cell *matCellDef=\"let element\">\n <!-- <button\n mat-icon-button\n hoverDropDown\n >\n <mat-icon>more_vert</mat-icon>\n </button> -->\n <sp-mat-context-menu\n [menuItems]=\"_itemActions\"\n (selected)=\"onItemAction($event, element)\"\n [contextData]=\"element\"\n [hasBackdrop]=\"true\"\n ></sp-mat-context-menu>\n </td>\n </ng-container>\n </as-split-area>\n <as-split-area [size]=\"entityPaneWidth()\" [visible]=\"entityPaneActive()\">\n <div\n [class]=\"'entity-pane-wrapper ' + previewPaneWrapperClass()\"\n spHostBusyWheel=\"formBusyWheel\"\n >\n <sp-entity-crud-preview-host\n [ngClass]=\"createEditViewActive() ? 'sp-hidden' : 'sp-visible'\"\n [entityCrudComponentBase]=\"this\"\n [clientViewTemplate]=\"previewTemplate()!\"\n ></sp-entity-crud-preview-host>\n <!-- Create/Edit Entity -->\n <sp-create-edit-entity-host\n [ngClass]=\"createEditViewActive() ? 'sp-visible' : 'sp-hidden'\"\n [entityCrudComponentBase]=\"this\"\n [clientViewTemplate]=\"createEditFormTemplate()\"\n ></sp-create-edit-entity-host>\n </div>\n </as-split-area>\n </as-split>\n `,\n styles: `\n .sp-hidden {\n display: none;\n }\n .sp-visible {\n display: inherit;\n height: 100% !important;\n width: 100% !important;\n }\n .entity-pane-wrapper {\n height: 100% !important;\n width: 100% !important;\n }\n .action-bar {\n display: flex;\n flex-direction: row;\n align-items: center;\n padding-bottom: 0.5em;\n }\n .action-bar-title {\n font-size: 1.5em;\n font-weight: 600;\n }\n .spacer {\n flex-grow: 1;\n }\n .action-bar-actions {\n text-align: end;\n }\n .active-row {\n font-weight: bold;\n }\n `,\n changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SPMatEntityCrudComponent<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n >\n extends SPMatEntityListComponent<TEntity, IdKey>\n implements SPMatEntityCrudComponentBase<TEntity, IdKey>, AfterViewInit\n{\n // entityName = input.required<string>();\n // entityNamePlural = input<string>();\n\n itemLabel = input<string | Observable<string>>();\n itemLabelPlural = input<string | Observable<string>>();\n\n /**\n * Title string displayed above the component. If not specified, will use\n * itemLabelPlural() as the title.\n */\n title = input<string | Observable<string>>();\n /**\n *\n */\n itemActions = input<MatEntityCrudItemAction<TEntity, IdKey>[]>([]);\n /**\n * Specify the list of router paths that will be set as the value for\n * [routerLink] for the 'New {{ item }}' button. If not specified,\n * if createEditTemplate is specified, it will be shown. If not, `action`\n * out event will be raised with `{ role: '_new_' }`.\n */\n newItemLink = input<string | string[]>();\n /**\n * If not specified, will default to 'New <itemLabel()>'.\n */\n newItemLabel = input<string | string[]>();\n /**\n * Text for the Edit <item> pane title\n */\n editItemTitle = input<string>();\n /**\n * If you want \"New {{item}}\" button to support multiple entity types,\n * you can set this to `NewItemSubType[]`, where each element stands for for\n * a dropdown menu item. Refer to `NewItemSubType` for details on this\n * interface.\n */\n newItemSubTypes = input<NewItemSubType[]>();\n /**\n * If you want to take control of the network operations for the CRUD\n * operations (GET/CREATE/UPDATE/DELETE), provide a value for this property.\n */\n crudOpFn = input<CRUD_OP_FN<TEntity, IdKey>>();\n /**\n * Item preview template.\n */\n previewTemplate = input<TemplateRef<any>>();\n /**\n * Whether to allow a context menu action or not. Return false to disable\n * the action.\n */\n allowEntityActionFn = input<ALLOW_ITEM_ACTION_FN<TEntity>>();\n /**\n * A template that allows the header to be replaced. Usage:-\n *\n * ```\n * <sp-map-entity-crud\n * [headerTemplate]=\"myCrudViewHeader\"\n * ></sp-map-entity-crud>\n * <ng-template #myCrudViewHeader>...</ng-template>\n * ```\n */\n headerTemplate = input<TemplateRef<any>>();\n /**\n * Set this to the custom template identifier that will replace the\n * \"New {{Item}}\" button portion. This template will expand towards the\n * title which will be placed to its left (right in rtl).\n *\n * ```\n * <sp-map-entity-crud\n * [actionsTemplate]=\"myCrudActions\"\n * ></sp-map-entity-crud>\n * <ng-template #myCrudActions>...</ng-template>\n * ```\n */\n actionsTemplate = input<TemplateRef<any>>();\n\n /**\n * CRUD action response parser. This will be called with the response\n * from CREATE & UPDATE operations to parse the response JSON and return\n * the created/updated TEntity.\n */\n crudResponseParser = input<SPMatEntityCrudResponseParser>();\n /**\n * An ng-template name that contains the component which provides the\n * create/edit CRUD action.\n *\n * ```\n * <ng-template #createEdit let-data>\n * <app-create-edit-entity-demo [bridge]=\"data.bridge\" [entity]=\"data.entity\"></app-create-edit-entity-demo>\n * </ng-template>\n * ```\n * Note how [bridge] & [entity] properties are set deriving them from the\n * implicit template context. [entity] will be undefined for Create\n * opreation and will be the valid entity for an Update.\n * (app-create-edit-entity-demo here is the client code that implements the\n * Create/Edit form)\n */\n createEditFormTemplate = input<TemplateRef<any> | null>(null);\n /**\n * Disables the per item actions column, preventing 'Edit' & 'Delete'\n * (and other user defined) item operations.\n */\n disableItemActions = input<boolean>(false);\n /**\n * Disables the Create function.\n */\n disableCreate = input<boolean>(false);\n /**\n * View refresh policy after a CREATE/UPDATE operation. Values\n * 'none' - Objects are not refreshed after an edit operation. The return\n * value of the edit operation is used as the object to\n * add/update the component's internal store. This is the default.\n * 'object' - Refresh just the object that was returned from the\n * CREATE/UPDATE operation. Use this if the JSON object returned\n * after a successful CREATE/UPDATE op differs from the JSON\n * object returned for the GET request.\n * 'all' - Refresh the entire list after a CREATE/UPDATE operation. This\n * mimics the behaviour of legacy HTML apps with pure server\n * defined UI.\n */\n refreshAfterEdit = input<'none' | 'object' | 'all'>('none');\n /**\n * HttpContext for crud requests - `create`, `retrieve`, `update` & `delete`.\n * Note that HttpContext for `list` operation should be set using the\n * `httpReqContext` property inherited from SPMatEntityListComponent.\n * The value of the property is an object where the key names reflect the CRUD\n * operation with each of these keys expected to have a value of type\n * `[[HttpContextToken<any>, any]] | [HttpContextToken<any>, any] | HttpContext`.\n *\n * Alternatively the property value can be set to\n * `[[HttpContextToken<any>, any]] | [HttpContextToken<any>, any] | HttpContext`,\n * in which case the same context would be used for all HTTP requests.\n */\n crudHttpReqContext = input<\n | [[HttpContextToken<any>, any]]\n | [HttpContextToken<any>, any]\n | {\n create?: [[HttpContextToken<any>, any]] | [HttpContextToken<any>, any]; // CREATE\n retrieve?:\n | [[HttpContextToken<any>, any]]\n | [HttpContextToken<any>, any]; // RETRIEVE\n update?: [[HttpContextToken<any>, any]] | [HttpContextToken<any>, any]; // UPDATE\n delete?: [[HttpContextToken<any>, any]] | [HttpContextToken<any>, any]; // DELETE\n }\n >();\n /**\n * Width of the edit pane as a percentange of the overall <as-split> width.\n */\n editPaneWidth = input<number>(100);\n /**\n * Width of the preview pane as a percentange of the overall <as-split> width.\n */\n previewPaneWidth = input<number>(50);\n\n /**\n * The class class that will be applied to the list pane wrapper.\n */\n listPaneWrapperClass = input<string>('sp-mat-crud-list-pane-wrapper-class');\n\n /**\n * The class class that will be applied to the preview pane wrapper.\n * Inside this wrapper, another div element with class='previewPaneContentClass()'\n * will be created to host the preview content. The client supplied preview\n * template will be rendered inside this div.\n *\n * +----------------------------------------+\n * | previewPaneWrapperClass |\n * |----------------------------------------|\n * | preview toolbar |\n * |----------------------------------------|\n * | +------------------------------------+ |\n * | | div class=previewPaneContentClass()| |\n * | |------------------------------------| |\n * | | client supplied preview template | |\n * | +------------------------------------+ |\n * +----------------------------------------+\n */\n previewPaneWrapperClass = input<string>(\n 'sp-mat-crud-entity-pane-wrapper-class'\n );\n\n /**\n * The class class that will be applied to the preview pane content.\n */\n previewPaneContentClass = input<string>(\n 'sp-mat-crud-preview-pane-content-class'\n );\n\n /**\n * The class class that will be applied to the form pane wrapper. Inside this\n * wrapper, another div element with class='formPaneContentClass()' will be\n * created to host the form content. The client supplied form template will\n * be rendered inside this div.\n *\n * +----------------------------------------+\n * | formPaneWrapperClass |\n * | +------------------------------------+ |\n * | | div class=formPaneContentClass() | |\n * | |------------------------------------| |\n * | | form title + close button | |\n * | |------------------------------------| |\n * | | client supplied form template | |\n * | +------------------------------------+ |\n * +----------------------------------------+\n */\n formPaneWrapperClass = input<string>('sp-mat-crud-form-pane-wrapper-class');\n\n /**\n * The CSS class that will be applied to the form pane content.\n */\n formPaneContentClass = input<string>('sp-mat-crud-form-pane-content-class');\n\n // INTERNAL PROPERTIES //\n // Derive a label from a camelCase source string. If the camelCase string\n // can be translated, it returns the translated string. If not, the function\n // converts the camelCase to 'Title Case' and returns it.\n private getLabel = (source: string) => {\n const label = this.transloco.translate(source);\n if (label.localeCompare(source) !== 0) {\n // Successful translation, return it\n return label;\n }\n return startCase(source);\n };\n\n _itemLabel = computed<Observable<string>>(() => {\n const itemLabel = this.itemLabel();\n const label = itemLabel ? itemLabel : this.getLabel(this.entityName());\n return label instanceof Observable ? label : of(label);\n });\n _itemLabelPlural = computed<Observable<string>>(() => {\n const itemLabelPlural = this.itemLabelPlural();\n const label = itemLabelPlural\n ? itemLabelPlural\n : this.getLabel(plural(this.entityName()));\n return label instanceof Observable ? label : of(label);\n });\n\n // Computed title\n _title = computed(() => {\n const title = this.title() ? this.title() : this._itemLabelPlural();\n return title instanceof Observable ? title : of(title);\n });\n // endpoint with the QP string removed (if one was provided)\n _endpointSansParams = computed(() => this.endpoint().split('?')[0]);\n _endpointParams = computed(() => {});\n componentColumns = viewChildren(MatColumnDef);\n @ContentChildren(MatColumnDef) _clientColumnDefs!: QueryList<MatColumnDef>;\n\n /**\n * Event raised for user selecting an item action. It's also raised\n * for 'New <Item>' action, if 'newItemLink' property is not set.\n */\n @Output() action = new EventEmitter<{ role: string; entity?: TEntity }>();\n\n /**\n * Event raised when create Create/Edit pane is activated & deactivated.\n * Event contains two flags:-\n * activated - whether the createEdit form view was activated or\n * deactivated.\n * cancelled - whether the form view was cancelled by user. False for this\n * indicates that the form view was closed after a successful\n * edit operation.\n */\n @Output() entityViewPaneActivated = new EventEmitter<{\n activated: boolean;\n cancelled: boolean | undefined;\n mode: 'edit' | 'preview';\n }>();\n\n busyWheelId = `entityCrudBusyWheel-${Date.now()}`;\n sub$ = new Subscription();\n spEntitiesList =\n viewChild<SPMatEntityListComponent<TEntity, IdKey>>('entitiesList');\n\n // Theoritically, we ought to be able to initialize the mat-entities-list\n // columns from ngAfterViewInit lifecycle hook. But for some strange reason\n // when this hook is called, sp-mat-entities-list has not been initialized.\n // Therefore `spEntitiesList()` is null. So we have to use a computed signal,\n // which will be triggered when spEntitiesList() is initialized and use that\n // to initialize the columns.\n spEntitiesListInited = effect(() => {\n if (this.spEntitiesList()) {\n this._initEntitiesList();\n }\n });\n\n crudConfig!: SPMatEntityCrudConfig;\n\n // This is the internal component that will host the createEditFormTemplate\n createEditHostComponent = viewChild(FormViewHostComponent);\n // A flag to toggle the viewport's contents between the mat-table\n // and the create/edit form template.\n createEditViewActive = signal<boolean>(false);\n\n // Whether it's okay to cancel the edit\n canCancelEditCallback!: () => boolean;\n\n // Preview host component\n previewHostComponent = viewChild(PreviewHostComponent);\n previewActive = computed(() => this.previewedEntity() !== undefined);\n previewedEntity = signal<TEntity | undefined>(undefined);\n\n // Whether the pane that hosts the preview/edit-entity template is active.\n // We call it entityPane as it's used to either render a selected entity\n // or to edit one.\n entityPaneActive = computed(\n () => !!this.previewedEntity() || this.createEditViewActive()\n );\n // Effective width of the entity pane.\n entityPaneWidth = computed(() =>\n this.entityPaneActive()\n ? !!this.previewedEntity()\n ? this.previewPaneWidth()\n : this.editPaneWidth()\n : 0\n );\n\n // Width of the pane showing the list of entities. Calculated as\n entitiesPaneWidth = computed(() => 100 - this.entityPaneWidth());\n entitiesPaneHidden = computed(\n () => this.entityPaneActive() && this.entityPaneWidth() === 100\n );\n\n defaultItemCrudActions = signal<MatEntityCrudItemAction<TEntity, IdKey>[]>(\n []\n );\n columnsWithAction = computed(() => {\n const cols = clone(this.columns());\n const actionDefined =\n cols.find((c) =>\n typeof c === 'string' ? c === 'action' : c.name === 'action'\n ) !== undefined;\n if (!actionDefined) {\n cols.push('action');\n }\n return cols;\n });\n // Provide per entity actions as a function so that the actions are\n // enumerated only when the user clicks on the context menu button.\n _itemActions = (entity: TEntity) => this.getItemActions(entity);\n\n // This uses the previewActive signal to compute the visible columns\n // when preview is activated. For now we just hide the 'action' column when\n // preview is active. We can further customize this logic by allowing the\n // client to specify the columns to display when preview is active thereby\n // reducing column clutter when the table width becomes narrower owing to\n // preview pane taking up screen space.\n visibleColumns = computed(() =>\n this.entityPaneActive()\n ? this.columnsWithAction()\n .map((col) => (typeof col === 'string' ? col : col.name))\n .filter((name) => name !== 'action')\n : []\n );\n transloco = inject(TranslocoService);\n\n constructor(\n http: HttpClient,\n private snackBar: MatSnackBar,\n sanitizer: DomSanitizer,\n injector: Injector\n ) {\n super(http, sanitizer, injector);\n this.crudConfig = getEntityCrudConfig();\n if (this.crudConfig?.defaultItemActions) {\n this.defaultItemCrudActions.set(this.crudConfig?.defaultItemActions);\n } else {\n this.defaultItemCrudActions.set([\n {\n label: this.transloco.translate('spMatEntityCrud.edit'),\n role: '_update_',\n },\n {\n label: this.transloco.translate('spMatEntityCrud.delete'),\n role: '_delete_',\n },\n ]);\n }\n }\n\n override ngOnInit() {}\n\n override ngOnDestroy(): void {\n this.sub$.unsubscribe();\n }\n\n /**\n * Override so that we can suppress default action in SPMatEntityListComponent\n */\n override ngAfterViewInit(): void {}\n\n /**\n * If the create/edit entity form is active, it calls its registered\n * canCancelEdit callback to determine if it's okay to cancel the edit.\n * You can use this method from the host component's router guard to\n * ensure that any changes made to the form are not accidentally lost by\n * navigating away from the CRUD page.\n *\n * If your CRUD page has multiple sp-mat-entity-crud components, you have to\n * implement the logic to call this method on the appropriate component.\n *\n * If the the create/edit form is not active, this method returns true.\n * @returns\n */\n canDeactivate() {\n if (this.createEditViewActive()) {\n return this.canCancelEdit();\n }\n return true;\n }\n\n override refresh(force = false) {\n this.spEntitiesList()?.refresh(force);\n }\n\n // BEGIN SPMatEntityCrudComponentBase METHODS //\n getEntityName(): string {\n return this.entityName()\n }\n\n getEntityNamePlural(): string {\n return this._entityNamePlural();\n }\n\n getIdKey() {\n return this.idKey();\n }\n\n closeCreateEdit(cancelled: boolean) {\n this.createEditViewActive.set(false);\n this.entityViewPaneActivated.emit({\n activated: false,\n cancelled: !!cancelled,\n mode: 'edit',\n });\n }\n\n canCancelEdit() {\n if (this.canCancelEditCallback) {\n return this.canCancelEditCallback();\n }\n return true;\n }\n\n registerCanCancelEditCallback(callback: () => boolean) {\n this.canCancelEditCallback = callback;\n }\n\n triggerEntityUpdate(entity: TEntity) {\n this.onUpdate(entity);\n }\n\n triggerEntityDelete(entity: TEntity) {\n this.onDelete(entity);\n }\n\n create(entityValue: any) {\n let obs!: Observable<TEntity | null>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn('create', undefined, entityValue, this);\n } else {\n obs = this.http.post<TEntity>(this.getUrl(this.endpoint()), entityValue, {\n context: this.getCrudReqHttpContext('create'),\n });\n }\n\n return obs.pipe(\n showBusyWheelUntilComplete('formBusyWheel'),\n switchMap((resp) =>\n resp ? this.doRefreshAfterEdit(resp, 'create') : of(null)\n ),\n tap((entity) => {\n // If pagination is infinite or if the pagination if none or if the\n // count of items in the current page is less than pageSize()\n // wec an safely add the item to the list, which will cause the view\n // render the new item in the mat-table.\n if (entity) {\n this.spEntitiesList()?.addEntity(entity);\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n this.snackBar.open(\n this.transloco.translate('spMatEntityCrud.createSuccess', {\n item: itemLabel,\n })\n );\n });\n }\n })\n );\n }\n\n update(id: TEntity[IdKey], entityValue: any) {\n let obs!: Observable<TEntity | null>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn('update', id, entityValue, this);\n } else {\n obs = this.http.patch<TEntity>(this.getEntityUrl(id), entityValue, {\n context: this.getCrudReqHttpContext('update'),\n });\n }\n\n return obs.pipe(\n showBusyWheelUntilComplete('formBusyWheel'),\n switchMap((resp) =>\n resp ? this.doRefreshAfterEdit(resp, 'update') : of(null)\n ),\n tap((entity) => {\n if (entity) {\n this.spEntitiesList()?.updateEntity(id, entity);\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n this.snackBar.open(\n this.transloco.translate('spMatEntityCrud.updateSuccess', {\n item: itemLabel,\n })\n );\n });\n }\n })\n );\n }\n\n loadEntity(\n id: TEntity[IdKey],\n params: string | HttpParams\n ): Observable<TEntity> {\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n return crudOpFn('get', id, undefined, this) as Observable<TEntity>;\n } else {\n const httpParams =\n params instanceof HttpParams\n ? params\n : new HttpParams({ fromString: params });\n return this.http.get<TEntity>(this.getEntityUrl(id), {\n context: this.getCrudReqHttpContext('retrieve'),\n params: httpParams,\n });\n }\n }\n // END SPMatEntityCrudComponentBase METHODS //\n\n /**\n * Thunk these methods to the internal <sp-mat-entity-list> component.\n */\n override addEntity(entity: TEntity): void {\n this.spEntitiesList()?.addEntity(entity);\n }\n\n override removeEntity(id: TEntity[IdKey]): void {\n this.spEntitiesList()?.removeEntity(id);\n }\n\n override updateEntity(id: TEntity[IdKey], entity: TEntity): void {\n this.spEntitiesList()?.updateEntity(id, entity);\n }\n\n /**\n * Refresh the entity list, after a CRUD CREATE or UPDATE operation.\n * @param resp This is the response from the CRUD operation (CREATE/UPDATE).\n * @param method The CRUD operation post which REFRESH is requested.\n * @returns Observable<TEntity|null>\n */\n doRefreshAfterEdit(resp: any, method: 'create' | 'update') {\n const refreshAfterEdit = this.refreshAfterEdit();\n const entity = this.getCrudOpResponseParser()(\n this.entityName(),\n this.idKey(),\n method,\n resp\n );\n if (refreshAfterEdit === 'object') {\n let obs!: Observable<TEntity>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn(\n 'get',\n (entity as any)[this.idKey()],\n undefined,\n this\n ) as Observable<TEntity>;\n } else {\n obs = this.http.get<TEntity>(\n this.getEntityUrl((entity as any)[this.idKey()]),\n { context: this.getCrudReqHttpContext('retrieve') }\n );\n }\n return obs.pipe(\n map((entity) => {\n return this.getCrudOpResponseParser()(\n this.entityName(),\n this.idKey(),\n 'retrieve',\n entity\n );\n })\n );\n } else if (refreshAfterEdit === 'all') {\n this.refresh(true);\n return of(null);\n }\n\n return of(entity);\n }\n\n getCrudOpResponseParser(): SPMatEntityCrudResponseParser {\n if (this.crudResponseParser()) {\n // Without the `as SPMatEntityCrudResponseParser`, TSC will complain.\n return this.crudResponseParser() as SPMatEntityCrudResponseParser;\n }\n // crudConfig will have a parser as our default config provides one.\n return this.crudConfig\n .crudOpResponseParser as SPMatEntityCrudResponseParser;\n }\n\n // SPMatEntityCrudComponentBase interface method. Thunk to the implementation\n // method closePreviewImpl().\n closePreview() {\n this.closePreviewImpl(true);\n }\n\n private closePreviewImpl(toggleEntityListActiveEntity: boolean) {\n if (this.previewedEntity()) {\n if (toggleEntityListActiveEntity) {\n this.spEntitiesList()?.toggleActiveEntity(this.previewedEntity());\n }\n this.previewedEntity.set(undefined);\n this.entityViewPaneActivated.emit({\n activated: false,\n cancelled: undefined,\n mode: 'preview',\n });\n }\n }\n\n onItemAction(role: string, entity: TEntity) {\n // Handle default roles, which always get default behavior. If you want\n // a custom behavior, change the role to a custom one.\n if (role === '_update_' || role === '_delete_') {\n role === '_update_' ? this.onUpdate(entity) : this.onDelete(entity);\n return;\n }\n\n // Find the SPMatEntityCrudItemAction for this role to see if it's\n // allowed.\n const actionItem = this.itemActions().find((a) => a.role === role);\n if (!actionItem) {\n return;\n }\n\n // If a custom action handler is specified, call it.\n if (actionItem?.action) {\n actionItem.action(entity);\n return;\n }\n\n // Perform custom HTTP action\n const httpRequestParameters: MatEntityCrudItemAction<\n TEntity,\n IdKey\n >['httpRequestParameters'] = actionItem?.httpRequestParameters || {\n method: 'POST',\n urlPath: actionItem.role,\n };\n const verb = httpRequestParameters?.urlPath || actionItem?.role;\n if (!verb) {\n return;\n }\n\n // If a confirm prompt is specified, display it and return if user\n // selects 'Cancel'.\n if (actionItem?.confirmPrompt) {\n if (!confirm(actionItem.confirmPrompt)) {\n return;\n }\n }\n\n this.doEntityAction(\n (entity as any)[this.idKey()],\n verb,\n httpRequestParameters.params || new HttpParams(),\n httpRequestParameters.body\n )\n .pipe(\n tap((response) => {\n const successMessage =\n actionItem?.successMessage ||\n this.transloco.translate('spMatEntityCrud.done');\n this.snackBar.open(successMessage || 'Done');\n }),\n catchError((error) => {\n /**\n * If an errorMessage is specified in the actionItem, display it.\n * Otherwise rethrow the error so that it can be handled by the\n * global error handler.\n */\n if (actionItem?.errorMessage) {\n this.snackBar.open(actionItem.errorMessage);\n return EMPTY;\n }\n return throwError(() => error);\n })\n )\n .subscribe();\n }\n\n onCreate(event: Event) {\n // If newItemLink() has not been provided, check if createEditFormTemplate\n // is specified. If it is, load it and make that cover the entire viewport.\n // If that too is not specified, emit an action event with role='_new_'.\n if (!this.newItemLink() || this.newItemLink()?.length == 0) {\n event.preventDefault();\n event.stopImmediatePropagation();\n this.showCreateEditView(undefined);\n if (!this.createEditViewActive()) {\n this.action.emit({ role: '_new_' });\n }\n }\n }\n\n onUpdate(entity: TEntity) {\n this.showCreateEditView(entity);\n if (!this.createEditViewActive()) {\n this.action.emit({ role: '_update_' });\n }\n }\n\n /**\n * Show the create/edit component. This is deliberately made public so as to\n * be callable from the client. This allows the client to dynamically\n * set the form edit template and then show the edit pane by calling this\n * method.\n * @param entity\n * @param params\n */\n showCreateEditView(entity?: TEntity | undefined, params?: any) {\n const tmpl = this.createEditFormTemplate();\n if (!this.createEditViewActive() && tmpl) {\n // If preview is active deactivate it\n if (this.previewActive()) {\n this.closePreviewImpl(true);\n }\n const createEditHost = this.createEditHostComponent();\n createEditHost!.show(entity, params);\n this.createEditViewActive.set(true);\n this.entityViewPaneActivated.emit({\n activated: true,\n cancelled: undefined,\n mode: 'edit',\n });\n }\n }\n\n showPreviewView(entity?: TEntity, params?: any) {\n const tmpl = this.previewTemplate();\n if (tmpl) {\n if (!this.createEditViewActive()) {\n const previewHost = this.previewHostComponent();\n this.previewedEntity.set(entity);\n previewHost?.show(entity, params);\n this.entityViewPaneActivated.emit({\n activated: true,\n cancelled: undefined,\n mode: 'preview',\n });\n // this.previewActivated.emit({ entity, activated: true });\n }\n }\n }\n\n hidePreviewView() {\n if (this.previewActive()) {\n const previewHost = this.previewHostComponent();\n previewHost?.close();\n this.closePreviewImpl(false);\n }\n }\n\n async onDelete(entity: TEntity) {\n // Do the delete prompt asynchronously so that the context menu is\n // dismissed before the prompt is displayed.\n setTimeout(() => {\n // We use firstValueFrom() to get the value of the observable\n // synchronously. firstValueFrom() also gracefully cleans up the\n // observable after a value is emitted.\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n const deletedItemPrompt = this.transloco.translate(\n 'spMatEntityCrud.deleteItemConfirm',\n { item: itemLabel.toLocaleLowerCase() }\n );\n const yes = confirm(deletedItemPrompt);\n if (yes) {\n const entityId = (entity as any)[this.idKey()];\n\n // If preview is active deactivate it\n if (this.previewActive()) {\n this.closePreviewImpl(false);\n }\n\n let obs!: Observable<any>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn('delete', entityId, undefined, this);\n } else {\n obs = this.http.delete<void>(this.getEntityUrl(entityId), {\n context: this.getCrudReqHttpContext('delete'),\n });\n }\n\n this.sub$.add(\n obs\n .pipe(\n // TODO: how to display a busy wheel?\n // showBusyWheelUntilComplete(this.busyWheelId),\n tap(() => {\n this.spEntitiesList()!.removeEntity(entityId);\n // TODO: customize by providing an interface via SPMatEntityCrudConfig?\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n const deletedMessage = this.transloco.translate(\n 'spMatEntityCrud.deleteItemSuccess',\n { item: itemLabel }\n );\n this.snackBar.open(deletedMessage);\n });\n })\n )\n .subscribe()\n );\n }\n });\n });\n }\n\n override getUrl(endpoint: string) {\n return this.entityListConfig?.urlResolver\n ? this.entityListConfig?.urlResolver(endpoint)\n : endpoint;\n }\n\n getEntityUrl(entityId: TEntity[IdKey]) {\n const endpoint = this.endpoint();\n const endpointParts = endpoint.split('?');\n const entityEndpoint =\n (endpointParts[0].endsWith('/')\n ? endpointParts[0]\n : endpointParts[0] + '/') + `${String(entityId)}/`;\n if (endpointParts.length > 1) {\n return this.getUrl(entityEndpoint + `?${endpointParts[1]}`);\n }\n return this.getUrl(entityEndpoint);\n }\n\n getEntityActionUrl(entityId: TEntity[IdKey], action: string) {\n const url = this.getEntityUrl(entityId);\n const urlParts = url.split('?');\n if (action.endsWith('/')) {\n action = action.slice(0, -1); // We'll be adding the trailing slash\n }\n const actionUrl =\n (urlParts[0].endsWith('/') ? urlParts[0] : urlParts[0] + '/') +\n `${action}/`;\n return urlParts.length === 1 ? actionUrl : actionUrl + `?${urlParts[1]}`;\n }\n\n handleSelectEntity(entity: TEntity | undefined) {\n if (!this.createEditViewActive()) {\n if (this.previewTemplate()) {\n entity ? this.showPreviewView(entity) : this.hidePreviewView();\n // this.previewedEntity.set(entity);\n } else {\n // If 'previewTemplate' is not set, propagate the event to client.\n this.selectEntity.emit(entity);\n }\n }\n }\n\n handleNewItemSubType(subtype: NewItemSubType) {\n // console.log(`handleNewItemSubType: ${subtype}`);\n if (subtype.role === '_new_') {\n this.showCreateEditView(undefined, subtype?.params);\n } else {\n this.action.emit({ role: subtype.role });\n }\n }\n\n private getCrudReqHttpContext(op: CrudOp) {\n\n let context = new HttpContext();\n // HttpContext for crud operations are taken from either the global httpReqContext\n // or from the crudHttpReqContext, with the latter taking precedence.\n const crudHttpReqContext = this.crudHttpReqContext()\n ? this.crudHttpReqContext()\n : this.httpReqContext();\n if (crudHttpReqContext) {\n if (crudHttpReqContext instanceof HttpContext) {\n // crudHttpReqContext is an object of type HttpContext. Which means\n // the same context is to be applied to all crud requests.\n for (const k of crudHttpReqContext.keys()) {\n context.set(k, crudHttpReqContext.get(k));\n }\n } else {\n if (Array.isArray(crudHttpReqContext)) {\n // Same HttpContext for all crud requests. Being an array, it must\n // be an array of HttpContextToken key, value pairs.\n convertHttpContextInputToHttpContext(context, crudHttpReqContext);\n } else if (\n typeof crudHttpReqContext === 'object' &&\n op &&\n Object.keys(crudHttpReqContext).find((k) => k === op)\n ) {\n // HttpContext specific to this crud operation, 'create'|'retrieve'|'update'|'delete'\n convertHttpContextInputToHttpContext(\n context,\n crudHttpReqContext[op]!\n );\n }\n }\n }\n\n // Add standard SP_MAT_ENTITY_CRUD_HTTP_CONTEXT info which is set for all\n // HTTP requests made by this component.\n context.set(SP_MAT_ENTITY_CRUD_HTTP_CONTEXT, {\n entityName: this.entityName(),\n entityNamePlural: this._entityNamePlural(),\n endpoint: this.endpoint(),\n op,\n });\n return context;\n }\n\n isItemActionAllowed(action: string, entity: TEntity) {\n return false;\n }\n\n /**\n * Returns the list of item actions. Calls 'allowItemActionFn' for each action\n * to determine if the action is allowed for the given entity.\n * @returns\n */\n getItemActions(entity: TEntity): MatEntityCrudItemAction<TEntity, IdKey>[] {\n // console.log(`SPMatEntityCrudComponent.getItemActions - entity: ${JSON.stringify(entity, null, 2)}`);\n const actions =\n this.itemActions() && this.itemActions().length\n ? this.itemActions()\n : this.defaultItemCrudActions();\n let actionsCopy: MatEntityCrudItemAction<TEntity, IdKey>[] = clone(actions);\n actionsCopy.forEach(\n (action: MatEntityCrudItemAction<TEntity, IdKey>, index: number) => {\n // localize default action item labels (Update & Delete)\n // Client specified action labels are to be localized by the client\n // before supplying them to the component.\n if (action.label.startsWith('spMatEntityCrud.')) {\n action.label = this.transloco.translate(action.label);\n }\n const orgDisable = actions[index]?.disable;\n action.disable = (entity: TEntity) => {\n if (orgDisable) {\n return orgDisable(entity);\n }\n const allowItemActionFn = this.allowEntityActionFn();\n if (allowItemActionFn) {\n return !allowItemActionFn(entity, action.role ?? action.label);\n }\n return false;\n };\n }\n );\n // If the item actions are disabled, disable all actions. Event user\n // defined actions.\n if (this.disableItemActions()) {\n actionsCopy.forEach((a) => (a.disable = () => true));\n }\n return actionsCopy;\n }\n\n getPreviewPaneContentClass() {\n return this.previewPaneContentClass();\n }\n\n getFormPaneWrapperClass() {\n return this.formPaneWrapperClass();\n }\n\n getFormPaneContentClass(): string {\n return this.formPaneContentClass();\n }\n\n getItemLabel(): string | Observable<string> {\n return this._itemLabel();\n }\n\n getItemLabelPlural(): string | Observable<string> {\n return this._itemLabelPlural();\n }\n\n /**\n * Perform an action on the entity with the given id. The endpoint for the\n * action is derived from the `verb` argument which is appended to the\n * entity URL. This is following DRF specification where ViewSets can be\n * extended with custom actions that are not part of the standard\n * CRUD operations. Such methods will have a URL like\n * `/api/v1/entity/<id>/<verb>/` where `<verb>` is the custom action verb.\n * @param id\n * @param verb\n * @param addlParams\n * @param data\n */\n doEntityAction(\n id: TEntity[IdKey],\n verb: string,\n addlParams: HttpParams,\n data: any,\n busyWheelName: string = 'formBusyWheel'\n ) {\n let obs!: Observable<TEntity | null>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn(verb, id, data, this);\n } else {\n const url = this.getEntityActionUrl(id, verb);\n obs = this.http.post<TEntity>(url, data, {\n params: addlParams || {},\n context: this.getCrudReqHttpContext('update'), // KLUDGE!: use 'update' request context\n });\n }\n\n return obs.pipe(\n showBusyWheelUntilComplete(busyWheelName),\n switchMap((resp) =>\n resp ? this.doRefreshAfterEdit(resp, 'update') : of(null)\n ),\n tap((entity) => {\n if (entity) {\n this.spEntitiesList()?.updateEntity(id, entity);\n }\n })\n );\n }\n\n /**\n * Initialize the columns for the mat-entities-list component. This is\n * called when the <sp-mat-entities-list> component has been properly\n * initialized.\n */\n private _initEntitiesList() {\n const spEntitiesList = this.spEntitiesList();\n if (spEntitiesList) {\n // Build contentColumnDefs using our component's content. Then add our own\n // 'action' column definition to it. Then set this as the value of\n // child SPMatEntityListComponent.contentColumnDef. This way we force\n // SPMatEntityListComponent to use our component's any project MatColumnDef\n // content in the final mat-table.\n const clientColumnDefs = this.clientColumnDefs;\n let contentColumnDefs = new Array<MatColumnDef>();\n if (clientColumnDefs.length) {\n // Note that we process any content projected matColumnDef first and\n // our own internal content later. And when we process our own internal\n // columns (for now only 'action'), it's not added if a column with that\n // name has already been defined via content projection. This allows the\n // clients to override even internal columns with their column defintion.\n clientColumnDefs.toArray().forEach((c) => contentColumnDefs.push(c));\n }\n this.componentColumns().forEach((ic) => {\n if (!contentColumnDefs.find((c) => c.name === ic.name)) {\n contentColumnDefs.push(ic);\n }\n });\n spEntitiesList.contentColumnDefs = contentColumnDefs;\n // This is a replication of SPMatEntityCrudList.ngAfterViewInit. That\n // code is skipped as we declare <sp-mat-entity-list> with\n // deferViewInit=true.\n spEntitiesList.buildColumns();\n spEntitiesList.setupSort();\n spEntitiesList.loadMoreEntities();\n }\n }\n}\n","import { HttpParams } from \"@angular/common/http\";\nimport { SPContextMenuItem } from \"@smallpearl/ngx-helper/mat-context-menu\";\nimport { Observable } from \"rxjs\";\n\nexport const ITEM_ACTION_UPDATE = '_update_';\nexport const ITEM_ACTION_DELETE = '_delete_';\n\n/**\n * SPMatEntityCrudCreateEditBridge implementer uses this interface to\n * communicate with the parent SPMatEntityCreateComponent. The bridge\n * component would use the hideCreateEdit() to close itself, when user cancels\n * the create/edit operation.\n */\nexport interface SPMatEntityCrudComponentBase<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> {\n /**\n * Wrappers around entityName & entityNamePlural properties.\n */\n getEntityName(): string;\n getEntityNamePlural(): string;\n /**\n * Wrapper around idKey property.\n */\n getIdKey(): string;\n /**\n * This method should return the entity URL for the given entity id.\n * The entity URL should include any additional query parameters that were\n * passed to the SPMatEntityCrudComponentBase component's `url` property.\n * @param id\n */\n getEntityUrl(id: TEntity[IdKey]|undefined): string;\n /**\n * FormViewHostComponent will call this to close the Create/Edit pane.\n * SPMatEntityCrudComponentBase implementor will destroy the client form\n * view and hide the Create/Edit form view pane and show the hidden\n * entity list view.\n * @returns\n */\n closeCreateEdit: (cancelled: boolean) => void;\n /**\n * Used internally by FormViewHostComponent to determine if the client form\n * view wants to intercept user's cancel the create/edit operation. Perhaps\n * with the Yes/No prompt 'Lose changes?'\n * @returns boolean indicating it's okay to cancel the create/edit operation.\n */\n canCancelEdit: () => boolean;\n /**\n * Client form view can register a callback that will be invoked by the\n * framework when user cancels the create/edit operation by clicking on the\n * close button on the top right.\n * @param callback\n * @returns\n */\n registerCanCancelEditCallback: (callback: () => boolean) => void;\n /**\n * Initiates update on the given entity.\n * @returns\n */\n triggerEntityUpdate: (entity: TEntity) => void;\n /**\n * Initiates entity delete.\n * @returns\n */\n triggerEntityDelete: (entity: TEntity) => void;\n /**\n * Called by client form-view host component to close a new entity.\n * @param entityValue The ReactiveForm.value object that the server expects\n * to create a new object.\n * @returns The new Entity object returned by the server. For typical REST\n * API, this would be of the same shape as the objects returned by the\n * REST's GET request.\n */\n create: (entityValue: any) => Observable<any>;\n /**\n * Called by client form-view host component to close a new entity.\n * @param id The id of the entity being edited.\n * @param entityValue The ReactiveForm.value object that the server expects\n * to update the new object.\n * @returns The new Entity object returned by the server. For typical REST\n * API, this would be of the same shape as the objects returned by the\n * REST's GET request.\n */\n update: (id: any, entityValue: any) => Observable<any>;\n /**\n * Load the entity with the given id from server.\n * @param id The id of the entity to load.\n * @param params Additional parameters for loading the entity.\n * @returns An observable of the loaded entity.\n */\n loadEntity: (id: any, params: string | HttpParams) => Observable<TEntity>;\n /**\n * Close the preview pane.\n * @returns\n */\n closePreview: () => void;\n /**\n * Returns the context menu items for the entity. This can be used to build\n * the context menu for an entity in its preview pane toolbar.\n * @returns\n */\n getItemActions(entity?: TEntity): SPContextMenuItem[];\n /**\n * Returns the class to be used for the preview pane content. This interface\n * is provided to allow the PreviewPaneComponent to access the client\n * configured class for the preview pane content.\n */\n getPreviewPaneContentClass(): string;\n\n getFormPaneWrapperClass(): string;\n\n getFormPaneContentClass(): string;\n\n getItemLabel(): string | Observable<string>;\n\n getItemLabelPlural(): string | Observable<string>;\n /*\n * Remove the entity with the given id from the list of entities.\n * This is typically called by the client when it peforms the delete\n * operation itself without using the MatEntityCrud's delete operation.\n *\n * @param id The id of the entity to remove.\n * @returns None\n **/\n removeEntity(id: TEntity[IdKey]): void;\n /**\n * Update the entity with the given id in the list of entities.\n * This is typically called by the client when it has performed an update\n * on the entity itself and want to reflect the resulting changed\n * entity in the list view.\n *\n * @param id The id of the entity to update.\n * @param data The updated entity.\n */\n updateEntity(id: TEntity[IdKey], data: TEntity): void;\n /**\n * Perform a custom action on the entity endpoint. The action is specified\n * by the verb argument, which will be used to derive the final URL. This\n * is keeping in line with DRF specification where viewsets can define\n * custom action methods, which translate into endpoints with the same name\n * ast he action method.\n * @param id id of the entity to perform the action on.\n * @param verb The action verb, which will be appended to the entity URL to\n * derive the final URL for the POST request.\n * @param addlParams additional query parameters to include in the request.\n * Called `additional` as these are in addition to the query params specified\n * in the CRUD's endpoint.\n * @param data the data to send with the request for the POST\n * @returns Observable<TEntity>\n */\n doEntityAction(\n id: TEntity[IdKey],\n verb: string,\n addlParams: HttpParams,\n data: any,\n busyWheelName: string\n ): Observable<any>;\n}\n","import {\n ChangeDetectionStrategy,\n Component,\n computed,\n input,\n InputSignal,\n OnDestroy,\n OnInit,\n} from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatToolbarModule } from '@angular/material/toolbar';\nimport { SPContextMenuItem } from '@smallpearl/ngx-helper/mat-context-menu';\nimport {\n ITEM_ACTION_DELETE,\n ITEM_ACTION_UPDATE,\n SPMatEntityCrudComponentBase,\n} from './mat-entity-crud-internal-types';\n\n/**\n * A preview pane container to provide a consistent UX for all preview panes.\n * It consits of a toolbar on the top and a container div below that takes up\n * the rest of the preview pane area.\n */\n@Component({\n imports: [MatToolbarModule, MatButtonModule, MatIconModule],\n selector: 'sp-mat-entity-crud-preview-pane',\n template: `\n <div class=\"preview-wrapper\">\n <mat-toolbar>\n <mat-toolbar-row>\n @if (title()) {\n <h2>{{ title() }}</h2>\n } @if (!hideUpdate()) {\n <button\n mat-icon-button\n aria-label=\"Edit\"\n (click)=\"onEdit()\"\n [disabled]=\"_disableUpdate()\"\n >\n <mat-icon>edit</mat-icon>\n </button>\n } @if (!hideDelete()) {\n <button\n mat-icon-button\n aria-label=\"Delete\"\n (click)=\"onDelete()\"\n [disabled]=\"_disableDelete()\"\n >\n <mat-icon>delete</mat-icon>\n </button>\n }\n <ng-content select=\"[previewToolbarContent]\"></ng-content>\n <span class=\"spacer\"></span>\n <button mat-icon-button aria-label=\"Close\" (click)=\"onClose()\">\n <mat-icon>close</mat-icon>\n </button>\n </mat-toolbar-row>\n </mat-toolbar>\n <div\n [class]=\"\n 'preview-content ' +\n entityCrudComponent().getPreviewPaneContentClass()\n \"\n >\n <ng-content select=\"[previewContent]\"></ng-content>\n </div>\n </div>\n `,\n styles: [\n `\n .preview-wrapper {\n display: flex;\n flex-direction: column;\n height: 100% !important;\n width: 100% !important;\n }\n mat-toolbar {\n background-color: var(--mat-sys-surface-variant);\n }\n .spacer {\n flex: 1 1 auto;\n }\n .preview-content {\n padding: 0.4em;\n flex-grow: 1;\n overflow: scroll;\n }\n `,\n ],\n changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SPMatEntityCrudPreviewPaneComponent<TEntity extends { [P in IdKey]: PropertyKey }, IdKey extends string = 'id'>\n implements OnInit, OnDestroy\n{\n entity = input.required<TEntity>();\n entityCrudComponent = input.required<SPMatEntityCrudComponentBase<TEntity, IdKey>>();\n title = input<string>();\n disableUpdate = input<boolean>(false);\n hideUpdate = input<boolean>(false);\n disableDelete = input<boolean>(false);\n hideDelete = input<boolean>(false);\n itemActions!: SPContextMenuItem[];\n\n _disableActionFactory = (role: string, signal?: InputSignal<boolean>) => {\n return computed(() => {\n if (signal && signal()) {\n return true;\n }\n const actionFn = this.itemActions.find((a) => a.role === role)?.disable;\n if (actionFn && actionFn(this.entity())) {\n return true;\n }\n return false;\n });\n };\n _disableUpdate = this._disableActionFactory(\n ITEM_ACTION_UPDATE,\n this.disableUpdate\n );\n _disableDelete = this._disableActionFactory(\n ITEM_ACTION_DELETE,\n this.disableDelete\n );\n\n ngOnInit() {\n this.itemActions = this.entityCrudComponent().getItemActions();\n }\n\n ngOnDestroy(): void {}\n\n onEdit() {\n this.entityCrudComponent().triggerEntityUpdate(this.entity());\n }\n\n onDelete() {\n this.entityCrudComponent().triggerEntityDelete(this.entity());\n }\n\n onClose() {\n this.entityCrudComponent().closePreview();\n }\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":["i2","i3","i6","i9","i10","i1"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"smallpearl-ngx-helper-mat-entity-crud.mjs","sources":["../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/convert-context-input-to-http-context.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud-form-base.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud-types.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/providers.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/default-config.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/form-view-host.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/preview-host.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/mat-entity-crud-internal-types.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/src/preview-pane.component.ts","../../../../projects/smallpearl/ngx-helper/mat-entity-crud/smallpearl-ngx-helper-mat-entity-crud.ts"],"sourcesContent":["import { HttpContext, HttpContextToken } from '@angular/common/http';\n\nexport type HttpContextInput =\n | [[HttpContextToken<any>, any]]\n | [HttpContextToken<any>, any]\n | HttpContext;\n\n/**\n * Converts array of HttpContextToken key, value pairs to HttpContext\n * object in argument 'context'.\n * @param context HTTP context to which the key, value pairs are added\n * @param reqContext HttpContextToken key, value pairs array\n * @returns HttpContext object, with the key, value pairs added. This is\n * the same object as the 'context' argument.\n */\nexport function convertHttpContextInputToHttpContext(\n context: HttpContext,\n reqContext: HttpContextInput\n): HttpContext {\n if (reqContext instanceof HttpContext) {\n // reqContext is already an HttpContext object.\n for (const k of reqContext.keys()) {\n context.set(k, reqContext.get(k));\n }\n } else if (Array.isArray(reqContext) && reqContext.length == 2 && !Array.isArray(reqContext[0])) {\n // one dimensional array of a key, value pair.\n context.set(reqContext[0], reqContext[1]);\n } else {\n reqContext.forEach(([k, v]) => context.set(k, v));\n }\n return context;\n}\n","import { HttpClient, HttpContext, HttpParams } from '@angular/common/http';\nimport { ChangeDetectorRef, Component, computed, inject, input, OnDestroy, OnInit, signal } from '@angular/core';\nimport { AbstractControl, UntypedFormGroup } from '@angular/forms';\nimport { TranslocoService } from '@jsverse/transloco';\nimport { setServerErrorsAsFormErrors } from '@smallpearl/ngx-helper/forms';\nimport { map, Observable, Subscription, tap } from 'rxjs';\n// import { getEntityCrudConfig } from './default-config';\nimport { sideloadToComposite } from '@smallpearl/ngx-helper/sideload';\nimport { convertHttpContextInputToHttpContext, HttpContextInput } from './convert-context-input-to-http-context';\nimport { SPMatEntityCrudCreateEditBridge } from './mat-entity-crud-types';\n\n/**\n * This is a convenience base class that clients can derive from to implement\n * their CRUD form component. Particularly this class registers the change\n * detection hook which will be called when the user attempts to close the\n * form's parent container pane via the Close button on the top right.\n *\n * This button behaves like a Cancel button in a desktop app and therefore if\n * the user has entered any data in the form's controls, (determined by\n * checking form.touched), then a 'Lose Changes' prompt is displayed allowing\n * the user to cancel the closure.\n *\n * The `@Component` decorator is fake to keep the VSCode angular linter quiet.\n *\n * This class can be used in two modes:\n *\n * I. SPMatEntityCrudComponent mode\n * This mode relies on a bridge interface that implements the\n * SPMatEntityCrudCreateEditBridge interface to perform the entity\n * load/create/update operations. This is the intended mode when the\n * component is used as a part of the SPMatEntityCrudComponent to\n * create/update an entity. This mode requires the following properties\n * to be set:\n * - entity: TEntity | TEntity[IdKey] | undefined (for create)\n * - bridge: SPMatEntityCrudCreateEditBridge\n *\n * II. Standalone mode\n * This mode does not rely on the bridge interface and the component\n * itself performs the entity load/create/update operations.\n * This mode requires the following properties to be set:\n * - entity: TEntity | TEntity[IdKey] | undefined (for create)\n * - baseUrl: string - Base URL for CRUD operations. This URL does not\n * include the entity id. The entity id will be appended to this URL\n * for entity load and update operations. For create operation, this\n * URL is used as is.\n * - entityName: string - Name of the entity, used to parse sideloaded\n * entity responses.\n * - httpReqContext?: HttpContextInput - Optional HTTP context to be\n * passed to the HTTP requests. For instance, if your app has a HTTP\n * interceptor that adds authentication tokens to the requests based\n * on a HttpContextToken, then you can pass that token here.\n *\n * I. SPMatEntityCrudComponent mode:\n *\n * 1. Declare a FormGroup<> type as\n *\n * ```\n * type MyForm = FormGroup<{\n * name: FormControl<string>;\n * type: FormControl<string>;\n * notes: FormControl<string>;\n * }>;\n * ```\n *\n * 2. Derive your form's component class from this and implement the\n * createForm() method returing the FormGroup<> instance that matches\n * the FormGroup concrete type above.\n *\n * ```\n * class MyFormComponent extends SPMatEntityCrudFormBase<MyForm, MyEntity> {\n * constructor() {\n * super()\n * }\n * createForm() {\n * return new FormGroup([...])\n * }\n * }\n * ```\n *\n * 3. If your form's value requires manipulation before being sent to the\n * server, override `getFormValue()` method and do it there before returning\n * the modified values.\n *\n * 4. Wire up the form in the template as below\n *\n * ```html\n * @if (loadEntity$ | async) {\n * <form [formGroup]='form'.. (ngSubmit)=\"onSubmit()\">\n * <button type=\"submit\">Submit</button>\n * </form>\n * } @else {\n * <div>Loading...</div>\n * }\n * ```\n *\n * Here `loadEntity$` is an Observable<boolean> that upon emission of `true`\n * indicates that the entity has been loaded from server (in case of edit)\n * and the form is ready to be displayed. Note that if the full entity was\n * passed in the `entity` input property, then no server load is necessary\n * and the form will be created immediately.\n *\n * 5. In the parent component that hosts the SPMatEntityCrudComponent, set\n * the `entity` and `bridge` input properties of this component to\n * appropriate values. For instance, if your form component has the\n * selector `app-my-entity-form`, then the parent component's template\n * will have:\n *\n * ```html\n * <sp-mat-entity-crud\n * ...\n * createEditFormTemplate=\"entityFormTemplate\"\n * ></sp-mat-entity-crud>\n * <ng-template #entityFormTemplate let-data=\"data\">\n * <app-my-entity-form\n * [entity]=\"data.entity\"\n * [bridge]=\"data.bridge\"\n * ></app-my-entity-form>\n * </ng-template>\n * ```\n *\n * II. Standalone mode\n *\n * 1..4. Same as above, except set the required `bridge` input to `undefined`.\n * 5. Initialize the component's inputs `baseUrl` and `entityName` with the\n * appropriate values. If you would like to pass additional HTTP context to\n * the HTTP requests, then set the `httpReqContext` input as well.\n * If the entity uses an id key other than 'id', then set the `idKey` input\n * to the appropriate id key name.\n * 6. If you want to retrieve the created/updated entity after the create/update\n * operation, override the `onPostCreate()` and/or `onPostUpdate()` methods\n * respectively.\n */\n@Component({\n selector: '_#_sp-mat-entity-crud-form-base_#_',\n template: ``,\n standalone: false,\n})\nexport abstract class SPMatEntityCrudFormBase<\n TFormGroup extends AbstractControl,\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> implements OnInit, OnDestroy\n{\n // bridge mode inputs\n entity = input<TEntity | TEntity[IdKey]>();\n bridge = input<SPMatEntityCrudCreateEditBridge | undefined>();\n params = input<any>();\n // END bridge mode inputs\n\n // standalone mode inputs\n // Entity name, which is used to parse sideloaded entity responses\n entityName = input<string>();\n // Base CRUD URL, which is the GET-list-of-entities/POST-to-create\n // URL. Update URL will be derived from this ias `baseUrl()/${TEntity[IdKey]}`\n baseUrl = input<string>();\n // Additional request context to be passed to the request\n httpReqContext = input<HttpContextInput | undefined>();\n // ID key, defaults to 'id'\n idKey = input<string>('id');\n // END standalone mode inputs\n\n // IMPLEMENTATION\n loadEntity$!: Observable<boolean>;\n _entity = signal<TEntity | undefined>(undefined);\n sub$ = new Subscription();\n\n // Store for internal form signal. form() is computed from this.\n _form = signal<TFormGroup | undefined>(undefined);\n // Force typecast to TFormGroup so that we can use it in the template\n // without having to use the non-nullable operator ! with every reference\n // of form(). In any case the form() signal is always set in ngOnInit()\n // method after the form is created. And if form() is not set, then there\n // will be errors while loading the form in the template.\n form = computed(() => this._form() as TFormGroup);\n\n transloco = inject(TranslocoService);\n cdr = inject(ChangeDetectorRef);\n http = inject(HttpClient);\n\n // This is really not necessary. We can check for this.bridge() directly.\n mode = computed<'standalone' | 'bridge'>(() => {\n return this.bridge() ? 'bridge' : 'standalone';\n });\n\n canCancelEdit = () => {\n return this._canCancelEdit();\n };\n\n _canCancelEdit() {\n const form = this._form();\n if (form && form.touched) {\n return window.confirm(\n this.transloco.translate('spMatEntityCrud.loseChangesConfirm')\n );\n }\n return true;\n }\n\n ngOnInit() {\n // Validate inputs. Either bridge or (baseUrl and entityName) must be\n // defined.\n if (\n this.mode() === 'standalone' &&\n (!this.getBaseUrl() || !this.getEntityName())\n ) {\n throw new Error(\n 'SPMatEntityCrudFormBase: baseUrl and entityName inputs must be defined in standalone mode.'\n );\n }\n this.loadEntity$ = (\n typeof this.entity() === 'object' || this.entity() === undefined\n ? new Observable<TEntity | undefined>((subscriber) => {\n subscriber.next(this.entity() as TEntity | undefined);\n subscriber.complete();\n })\n : this.load(this.entity() as any)\n ).pipe(\n map((resp) => {\n const compositeEntity = this.getEntityFromLoadResponse(resp);\n this._entity.set(compositeEntity);\n this._form.set(this.createForm(compositeEntity));\n const bridge = this.bridge();\n if (bridge && bridge.registerCanCancelEditCallback) {\n bridge.registerCanCancelEditCallback(this.canCancelEdit);\n }\n return true;\n })\n );\n }\n\n ngOnDestroy() {\n this.sub$.unsubscribe();\n }\n\n /**\n * Additional parameters for loading the entity, in case this.entity() value\n * is of type TEntity[IdKey].\n * @returns\n */\n getLoadEntityParams(): string | HttpParams {\n return '';\n }\n\n /**\n * Return the TEntity object from the response returned by the\n * load() method. Typically entity load returns the actual\n * entity object itself. In some cases, where response is sideloaded, the\n * default implementation here uses the `sideloadToComposite()` utility to\n * extract the entity from the response after merging (inplace) the\n * sideloaded data into a composite.\n *\n * If you have a different response shape, or if your sideloaded object\n * response requires custom custom `sideloadDataMap`, override this method\n * and implement your custom logic to extract the TEntity object from the\n * response.\n * @param resp\n * @returns\n */\n getEntityFromLoadResponse(resp: any): TEntity | undefined {\n if (!resp || typeof resp !== 'object') {\n return undefined;\n }\n const entityName = this.getEntityName();\n if (resp.hasOwnProperty(this.getIdKey())) {\n return resp as TEntity;\n } else if (entityName && resp.hasOwnProperty(entityName)) {\n // const sideloadDataMap = this.sideloadDataMap();\n return sideloadToComposite(resp, entityName, this.getIdKey()) as TEntity;\n }\n return undefined;\n }\n\n /**\n * Create the TFormGroup FormGroup class that will be used for the reactive\n * form.\n * @param entity\n */\n abstract createForm(entity: TEntity | undefined): TFormGroup;\n\n /**\n * Override to customize the id key name if it's not 'id'\n * @returns The name of the unique identifier key that will be used to\n * extract the entity's id for UPDATE operation.\n */\n getIdKey() {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.getIdKey();\n }\n return this.idKey();\n }\n\n /**\n * Return the form's value to be sent to server as Create/Update CRUD\n * operation data.\n * @returns\n */\n getFormValue() {\n const form = this.form();\n return form ? form.value : undefined;\n }\n\n onSubmit() {\n const value = this.getFormValue();\n const obs = !this._entity()\n ? this.create(value)\n : this.update((this._entity() as any)[this.getIdKey()], value);\n this.sub$.add(\n obs\n ?.pipe(\n tap((entity) =>\n this._entity()\n ? this.onPostUpdate(entity)\n : this.onPostCreate(entity)\n ),\n setServerErrorsAsFormErrors(\n this._form() as unknown as UntypedFormGroup,\n this.cdr\n )\n )\n .subscribe()\n );\n }\n\n onPostCreate(entity: TEntity) {\n /* empty */\n }\n\n onPostUpdate(entity: TEntity) {\n /* empty */\n }\n\n /**\n * Loads the entity if `this.entity()` is of type TEntity[IdKey]. If `bridge`\n * input is defined, then it's `loadEntity()` method is used to load the\n * entity. Otherwise, then this method attempts to load the entity using\n * HTTP GET from the URL derived from `baseUrl` input.\n * @param entityId\n * @param params\n * @returns\n */\n load(entityId: any): Observable<TEntity> {\n const bridge = this.bridge();\n const params = this.getLoadEntityParams();\n if (bridge) {\n return bridge.loadEntity(entityId, params);\n }\n\n // Try to load using baseUrl.\n const url = this.getEntityUrl(entityId);\n return this.http\n .get<TEntity>(this.getEntityUrl(entityId), {\n params:\n typeof params === 'string'\n ? new HttpParams({ fromString: params })\n : params,\n context: this.getRequestContext(),\n })\n .pipe(map((resp) => this.getEntityFromLoadResponse(resp) as TEntity));\n }\n\n /**\n * Create a new entity using the bridge if defined, otherwise using HTTP\n * POST to the `baseUrl`.\n * @param values\n * @returns\n */\n protected create(values: any): Observable<TEntity> {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.create(values);\n }\n return this.http\n .post<TEntity>(this.getBaseUrl()!, values, {\n context: this.getRequestContext(),\n })\n .pipe(map((resp) => this.getEntityFromLoadResponse(resp) as TEntity));\n }\n\n /**\n * Update an existing entity using the bridge if defined, otherwise using HTTP\n * PATCH to the URL derived from `baseUrl` and the entity id.\n * @param id\n * @param values\n * @returns\n */\n protected update(id: any, values: any): Observable<TEntity> {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.update(id, values);\n }\n return this.http\n .patch<TEntity>(this.getEntityUrl(id), values, {\n context: this.getRequestContext(),\n })\n .pipe(map((resp) => this.getEntityFromLoadResponse(resp) as TEntity));\n }\n\n /**\n * Wrapper around entityName input to get the entity name. If `bridge` input\n * is defined, then its `getEntityName()` method is used. This allows\n * derived classes to override this method to provide custom logic to\n * determine the entity name.\n * @returns\n */\n protected getEntityName(): string | undefined {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.getEntityName();\n }\n return this.entityName();\n }\n\n /**\n * Returns the baseUrl. Derived classes can override this to provide custom\n * logic to determine the baseUrl.\n * @returns\n */\n protected getBaseUrl(): string | undefined {\n return this.baseUrl();\n }\n\n /**\n * Returns the entity URL for the given entity id. If `bridge` input is\n * defined, then its `getEntityUrl()` method is used. Otherwise, the URL is\n * derived from `baseUrl` input.\n * @param entityId\n * @returns\n */\n protected getEntityUrl(entityId: any): string {\n const bridge = this.bridge();\n if (bridge) {\n return bridge.getEntityUrl(entityId);\n }\n const baseUrl = this.getBaseUrl();\n if (baseUrl) {\n const urlParts = baseUrl.split('?');\n return `${urlParts[0]}${String(entityId)}/${\n urlParts[1] ? '?' + urlParts[1] : ''\n }`;\n }\n console.warn(\n 'SPMatEntityCrudFormBase.getEntityUrl: Cannot determine entity URL as neither baseUrl nor bridge inputs are provided.'\n );\n return '';\n }\n\n protected getRequestContext(): HttpContext {\n let context = new HttpContext();\n const httpReqContext = this.httpReqContext();\n if (httpReqContext) {\n context = convertHttpContextInputToHttpContext(context, httpReqContext);\n }\n return context;\n }\n}\n","import { HttpContextToken, HttpParams } from \"@angular/common/http\";\nimport { SPContextMenuItem } from \"@smallpearl/ngx-helper/mat-context-menu\";\nimport { Observable } from \"rxjs\";\n\n/**\n * Prototype of the function to parse the CRUD action response.\n */\nexport type SPMatEntityCrudResponseParser = (\n entityName: string,\n idKey: string,\n method: string, // 'create' | 'retrieve' | 'update' | 'delete',\n resp: any\n) => any|undefined;\n\n\n/**\n * Global config for SPMatEntityList component.\n */\nexport interface SPMatEntityCrudConfig {\n /**\n * The item actions that will be shown for each item in the list.\n * This defaults to 'Update' & 'Delete' actions, but can be customized\n * by this property. Note the item actions can be set for individual\n * <sp-mat-entity-crud> component through its itemActions property.\n */\n defaultItemActions?: SPContextMenuItem[];\n /**\n * Global crud response parser.\n */\n crudOpResponseParser?: SPMatEntityCrudResponseParser;\n}\n\n/**\n * This is the interface through which the client provided CRUD form component\n * interacts with the 'host' SPMatEntityCrudComponent. When the form wants to\n * submit an entity to the server (for create or update), it should call the\n * one of the create or update methods. The interface also provides other\n * methods for the form component to interact with SPMatEntityCrudComponent\n * such as refresh its entities list, close the form pane, etc.\n *\n * The interface name has a 'Bridge' as the interface acts as a bridge between\n * the client provided form handler component and the host\n * SPMatEntityCrudComponent.\n */\nexport interface SPMatEntityCrudCreateEditBridge {\n /**\n * Returns the entity name as provided to the host SPMatEntityCrudComponent.\n * @returns The entity name string.\n */\n getEntityName(): string;\n\n /**\n * Returns the entity id key as provided to the host SPMatEntityCrudComponent.\n * @returns The entity id key string.\n */\n getIdKey(): string;\n\n /**\n * Get Entity url\n * @param cancel\n * @returns\n */\n getEntityUrl(entityId: any): string;\n\n /**\n * Close the edit/update form pane. This WON'T call the 'cancelEditCallback'\n * even if one is registered.\n */\n close: (cancel: boolean) => void;\n /**\n * Client form view can register a callback that will be invoked by the\n * framework when user cancels the create/edit operation by clicking on the\n * close button on the top right.\n * @param callback\n * @returns None\n */\n registerCanCancelEditCallback: (callback: () => boolean) => void;\n // Parameters of type any are entity values are typically the output of\n // form.value and therefore their types would not necessarily match TEntity.\n // id can be typed as TEntity[Idkey], but TSC doesn't allow that yet.\n /**\n * Create a new instance of TEntity, by sending a POST request to remote.\n * @param entityValue This is the typically the output of Reactive form's\n * form.value. Since this value's shape may be different from TEntity and is\n * known only to client form, we use 'any'.\n * @returns None\n * @inner Implementation will show a busy wheel centered on the form\n * view while the async function to update the object remains active.\n */\n create: (entityValue: any) => Observable<any>;\n /**\n * Update the entity with id `id` with new values in entityValue.\n * @param id TEntity id\n * @param entityValue Entity values to be updated.\n * @returns None\n * @inner Implementation will show a busy wheel centered on the form\n * view while the async function to update the object remains active.\n */\n update: (id: any, entityValue: any) => Observable<any>;\n\n /**\n * Load the entity with the given id from server.\n * @param id The id of the entity to load.\n * @param params Additional parameters for loading the entity. This will\n * be passed to the underlying data service's 'get' method. This can be a\n * query parameters string or HttpParams object.\n * @returns Observable of the loaded entity.\n */\n loadEntity: (id: any, params: string|HttpParams) => Observable<any>;\n}\n\n/**\n * Prototype of the function that will be used instead of HttpClient for\n * CRUD operations.\n * @param op - the CRUD operation being requested\n * @param entityValue - The entity or entity value upon which the operation\n * is being requested. for 'create' & 'update' this will be the value\n * of the reactive form. This is typically form.value or the 2nd arg to create\n * & update methods of SPMatEntityCrudCreateEditBridge.\n */\nexport type CRUD_OP_FN<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> = (\n op: string,\n id: TEntity[IdKey] | undefined, // valid only for 'get', 'update' & 'delete'\n entityValue: any, // valid only for 'create' & 'update'\n entityCrudComponent: any,\n) => Observable<TEntity | null>;\n\n\nexport type ALLOW_ITEM_ACTION_FN<TEntity> = (entity: TEntity, action: string) => boolean;\n\n/**\n * This interface is used to define sub types for the \"New {{item}}\" button on\n * top of the CRUD UI. An array of these is provided as the value to the\n * component property 'newItemSubTypes'.\n */\nexport interface NewItemSubType {\n /**\n * A role string that will be passed as argument to the (action) event\n * handler. This string allows the event handler to distinguish the selected\n * sub-type.\n *\n * The special keyword '_new_' can be used to activate the\n * `createEditTemplate` template if one is provided. In this case the params\n * element value (see below) can be used in the template to distinguish\n * between different menu items.\n */\n role: string;\n /**\n * Label displayed in the menu representing this role.\n */\n label: string;\n /**\n * Arbitrary value that will be passed to the 'createEditTemplate' in the\n * $implicit template context as 'params'. You can access this in the\n * template like below (see `data.params`):-\n ```\n <ng-template #createEdit let-data>\n <app-create-edit-entity-demo\n [bridge]=\"data.bridge\"\n [entity]=\"data.entity\"\n [params]=\"data.params\"\n ></app-create-edit-entity-demo>\n </ng-template>\n ```\n\n If params is an object and it includes the key 'title', its value will be\n used as the title for the edit form.\n */\n params?: any;\n}\n\nexport type CrudOp = 'create'|'retrieve'|'update'|'delete'|undefined;\n\nexport interface SPMatEntityCrudHttpContext {\n entityName: string;\n entityNamePlural: string;\n endpoint: string;\n op: CrudOp;\n}\n\nexport const SP_MAT_ENTITY_CRUD_HTTP_CONTEXT =\n new HttpContextToken<SPMatEntityCrudHttpContext>(() => ({\n entityName: '',\n entityNamePlural: '',\n endpoint: '',\n op: undefined,\n }));\n","import { InjectionToken } from '@angular/core';\nimport { SPMatEntityCrudConfig } from './mat-entity-crud-types';\n\nexport const SP_MAT_ENTITY_CRUD_CONFIG = new InjectionToken<SPMatEntityCrudConfig>('SPMatEntityCrudConfig');\n","import { inject } from \"@angular/core\";\nimport { SPMatEntityCrudConfig } from \"./mat-entity-crud-types\";\nimport { SP_MAT_ENTITY_CRUD_CONFIG } from \"./providers\";\n\nfunction defaultCrudResponseParser(\n entityName: string,\n idKey: string,\n method: string, // 'create' | 'retrieve' | 'update' | 'delete',\n resp: any\n) {\n // If the response is an object with a property '<idKey>', return it as\n // TEntity.\n if (resp.hasOwnProperty(idKey)) {\n return resp;\n }\n // If the response has an object indexed at '<entityName>' and it has\n // the property '<idKey>', return it as TEntity.\n if (resp.hasOwnProperty(entityName)) {\n const obj = resp[entityName];\n if (obj.hasOwnProperty(idKey)) {\n return obj;\n }\n }\n // Return undefined, indicating that we could't parse the response.\n return undefined;\n}\n\nexport const DefaultSPMatEntityCrudConfig: SPMatEntityCrudConfig = {\n crudOpResponseParser: defaultCrudResponseParser\n};\n\n/**\n * To be called from an object constructor as it internally calls Angular's\n * inject() API.\n * @param userConfig\n * @returns\n */\nexport function getEntityCrudConfig(): SPMatEntityCrudConfig {\n const userCrudConfig = inject(SP_MAT_ENTITY_CRUD_CONFIG, {\n optional: true,\n });\n return {\n ...DefaultSPMatEntityCrudConfig,\n ...(userCrudConfig ?? {}),\n };\n}\n","import { CommonModule } from '@angular/common';\nimport { HttpParams } from '@angular/common/http';\nimport {\n ChangeDetectionStrategy,\n Component,\n computed,\n EmbeddedViewRef,\n inject,\n input,\n OnDestroy,\n OnInit,\n signal,\n TemplateRef,\n viewChild,\n ViewContainerRef\n} from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport { TranslocoModule, TranslocoService } from '@jsverse/transloco';\nimport { SPMatHostBusyWheelDirective } from '@smallpearl/ngx-helper/mat-busy-wheel';\nimport { Observable, of, Subscription, tap } from 'rxjs';\nimport { getEntityCrudConfig } from './default-config';\nimport { SPMatEntityCrudComponentBase } from './mat-entity-crud-internal-types';\nimport { SPMatEntityCrudConfig, SPMatEntityCrudCreateEditBridge } from './mat-entity-crud-types';\n\n@Component({\n imports: [\n CommonModule,\n MatButtonModule,\n MatIconModule,\n TranslocoModule,\n SPMatHostBusyWheelDirective,\n ],\n selector: 'sp-create-edit-entity-host',\n template: `\n <div\n [class]=\"\n 'sp-mat-crud-form-wrapper ' +\n entityCrudComponentBase().getFormPaneWrapperClass()\n \"\n spHostBusyWheel=\"formBusyWheel\"\n *transloco=\"let t\"\n >\n <div\n [class]=\"\n 'sp-mat-crud-form-content ' +\n entityCrudComponentBase().getFormPaneContentClass()\n \"\n >\n <div class=\"create-edit-topbar\">\n <div class=\"title\">\n @if (title()) {\n {{ title() | async }}\n } @else {\n {{\n t('spMatEntityCrud.' + (entity() ? 'editItem' : 'newItem'), {\n item: (this._itemLabel() | async)\n })\n }}\n }\n </div>\n <div class=\"spacer\"></div>\n <div class=\"close\">\n <button mat-icon-button (click)=\"onClose()\">\n <mat-icon>cancel</mat-icon>\n </button>\n </div>\n </div>\n <div class=\"form-container\">\n <ng-container #clientFormContainer></ng-container>\n </div>\n </div>\n </div>\n `,\n styles: `\n .sp-mat-crud-form-wrapper {\n width: 100% !important;\n height: 100% !important;\n }\n .sp-mat-crud-form-content {\n height: 100%;\n width: 100%;\n display: flex;\n flex-direction: column;\n padding: 0.4em;\n }\n .create-edit-topbar {\n display: flex;\n flex-direction: row;\n align-items: center;\n min-height: 48px;\n padding: 0.4em;\n }\n .form-container {\n padding-top: 0.4em;\n flex-grow: 1;\n overflow: auto;\n }\n .create-edit-topbar .title {\n font-size: 1.5em;\n font-weight: 500;\n }\n .create-edit-topbar .spacer {\n flex-grow: 1;\n }\n .create-edit-topbar .close {\n }\n `,\n changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FormViewHostComponent<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> implements SPMatEntityCrudCreateEditBridge, OnInit, OnDestroy\n{\n entityCrudComponentBase =\n input.required<SPMatEntityCrudComponentBase<TEntity, IdKey>>();\n clientViewTemplate = input<TemplateRef<any> | null>(null);\n\n _itemLabel = computed<Observable<string>>(() => {\n const label = this.entityCrudComponentBase().getItemLabel();\n return label instanceof Observable ? label : of(label);\n });\n _itemLabelPlural = computed<Observable<string>>(() => {\n const label = this.entityCrudComponentBase().getItemLabelPlural();\n return label instanceof Observable ? label : of(label);\n });\n\n entity = signal<TEntity | undefined>(undefined);\n title = signal<Observable<string> | undefined>(undefined);\n params = signal<any>(undefined);\n clientFormView!: EmbeddedViewRef<any> | null;\n vc = viewChild('clientFormContainer', { read: ViewContainerRef });\n config!: SPMatEntityCrudConfig;\n sub$ = new Subscription();\n transloco = inject(TranslocoService);\n\n constructor() {\n this.config = getEntityCrudConfig();\n }\n\n ngOnInit() {}\n\n ngOnDestroy(): void {\n this.sub$.unsubscribe();\n }\n\n show(entity: TEntity | undefined, params?: any) {\n this.entity.set(entity);\n if (params && params?.title) {\n this.title.set(\n params.title instanceof Observable ? params.title : of(params.title)\n );\n } else {\n // this.title.set(entity ? this.config.i18n.editItemLabel(this.itemLabel()) : this.config.i18n.newItemLabel(this.itemLabel()));\n // this.title.set(\n // this.transloco.translate(entity ? 'editItem' : 'newItem', {\n // item: this.itemLabel(),\n // })\n // );\n }\n this.params.set(params);\n this.createClientView();\n }\n\n // BEGIN SPMatEntityCrudCreateEditBridge METHODS //\n getEntityName(): string {\n return this.entityCrudComponentBase().getEntityName();\n }\n\n getIdKey(): string {\n return this.entityCrudComponentBase().getIdKey();\n }\n\n getEntityUrl(entityId: any): string {\n return this.entityCrudComponentBase().getEntityUrl(entityId);\n }\n\n close(cancel: boolean) {\n this.entityCrudComponentBase().closeCreateEdit(cancel);\n // destroy the client's form component\n this.destroyClientView();\n }\n\n registerCanCancelEditCallback(callback: () => boolean) {\n this.entityCrudComponentBase().registerCanCancelEditCallback(callback);\n }\n\n create(entityValue: any) {\n // console.log(\n // `SPCreateEditEntityHostComponent.create - entity: ${JSON.stringify(\n // entityValue\n // )}`\n // );\n const crudComponent = this.entityCrudComponentBase();\n return crudComponent\n ?.create(entityValue)\n .pipe(tap(() => this.close(false)));\n }\n\n update(id: any, entityValue: any) {\n // console.log(\n // `SPCreateEditEntityHostComponent.update - id: ${String(\n // id\n // )}, entity: ${entityValue}`\n // );\n const crudComponent = this.entityCrudComponentBase();\n return crudComponent\n ?.update(id, entityValue)\n .pipe(tap(() => this.close(false)));\n }\n\n loadEntity(\n id: string | number,\n params: string | HttpParams\n ): Observable<TEntity> {\n return this.entityCrudComponentBase().loadEntity(id, params);\n }\n // END SPMatEntityCrudCreateEditBridge METHODS //\n\n /**\n * Creates the client view provided via template\n */\n createClientView() {\n /** Render preview component if one was provided */\n const ft = this.clientViewTemplate();\n const vc = this.vc();\n if (ft && vc) {\n this.clientFormView = vc.createEmbeddedView(ft, {\n $implicit: {\n bridge: this,\n entity: this.entity(),\n params: this.params(),\n },\n });\n this.clientFormView.detectChanges();\n }\n }\n\n destroyClientView() {\n if (this.clientFormView) {\n this.clientFormView.destroy();\n this.clientFormView = null;\n }\n }\n\n onClose() {\n // Can we give the client form component a chance to intercept this\n // and cancel the closure?\n if (this.entityCrudComponentBase().canCancelEdit()) {\n this.entityCrudComponentBase().closeCreateEdit(true);\n this.destroyClientView();\n }\n }\n}\n","import {\n ChangeDetectionStrategy,\n Component,\n EmbeddedViewRef,\n input,\n OnDestroy,\n OnInit,\n signal,\n TemplateRef,\n viewChild,\n ViewContainerRef\n} from '@angular/core';\nimport { SPMatEntityCrudComponentBase } from './mat-entity-crud-internal-types';\n\n@Component({\n imports: [],\n selector: 'sp-entity-crud-preview-host',\n template: ` <ng-container #previewComponent></ng-container> `,\n changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PreviewHostComponent<TEntity extends { [P in IdKey]: PropertyKey }, IdKey extends string = 'id'> implements OnInit, OnDestroy {\n vc = viewChild('previewComponent', { read: ViewContainerRef });\n\n entityCrudComponentBase = input.required<SPMatEntityCrudComponentBase<TEntity, IdKey>>();\n clientViewTemplate = input<TemplateRef<any> | null>(null);\n entity = signal<TEntity|undefined>(undefined);\n clientView!: EmbeddedViewRef<any> | null;\n\n constructor() {\n // effect(() => {\n // const tmpl = this.clientViewTemplate();\n // this.createClientView(tmpl);\n // });\n }\n\n ngOnInit(): void {}\n\n ngOnDestroy(): void {}\n\n show(entity: TEntity|undefined, params?: any) {\n this.entity.set(entity);\n // if (params && params?.title) {\n // this.title.set(params.title);\n // } else {\n // this.title.set(entity ? this.config.i18n.editItemLabel(this.itemLabel()) : this.config.i18n.newItemLabel(this.itemLabel()));\n // }\n // this.params.set(params);\n this.createClientView();\n }\n\n close() {\n // this.entityCrudComponentBase().closeCreateEdit(cancel);\n // destroy the client's form component\n this.destroyClientView();\n }\n\n private createClientView() {\n if (this.clientView) {\n // We have only one view in the ng-container. So we might as well\n // call clear() to remove all views contained in it.\n this.vc()!.clear();\n this.clientView.destroy();\n }\n /** Render preview component if one was provided */\n const ft = this.clientViewTemplate();\n const vc = this.vc();\n if (ft && vc) {\n this.clientView = this.vc()!.createEmbeddedView(\n ft,\n {\n $implicit: {\n entity: this.entity(),\n entityCrudComponent: this.entityCrudComponentBase(),\n },\n }\n );\n this.clientView.detectChanges();\n }\n }\n\n\n destroyClientView() {\n if (this.clientView) {\n this.clientView.destroy();\n this.clientView = null;\n }\n }\n\n // close() {\n // this.closePreview.emit();\n // }\n}\n","import { CommonModule } from '@angular/common';\nimport { HttpClient, HttpContext, HttpContextToken, HttpParams } from '@angular/common/http';\nimport {\n AfterViewInit,\n ChangeDetectionStrategy,\n Component,\n computed,\n ContentChildren,\n effect,\n EventEmitter,\n inject,\n Injector,\n input,\n Output,\n QueryList,\n signal,\n TemplateRef,\n viewChild,\n viewChildren\n} from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\nimport { MatSortModule } from '@angular/material/sort';\nimport { MatColumnDef, MatTableModule } from '@angular/material/table';\nimport { RouterModule } from '@angular/router';\nimport {\n showBusyWheelUntilComplete,\n SPMatHostBusyWheelDirective\n} from '@smallpearl/ngx-helper/mat-busy-wheel';\nimport {\n SPMatContextMenuComponent\n} from '@smallpearl/ngx-helper/mat-context-menu';\nimport { SPMatEntityListComponent } from '@smallpearl/ngx-helper/mat-entity-list';\n\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { provideTranslocoScope, TranslocoModule, TranslocoService } from '@jsverse/transloco';\nimport { AngularSplitModule } from 'angular-split';\nimport { clone, startCase } from 'lodash';\nimport { plural } from 'pluralize';\nimport { catchError, EMPTY, firstValueFrom, map, Observable, of, Subscription, switchMap, tap, throwError } from 'rxjs';\nimport { convertHttpContextInputToHttpContext } from './convert-context-input-to-http-context';\nimport { getEntityCrudConfig } from './default-config';\nimport { FormViewHostComponent } from './form-view-host.component';\nimport { SPMatEntityCrudComponentBase } from './mat-entity-crud-internal-types';\nimport { MatEntityCrudItemAction } from './mat-entity-crud-item-action';\nimport {\n ALLOW_ITEM_ACTION_FN,\n CRUD_OP_FN,\n CrudOp,\n NewItemSubType,\n SP_MAT_ENTITY_CRUD_HTTP_CONTEXT,\n SPMatEntityCrudConfig,\n SPMatEntityCrudResponseParser,\n} from './mat-entity-crud-types';\nimport { PreviewHostComponent } from './preview-host.component';\n\n@Component({\n imports: [\n CommonModule,\n RouterModule,\n MatButtonModule,\n MatTableModule,\n MatSortModule,\n MatMenuModule,\n MatSnackBarModule,\n MatIconModule,\n TranslocoModule,\n AngularSplitModule,\n SPMatEntityListComponent,\n SPMatContextMenuComponent,\n FormViewHostComponent,\n SPMatHostBusyWheelDirective,\n PreviewHostComponent,\n ],\n providers: [provideTranslocoScope('sp-mat-entity-crud')],\n selector: 'sp-mat-entity-crud',\n template: `\n <as-split direction=\"horizontal\" [gutterSize]=\"6\" *transloco=\"let t\">\n <as-split-area\n [size]=\"entitiesPaneWidth()\"\n [visible]=\"!entitiesPaneHidden()\"\n >\n <div [class]=\"listPaneWrapperClass()\">\n <ng-content select=\"[breadCrumbs]\"></ng-content>\n\n <ng-template #defaultActionButtons>\n <div class=\"action-bar-actions\">\n @if (!disableCreate()) { @if (newItemSubTypes()) {\n <!-- New {{item}} displays a dropdown menu from which the subtype can be selected -->\n <button\n type=\"button\"\n mat-raised-button\n color=\"primary\"\n [matMenuTriggerFor]=\"newSubTypesMenu\"\n >\n {{\n newItemLabel() ??\n t('spMatEntityCrud.newItem', {\n item: _itemLabel() | async\n })\n }}\n <mat-icon>expand_circle_down</mat-icon>\n </button>\n <mat-menu #newSubTypesMenu=\"matMenu\">\n @for (subtype of newItemSubTypes(); track $index) { @if\n (subtype.role) {\n <button mat-menu-item (click)=\"handleNewItemSubType(subtype)\">\n {{ subtype.label }}\n </button>\n } @else {\n <div style=\"padding: .2em 0.5em;\">\n <strong>{{ subtype.label }}</strong>\n </div>\n } }\n </mat-menu>\n } @else {\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"onCreate($event)\"\n [routerLink]=\"newItemLink()\"\n >\n {{\n newItemLabel() ??\n t('spMatEntityCrud.newItem', {\n item: _itemLabel() | async\n })\n }}\n <mat-icon>add_circle</mat-icon>\n </button>\n } }\n </div>\n </ng-template>\n\n <ng-template #defaultHeaderTemplate>\n <div class=\"action-bar\">\n <div class=\"action-bar-title\">\n {{ _title() | async }}\n </div>\n <span class=\"spacer\"></span>\n <!-- Hide the action buttons when Preview/Edit pane is active -->\n @if (!entityPaneActive()) {\n <ng-container\n [ngTemplateOutlet]=\"actionsTemplate() || defaultActionButtons\"\n ></ng-container>\n }\n </div>\n </ng-template>\n <ng-container\n [ngTemplateOutlet]=\"headerTemplate() || defaultHeaderTemplate\"\n ></ng-container>\n <sp-mat-entity-list\n #entitiesList\n [entityName]=\"entityName()\"\n [entityNamePlural]=\"entityNamePlural()\"\n [deferViewInit]=\"true\"\n [endpoint]=\"endpoint()\"\n [entityLoaderFn]=\"entityLoaderFn()\"\n [columns]=\"columnsWithAction()\"\n [displayedColumns]=\"visibleColumns()\"\n [idKey]=\"idKey()\"\n [pagination]=\"pagination()\"\n [paginator]=\"paginator()\"\n [pageSize]=\"pageSize()\"\n [sorter]=\"sorter()\"\n [disableSort]=\"disableSort()\"\n (selectEntity)=\"handleSelectEntity($event)\"\n [httpReqContext]=\"httpReqContext()\"\n >\n </sp-mat-entity-list>\n </div>\n\n <!--\n We'll be initializing the contentColumnDefs separately and not\n relying on <sp-mat-entity-list>'s internal @ContentChildre() querylist\n for building this. So we define this independenly and not as\n <sp-mat-entity-list> content.\n -->\n <ng-container matColumnDef=\"action\">\n <th mat-header-cell *matHeaderCellDef></th>\n <td mat-cell *matCellDef=\"let element\">\n <!-- <button\n mat-icon-button\n hoverDropDown\n >\n <mat-icon>more_vert</mat-icon>\n </button> -->\n <sp-mat-context-menu\n [menuItems]=\"_itemActions\"\n (selected)=\"onItemAction($event, element)\"\n [contextData]=\"element\"\n [hasBackdrop]=\"true\"\n ></sp-mat-context-menu>\n </td>\n </ng-container>\n </as-split-area>\n <as-split-area [size]=\"entityPaneWidth()\" [visible]=\"entityPaneActive()\">\n <div\n [class]=\"'entity-pane-wrapper ' + previewPaneWrapperClass()\"\n spHostBusyWheel=\"formBusyWheel\"\n >\n <sp-entity-crud-preview-host\n [ngClass]=\"createEditViewActive() ? 'sp-hidden' : 'sp-visible'\"\n [entityCrudComponentBase]=\"this\"\n [clientViewTemplate]=\"previewTemplate()!\"\n ></sp-entity-crud-preview-host>\n <!-- Create/Edit Entity -->\n <sp-create-edit-entity-host\n [ngClass]=\"createEditViewActive() ? 'sp-visible' : 'sp-hidden'\"\n [entityCrudComponentBase]=\"this\"\n [clientViewTemplate]=\"createEditFormTemplate()\"\n ></sp-create-edit-entity-host>\n </div>\n </as-split-area>\n </as-split>\n `,\n styles: `\n .sp-hidden {\n display: none;\n }\n .sp-visible {\n display: inherit;\n height: 100% !important;\n width: 100% !important;\n }\n .entity-pane-wrapper {\n height: 100% !important;\n width: 100% !important;\n }\n .action-bar {\n display: flex;\n flex-direction: row;\n align-items: center;\n padding-bottom: 0.5em;\n }\n .action-bar-title {\n font-size: 1.5em;\n font-weight: 600;\n }\n .spacer {\n flex-grow: 1;\n }\n .action-bar-actions {\n text-align: end;\n }\n .active-row {\n font-weight: bold;\n }\n `,\n changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SPMatEntityCrudComponent<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n >\n extends SPMatEntityListComponent<TEntity, IdKey>\n implements SPMatEntityCrudComponentBase<TEntity, IdKey>, AfterViewInit\n{\n // entityName = input.required<string>();\n // entityNamePlural = input<string>();\n\n itemLabel = input<string | Observable<string>>();\n itemLabelPlural = input<string | Observable<string>>();\n\n /**\n * Title string displayed above the component. If not specified, will use\n * itemLabelPlural() as the title.\n */\n title = input<string | Observable<string>>();\n /**\n *\n */\n itemActions = input<MatEntityCrudItemAction<TEntity, IdKey>[]>([]);\n /**\n * Specify the list of router paths that will be set as the value for\n * [routerLink] for the 'New {{ item }}' button. If not specified,\n * if createEditTemplate is specified, it will be shown. If not, `action`\n * out event will be raised with `{ role: '_new_' }`.\n */\n newItemLink = input<string | string[]>();\n /**\n * If not specified, will default to 'New <itemLabel()>'.\n */\n newItemLabel = input<string | string[]>();\n /**\n * Text for the Edit <item> pane title\n */\n editItemTitle = input<string>();\n /**\n * If you want \"New {{item}}\" button to support multiple entity types,\n * you can set this to `NewItemSubType[]`, where each element stands for for\n * a dropdown menu item. Refer to `NewItemSubType` for details on this\n * interface.\n */\n newItemSubTypes = input<NewItemSubType[]>();\n /**\n * If you want to take control of the network operations for the CRUD\n * operations (GET/CREATE/UPDATE/DELETE), provide a value for this property.\n */\n crudOpFn = input<CRUD_OP_FN<TEntity, IdKey>>();\n /**\n * Item preview template.\n */\n previewTemplate = input<TemplateRef<any>>();\n /**\n * Whether to allow a context menu action or not. Return false to disable\n * the action.\n */\n allowEntityActionFn = input<ALLOW_ITEM_ACTION_FN<TEntity>>();\n /**\n * A template that allows the header to be replaced. Usage:-\n *\n * ```\n * <sp-map-entity-crud\n * [headerTemplate]=\"myCrudViewHeader\"\n * ></sp-map-entity-crud>\n * <ng-template #myCrudViewHeader>...</ng-template>\n * ```\n */\n headerTemplate = input<TemplateRef<any>>();\n /**\n * Set this to the custom template identifier that will replace the\n * \"New {{Item}}\" button portion. This template will expand towards the\n * title which will be placed to its left (right in rtl).\n *\n * ```\n * <sp-map-entity-crud\n * [actionsTemplate]=\"myCrudActions\"\n * ></sp-map-entity-crud>\n * <ng-template #myCrudActions>...</ng-template>\n * ```\n */\n actionsTemplate = input<TemplateRef<any>>();\n\n /**\n * CRUD action response parser. This will be called with the response\n * from CREATE & UPDATE operations to parse the response JSON and return\n * the created/updated TEntity.\n */\n crudResponseParser = input<SPMatEntityCrudResponseParser>();\n /**\n * An ng-template name that contains the component which provides the\n * create/edit CRUD action.\n *\n * ```\n * <ng-template #createEdit let-data>\n * <app-create-edit-entity-demo [bridge]=\"data.bridge\" [entity]=\"data.entity\"></app-create-edit-entity-demo>\n * </ng-template>\n * ```\n * Note how [bridge] & [entity] properties are set deriving them from the\n * implicit template context. [entity] will be undefined for Create\n * opreation and will be the valid entity for an Update.\n * (app-create-edit-entity-demo here is the client code that implements the\n * Create/Edit form)\n */\n createEditFormTemplate = input<TemplateRef<any> | null>(null);\n /**\n * Disables the per item actions column, preventing 'Edit' & 'Delete'\n * (and other user defined) item operations.\n */\n disableItemActions = input<boolean>(false);\n /**\n * Disables the Create function.\n */\n disableCreate = input<boolean>(false);\n /**\n * View refresh policy after a CREATE/UPDATE operation. Values\n * 'none' - Objects are not refreshed after an edit operation. The return\n * value of the edit operation is used as the object to\n * add/update the component's internal store. This is the default.\n * 'object' - Refresh just the object that was returned from the\n * CREATE/UPDATE operation. Use this if the JSON object returned\n * after a successful CREATE/UPDATE op differs from the JSON\n * object returned for the GET request.\n * 'all' - Refresh the entire list after a CREATE/UPDATE operation. This\n * mimics the behaviour of legacy HTML apps with pure server\n * defined UI.\n */\n refreshAfterEdit = input<'none' | 'object' | 'all'>('none');\n /**\n * HttpContext for crud requests - `create`, `retrieve`, `update` & `delete`.\n * Note that HttpContext for `list` operation should be set using the\n * `httpReqContext` property inherited from SPMatEntityListComponent.\n * The value of the property is an object where the key names reflect the CRUD\n * operation with each of these keys expected to have a value of type\n * `[[HttpContextToken<any>, any]] | [HttpContextToken<any>, any] | HttpContext`.\n *\n * Alternatively the property value can be set to\n * `[[HttpContextToken<any>, any]] | [HttpContextToken<any>, any] | HttpContext`,\n * in which case the same context would be used for all HTTP requests.\n */\n crudHttpReqContext = input<\n | [[HttpContextToken<any>, any]]\n | [HttpContextToken<any>, any]\n | {\n create?: [[HttpContextToken<any>, any]] | [HttpContextToken<any>, any]; // CREATE\n retrieve?:\n | [[HttpContextToken<any>, any]]\n | [HttpContextToken<any>, any]; // RETRIEVE\n update?: [[HttpContextToken<any>, any]] | [HttpContextToken<any>, any]; // UPDATE\n delete?: [[HttpContextToken<any>, any]] | [HttpContextToken<any>, any]; // DELETE\n }\n >();\n /**\n * Width of the edit pane as a percentange of the overall <as-split> width.\n */\n editPaneWidth = input<number>(100);\n /**\n * Width of the preview pane as a percentange of the overall <as-split> width.\n */\n previewPaneWidth = input<number>(50);\n\n /**\n * The class class that will be applied to the list pane wrapper.\n */\n listPaneWrapperClass = input<string>('sp-mat-crud-list-pane-wrapper-class');\n\n /**\n * The class class that will be applied to the preview pane wrapper.\n * Inside this wrapper, another div element with class='previewPaneContentClass()'\n * will be created to host the preview content. The client supplied preview\n * template will be rendered inside this div.\n *\n * +----------------------------------------+\n * | previewPaneWrapperClass |\n * |----------------------------------------|\n * | preview toolbar |\n * |----------------------------------------|\n * | +------------------------------------+ |\n * | | div class=previewPaneContentClass()| |\n * | |------------------------------------| |\n * | | client supplied preview template | |\n * | +------------------------------------+ |\n * +----------------------------------------+\n */\n previewPaneWrapperClass = input<string>(\n 'sp-mat-crud-entity-pane-wrapper-class'\n );\n\n /**\n * The class class that will be applied to the preview pane content.\n */\n previewPaneContentClass = input<string>(\n 'sp-mat-crud-preview-pane-content-class'\n );\n\n /**\n * The class class that will be applied to the form pane wrapper. Inside this\n * wrapper, another div element with class='formPaneContentClass()' will be\n * created to host the form content. The client supplied form template will\n * be rendered inside this div.\n *\n * +----------------------------------------+\n * | formPaneWrapperClass |\n * | +------------------------------------+ |\n * | | div class=formPaneContentClass() | |\n * | |------------------------------------| |\n * | | form title + close button | |\n * | |------------------------------------| |\n * | | client supplied form template | |\n * | +------------------------------------+ |\n * +----------------------------------------+\n */\n formPaneWrapperClass = input<string>('sp-mat-crud-form-pane-wrapper-class');\n\n /**\n * The CSS class that will be applied to the form pane content.\n */\n formPaneContentClass = input<string>('sp-mat-crud-form-pane-content-class');\n\n // INTERNAL PROPERTIES //\n // Derive a label from a camelCase source string. If the camelCase string\n // can be translated, it returns the translated string. If not, the function\n // converts the camelCase to 'Title Case' and returns it.\n private getLabel = (source: string) => {\n const label = this.transloco.translate(source);\n if (label.localeCompare(source) !== 0) {\n // Successful translation, return it\n return label;\n }\n return startCase(source);\n };\n\n _itemLabel = computed<Observable<string>>(() => {\n const itemLabel = this.itemLabel();\n const label = itemLabel ? itemLabel : this.getLabel(this.entityName());\n return label instanceof Observable ? label : of(label);\n });\n _itemLabelPlural = computed<Observable<string>>(() => {\n const itemLabelPlural = this.itemLabelPlural();\n const label = itemLabelPlural\n ? itemLabelPlural\n : this.getLabel(plural(this.entityName()));\n return label instanceof Observable ? label : of(label);\n });\n\n // Computed title\n _title = computed(() => {\n const title = this.title() ? this.title() : this._itemLabelPlural();\n return title instanceof Observable ? title : of(title);\n });\n // endpoint with the QP string removed (if one was provided)\n _endpointSansParams = computed(() => this.endpoint().split('?')[0]);\n _endpointParams = computed(() => {});\n componentColumns = viewChildren(MatColumnDef);\n @ContentChildren(MatColumnDef) _clientColumnDefs!: QueryList<MatColumnDef>;\n\n /**\n * Event raised for user selecting an item action. It's also raised\n * for 'New <Item>' action, if 'newItemLink' property is not set.\n */\n @Output() action = new EventEmitter<{ role: string; entity?: TEntity }>();\n\n /**\n * Event raised when create Create/Edit pane is activated & deactivated.\n * Event contains two flags:-\n * activated - whether the createEdit form view was activated or\n * deactivated.\n * cancelled - whether the form view was cancelled by user. False for this\n * indicates that the form view was closed after a successful\n * edit operation.\n */\n @Output() entityViewPaneActivated = new EventEmitter<{\n activated: boolean;\n cancelled: boolean | undefined;\n mode: 'edit' | 'preview';\n }>();\n\n busyWheelId = `entityCrudBusyWheel-${Date.now()}`;\n sub$ = new Subscription();\n spEntitiesList =\n viewChild<SPMatEntityListComponent<TEntity, IdKey>>('entitiesList');\n\n // Theoritically, we ought to be able to initialize the mat-entities-list\n // columns from ngAfterViewInit lifecycle hook. But for some strange reason\n // when this hook is called, sp-mat-entities-list has not been initialized.\n // Therefore `spEntitiesList()` is null. So we have to use a computed signal,\n // which will be triggered when spEntitiesList() is initialized and use that\n // to initialize the columns.\n spEntitiesListInited = effect(() => {\n if (this.spEntitiesList()) {\n this._initEntitiesList();\n }\n });\n\n crudConfig!: SPMatEntityCrudConfig;\n\n // This is the internal component that will host the createEditFormTemplate\n createEditHostComponent = viewChild(FormViewHostComponent);\n // A flag to toggle the viewport's contents between the mat-table\n // and the create/edit form template.\n createEditViewActive = signal<boolean>(false);\n\n // Whether it's okay to cancel the edit\n canCancelEditCallback!: () => boolean;\n\n // Preview host component\n previewHostComponent = viewChild(PreviewHostComponent);\n previewActive = computed(() => this.previewedEntity() !== undefined);\n previewedEntity = signal<TEntity | undefined>(undefined);\n\n // Whether the pane that hosts the preview/edit-entity template is active.\n // We call it entityPane as it's used to either render a selected entity\n // or to edit one.\n entityPaneActive = computed(\n () => !!this.previewedEntity() || this.createEditViewActive()\n );\n // Effective width of the entity pane.\n entityPaneWidth = computed(() =>\n this.entityPaneActive()\n ? !!this.previewedEntity()\n ? this.previewPaneWidth()\n : this.editPaneWidth()\n : 0\n );\n\n // Width of the pane showing the list of entities. Calculated as\n entitiesPaneWidth = computed(() => 100 - this.entityPaneWidth());\n entitiesPaneHidden = computed(\n () => this.entityPaneActive() && this.entityPaneWidth() === 100\n );\n\n defaultItemCrudActions = signal<MatEntityCrudItemAction<TEntity, IdKey>[]>(\n []\n );\n columnsWithAction = computed(() => {\n const cols = clone(this.columns());\n const actionDefined =\n cols.find((c) =>\n typeof c === 'string' ? c === 'action' : c.name === 'action'\n ) !== undefined;\n if (!actionDefined) {\n cols.push('action');\n }\n return cols;\n });\n // Provide per entity actions as a function so that the actions are\n // enumerated only when the user clicks on the context menu button.\n _itemActions = (entity: TEntity) => this.getItemActions(entity);\n\n // This uses the previewActive signal to compute the visible columns\n // when preview is activated. For now we just hide the 'action' column when\n // preview is active. We can further customize this logic by allowing the\n // client to specify the columns to display when preview is active thereby\n // reducing column clutter when the table width becomes narrower owing to\n // preview pane taking up screen space.\n visibleColumns = computed(() =>\n this.entityPaneActive()\n ? this.columnsWithAction()\n .map((col) => (typeof col === 'string' ? col : col.name))\n .filter((name) => name !== 'action')\n : []\n );\n transloco = inject(TranslocoService);\n\n constructor(\n http: HttpClient,\n private snackBar: MatSnackBar,\n sanitizer: DomSanitizer,\n injector: Injector\n ) {\n super(http, sanitizer, injector);\n this.crudConfig = getEntityCrudConfig();\n if (this.crudConfig?.defaultItemActions) {\n this.defaultItemCrudActions.set(this.crudConfig?.defaultItemActions);\n } else {\n this.defaultItemCrudActions.set([\n {\n label: this.transloco.translate('spMatEntityCrud.edit'),\n role: '_update_',\n },\n {\n label: this.transloco.translate('spMatEntityCrud.delete'),\n role: '_delete_',\n },\n ]);\n }\n }\n\n override ngOnInit() {}\n\n override ngOnDestroy(): void {\n this.sub$.unsubscribe();\n }\n\n /**\n * Override so that we can suppress default action in SPMatEntityListComponent\n */\n override ngAfterViewInit(): void {}\n\n /**\n * If the create/edit entity form is active, it calls its registered\n * canCancelEdit callback to determine if it's okay to cancel the edit.\n * You can use this method from the host component's router guard to\n * ensure that any changes made to the form are not accidentally lost by\n * navigating away from the CRUD page.\n *\n * If your CRUD page has multiple sp-mat-entity-crud components, you have to\n * implement the logic to call this method on the appropriate component.\n *\n * If the the create/edit form is not active, this method returns true.\n * @returns\n */\n canDeactivate() {\n if (this.createEditViewActive()) {\n return this.canCancelEdit();\n }\n return true;\n }\n\n override refresh(force = false) {\n this.spEntitiesList()?.refresh(force);\n }\n\n // BEGIN SPMatEntityCrudComponentBase METHODS //\n getEntityName(): string {\n return this.entityName()\n }\n\n getEntityNamePlural(): string {\n return this._entityNamePlural();\n }\n\n getIdKey() {\n return this.idKey();\n }\n\n closeCreateEdit(cancelled: boolean) {\n this.createEditViewActive.set(false);\n this.entityViewPaneActivated.emit({\n activated: false,\n cancelled: !!cancelled,\n mode: 'edit',\n });\n }\n\n canCancelEdit() {\n if (this.canCancelEditCallback) {\n return this.canCancelEditCallback();\n }\n return true;\n }\n\n registerCanCancelEditCallback(callback: () => boolean) {\n this.canCancelEditCallback = callback;\n }\n\n triggerEntityUpdate(entity: TEntity) {\n this.onUpdate(entity);\n }\n\n triggerEntityDelete(entity: TEntity) {\n this.onDelete(entity);\n }\n\n create(entityValue: any) {\n let obs!: Observable<TEntity | null>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn('create', undefined, entityValue, this);\n } else {\n obs = this.http.post<TEntity>(this.getUrl(this.endpoint()), entityValue, {\n context: this.getCrudReqHttpContext('create'),\n });\n }\n\n return obs.pipe(\n showBusyWheelUntilComplete('formBusyWheel'),\n switchMap((resp) =>\n resp ? this.doRefreshAfterEdit(resp, 'create') : of(null)\n ),\n tap((entity) => {\n // If pagination is infinite or if the pagination if none or if the\n // count of items in the current page is less than pageSize()\n // wec an safely add the item to the list, which will cause the view\n // render the new item in the mat-table.\n if (entity) {\n this.spEntitiesList()?.addEntity(entity);\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n this.snackBar.open(\n this.transloco.translate('spMatEntityCrud.createSuccess', {\n item: itemLabel,\n })\n );\n });\n }\n })\n );\n }\n\n update(id: TEntity[IdKey], entityValue: any) {\n let obs!: Observable<TEntity | null>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn('update', id, entityValue, this);\n } else {\n obs = this.http.patch<TEntity>(this.getEntityUrl(id), entityValue, {\n context: this.getCrudReqHttpContext('update'),\n });\n }\n\n return obs.pipe(\n showBusyWheelUntilComplete('formBusyWheel'),\n switchMap((resp) =>\n resp ? this.doRefreshAfterEdit(resp, 'update') : of(null)\n ),\n tap((entity) => {\n if (entity) {\n this.spEntitiesList()?.updateEntity(id, entity);\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n this.snackBar.open(\n this.transloco.translate('spMatEntityCrud.updateSuccess', {\n item: itemLabel,\n })\n );\n });\n }\n })\n );\n }\n\n loadEntity(\n id: TEntity[IdKey],\n params: string | HttpParams\n ): Observable<TEntity> {\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n return crudOpFn('get', id, undefined, this) as Observable<TEntity>;\n } else {\n const httpParams =\n params instanceof HttpParams\n ? params\n : new HttpParams({ fromString: params });\n return this.http.get<TEntity>(this.getEntityUrl(id), {\n context: this.getCrudReqHttpContext('retrieve'),\n params: httpParams,\n });\n }\n }\n // END SPMatEntityCrudComponentBase METHODS //\n\n /**\n * Thunk these methods to the internal <sp-mat-entity-list> component.\n */\n override addEntity(entity: TEntity): void {\n this.spEntitiesList()?.addEntity(entity);\n }\n\n override removeEntity(id: TEntity[IdKey]): void {\n this.spEntitiesList()?.removeEntity(id);\n }\n\n override updateEntity(id: TEntity[IdKey], entity: TEntity): void {\n this.spEntitiesList()?.updateEntity(id, entity);\n }\n\n /**\n * Refresh the entity list, after a CRUD CREATE or UPDATE operation.\n * @param resp This is the response from the CRUD operation (CREATE/UPDATE).\n * @param method The CRUD operation post which REFRESH is requested.\n * @returns Observable<TEntity|null>\n */\n doRefreshAfterEdit(resp: any, method: 'create' | 'update') {\n const refreshAfterEdit = this.refreshAfterEdit();\n const entity = this.getCrudOpResponseParser()(\n this.entityName(),\n this.idKey(),\n method,\n resp\n );\n if (refreshAfterEdit === 'object') {\n let obs!: Observable<TEntity>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn(\n 'get',\n (entity as any)[this.idKey()],\n undefined,\n this\n ) as Observable<TEntity>;\n } else {\n obs = this.http.get<TEntity>(\n this.getEntityUrl((entity as any)[this.idKey()]),\n { context: this.getCrudReqHttpContext('retrieve') }\n );\n }\n return obs.pipe(\n map((entity) => {\n return this.getCrudOpResponseParser()(\n this.entityName(),\n this.idKey(),\n 'retrieve',\n entity\n );\n })\n );\n } else if (refreshAfterEdit === 'all') {\n this.refresh(true);\n return of(null);\n }\n\n return of(entity);\n }\n\n getCrudOpResponseParser(): SPMatEntityCrudResponseParser {\n if (this.crudResponseParser()) {\n // Without the `as SPMatEntityCrudResponseParser`, TSC will complain.\n return this.crudResponseParser() as SPMatEntityCrudResponseParser;\n }\n // crudConfig will have a parser as our default config provides one.\n return this.crudConfig\n .crudOpResponseParser as SPMatEntityCrudResponseParser;\n }\n\n // SPMatEntityCrudComponentBase interface method. Thunk to the implementation\n // method closePreviewImpl().\n closePreview() {\n this.closePreviewImpl(true);\n }\n\n private closePreviewImpl(toggleEntityListActiveEntity: boolean) {\n if (this.previewedEntity()) {\n if (toggleEntityListActiveEntity) {\n this.spEntitiesList()?.toggleActiveEntity(this.previewedEntity());\n }\n this.previewedEntity.set(undefined);\n this.entityViewPaneActivated.emit({\n activated: false,\n cancelled: undefined,\n mode: 'preview',\n });\n }\n }\n\n onItemAction(role: string, entity: TEntity) {\n // Handle default roles, which always get default behavior. If you want\n // a custom behavior, change the role to a custom one.\n if (role === '_update_' || role === '_delete_') {\n role === '_update_' ? this.onUpdate(entity) : this.onDelete(entity);\n return;\n }\n\n // Find the SPMatEntityCrudItemAction for this role to see if it's\n // allowed.\n const actionItem = this.itemActions().find((a) => a.role === role);\n if (!actionItem) {\n return;\n }\n\n // If a custom action handler is specified, call it.\n if (actionItem?.action) {\n actionItem.action(entity);\n return;\n }\n\n // Perform custom HTTP action\n const httpRequestParameters: MatEntityCrudItemAction<\n TEntity,\n IdKey\n >['httpRequestParameters'] = actionItem?.httpRequestParameters || {\n method: 'POST',\n urlPath: actionItem.role,\n };\n const verb = httpRequestParameters?.urlPath || actionItem?.role;\n if (!verb) {\n return;\n }\n\n // If a confirm prompt is specified, display it and return if user\n // selects 'Cancel'.\n if (actionItem?.confirmPrompt) {\n if (!confirm(actionItem.confirmPrompt)) {\n return;\n }\n }\n\n this.doEntityAction(\n (entity as any)[this.idKey()],\n verb,\n httpRequestParameters.params || new HttpParams(),\n httpRequestParameters.body\n )\n .pipe(\n tap((response) => {\n const successMessage =\n actionItem?.successMessage ||\n this.transloco.translate('spMatEntityCrud.done');\n this.snackBar.open(successMessage || 'Done');\n }),\n catchError((error) => {\n /**\n * If an errorMessage is specified in the actionItem, display it.\n * Otherwise rethrow the error so that it can be handled by the\n * global error handler.\n */\n if (actionItem?.errorMessage) {\n this.snackBar.open(actionItem.errorMessage);\n return EMPTY;\n }\n return throwError(() => error);\n })\n )\n .subscribe();\n }\n\n onCreate(event: Event) {\n // If newItemLink() has not been provided, check if createEditFormTemplate\n // is specified. If it is, load it and make that cover the entire viewport.\n // If that too is not specified, emit an action event with role='_new_'.\n if (!this.newItemLink() || this.newItemLink()?.length == 0) {\n event.preventDefault();\n event.stopImmediatePropagation();\n this.showCreateEditView(undefined);\n if (!this.createEditViewActive()) {\n this.action.emit({ role: '_new_' });\n }\n }\n }\n\n onUpdate(entity: TEntity) {\n this.showCreateEditView(entity);\n if (!this.createEditViewActive()) {\n this.action.emit({ role: '_update_' });\n }\n }\n\n /**\n * Show the create/edit component. This is deliberately made public so as to\n * be callable from the client. This allows the client to dynamically\n * set the form edit template and then show the edit pane by calling this\n * method.\n * @param entity\n * @param params\n */\n showCreateEditView(entity?: TEntity | undefined, params?: any) {\n const tmpl = this.createEditFormTemplate();\n if (!this.createEditViewActive() && tmpl) {\n // If preview is active deactivate it\n if (this.previewActive()) {\n this.closePreviewImpl(true);\n }\n const createEditHost = this.createEditHostComponent();\n createEditHost!.show(entity, params);\n this.createEditViewActive.set(true);\n this.entityViewPaneActivated.emit({\n activated: true,\n cancelled: undefined,\n mode: 'edit',\n });\n }\n }\n\n showPreviewView(entity?: TEntity, params?: any) {\n const tmpl = this.previewTemplate();\n if (tmpl) {\n if (!this.createEditViewActive()) {\n const previewHost = this.previewHostComponent();\n this.previewedEntity.set(entity);\n previewHost?.show(entity, params);\n this.entityViewPaneActivated.emit({\n activated: true,\n cancelled: undefined,\n mode: 'preview',\n });\n // this.previewActivated.emit({ entity, activated: true });\n }\n }\n }\n\n hidePreviewView() {\n if (this.previewActive()) {\n const previewHost = this.previewHostComponent();\n previewHost?.close();\n this.closePreviewImpl(false);\n }\n }\n\n async onDelete(entity: TEntity) {\n // Do the delete prompt asynchronously so that the context menu is\n // dismissed before the prompt is displayed.\n setTimeout(() => {\n // We use firstValueFrom() to get the value of the observable\n // synchronously. firstValueFrom() also gracefully cleans up the\n // observable after a value is emitted.\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n const deletedItemPrompt = this.transloco.translate(\n 'spMatEntityCrud.deleteItemConfirm',\n { item: itemLabel.toLocaleLowerCase() }\n );\n const yes = confirm(deletedItemPrompt);\n if (yes) {\n const entityId = (entity as any)[this.idKey()];\n\n // If preview is active deactivate it\n if (this.previewActive()) {\n this.closePreviewImpl(false);\n }\n\n let obs!: Observable<any>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn('delete', entityId, undefined, this);\n } else {\n obs = this.http.delete<void>(this.getEntityUrl(entityId), {\n context: this.getCrudReqHttpContext('delete'),\n });\n }\n\n this.sub$.add(\n obs\n .pipe(\n // TODO: how to display a busy wheel?\n // showBusyWheelUntilComplete(this.busyWheelId),\n tap(() => {\n this.spEntitiesList()!.removeEntity(entityId);\n // TODO: customize by providing an interface via SPMatEntityCrudConfig?\n firstValueFrom(this._itemLabel()).then((itemLabel) => {\n const deletedMessage = this.transloco.translate(\n 'spMatEntityCrud.deleteItemSuccess',\n { item: itemLabel }\n );\n this.snackBar.open(deletedMessage);\n });\n })\n )\n .subscribe()\n );\n }\n });\n });\n }\n\n override getUrl(endpoint: string) {\n return this.entityListConfig?.urlResolver\n ? this.entityListConfig?.urlResolver(endpoint)\n : endpoint;\n }\n\n getEntityUrl(entityId: TEntity[IdKey]) {\n const endpoint = this.endpoint();\n const endpointParts = endpoint.split('?');\n const entityEndpoint =\n (endpointParts[0].endsWith('/')\n ? endpointParts[0]\n : endpointParts[0] + '/') + `${String(entityId)}/`;\n if (endpointParts.length > 1) {\n return this.getUrl(entityEndpoint + `?${endpointParts[1]}`);\n }\n return this.getUrl(entityEndpoint);\n }\n\n getEntityActionUrl(entityId: TEntity[IdKey], action: string) {\n const url = this.getEntityUrl(entityId);\n const urlParts = url.split('?');\n if (action.endsWith('/')) {\n action = action.slice(0, -1); // We'll be adding the trailing slash\n }\n const actionUrl =\n (urlParts[0].endsWith('/') ? urlParts[0] : urlParts[0] + '/') +\n `${action}/`;\n return urlParts.length === 1 ? actionUrl : actionUrl + `?${urlParts[1]}`;\n }\n\n handleSelectEntity(entity: TEntity | undefined) {\n if (!this.createEditViewActive()) {\n if (this.previewTemplate()) {\n entity ? this.showPreviewView(entity) : this.hidePreviewView();\n // this.previewedEntity.set(entity);\n } else {\n // If 'previewTemplate' is not set, propagate the event to client.\n this.selectEntity.emit(entity);\n }\n }\n }\n\n handleNewItemSubType(subtype: NewItemSubType) {\n // console.log(`handleNewItemSubType: ${subtype}`);\n if (subtype.role === '_new_') {\n this.showCreateEditView(undefined, subtype?.params);\n } else {\n this.action.emit({ role: subtype.role });\n }\n }\n\n private getCrudReqHttpContext(op: CrudOp) {\n\n let context = new HttpContext();\n // HttpContext for crud operations are taken from either the global httpReqContext\n // or from the crudHttpReqContext, with the latter taking precedence.\n const crudHttpReqContext = this.crudHttpReqContext()\n ? this.crudHttpReqContext()\n : this.httpReqContext();\n if (crudHttpReqContext) {\n if (crudHttpReqContext instanceof HttpContext) {\n // crudHttpReqContext is an object of type HttpContext. Which means\n // the same context is to be applied to all crud requests.\n for (const k of crudHttpReqContext.keys()) {\n context.set(k, crudHttpReqContext.get(k));\n }\n } else {\n if (Array.isArray(crudHttpReqContext)) {\n // Same HttpContext for all crud requests. Being an array, it must\n // be an array of HttpContextToken key, value pairs.\n convertHttpContextInputToHttpContext(context, crudHttpReqContext);\n } else if (\n typeof crudHttpReqContext === 'object' &&\n op &&\n Object.keys(crudHttpReqContext).find((k) => k === op)\n ) {\n // HttpContext specific to this crud operation, 'create'|'retrieve'|'update'|'delete'\n convertHttpContextInputToHttpContext(\n context,\n crudHttpReqContext[op]!\n );\n }\n }\n }\n\n // Add standard SP_MAT_ENTITY_CRUD_HTTP_CONTEXT info which is set for all\n // HTTP requests made by this component.\n context.set(SP_MAT_ENTITY_CRUD_HTTP_CONTEXT, {\n entityName: this.entityName(),\n entityNamePlural: this._entityNamePlural(),\n endpoint: this.endpoint(),\n op,\n });\n return context;\n }\n\n isItemActionAllowed(action: string, entity: TEntity) {\n return false;\n }\n\n /**\n * Returns the list of item actions. Calls 'allowItemActionFn' for each action\n * to determine if the action is allowed for the given entity.\n * @returns\n */\n getItemActions(entity: TEntity): MatEntityCrudItemAction<TEntity, IdKey>[] {\n // console.log(`SPMatEntityCrudComponent.getItemActions - entity: ${JSON.stringify(entity, null, 2)}`);\n const actions =\n this.itemActions() && this.itemActions().length\n ? this.itemActions()\n : this.defaultItemCrudActions();\n let actionsCopy: MatEntityCrudItemAction<TEntity, IdKey>[] = clone(actions);\n actionsCopy.forEach(\n (action: MatEntityCrudItemAction<TEntity, IdKey>, index: number) => {\n // localize default action item labels (Update & Delete)\n // Client specified action labels are to be localized by the client\n // before supplying them to the component.\n if (action.label.startsWith('spMatEntityCrud.')) {\n action.label = this.transloco.translate(action.label);\n }\n const orgDisable = actions[index]?.disable;\n action.disable = (entity: TEntity) => {\n if (orgDisable) {\n return orgDisable(entity);\n }\n const allowItemActionFn = this.allowEntityActionFn();\n if (allowItemActionFn) {\n return !allowItemActionFn(entity, action.role ?? action.label);\n }\n return false;\n };\n }\n );\n // If the item actions are disabled, disable all actions. Event user\n // defined actions.\n if (this.disableItemActions()) {\n actionsCopy.forEach((a) => (a.disable = () => true));\n }\n return actionsCopy;\n }\n\n getPreviewPaneContentClass() {\n return this.previewPaneContentClass();\n }\n\n getFormPaneWrapperClass() {\n return this.formPaneWrapperClass();\n }\n\n getFormPaneContentClass(): string {\n return this.formPaneContentClass();\n }\n\n getItemLabel(): string | Observable<string> {\n return this._itemLabel();\n }\n\n getItemLabelPlural(): string | Observable<string> {\n return this._itemLabelPlural();\n }\n\n /**\n * Perform an action on the entity with the given id. The endpoint for the\n * action is derived from the `verb` argument which is appended to the\n * entity URL. This is following DRF specification where ViewSets can be\n * extended with custom actions that are not part of the standard\n * CRUD operations. Such methods will have a URL like\n * `/api/v1/entity/<id>/<verb>/` where `<verb>` is the custom action verb.\n * @param id\n * @param verb\n * @param addlParams\n * @param data\n */\n doEntityAction(\n id: TEntity[IdKey],\n verb: string,\n addlParams: HttpParams,\n data: any,\n busyWheelName: string = 'formBusyWheel'\n ) {\n let obs!: Observable<TEntity | null>;\n const crudOpFn = this.crudOpFn();\n if (crudOpFn) {\n obs = crudOpFn(verb, id, data, this);\n } else {\n const url = this.getEntityActionUrl(id, verb);\n obs = this.http.post<TEntity>(url, data, {\n params: addlParams || {},\n context: this.getCrudReqHttpContext('update'), // KLUDGE!: use 'update' request context\n });\n }\n\n return obs.pipe(\n showBusyWheelUntilComplete(busyWheelName),\n switchMap((resp) =>\n resp ? this.doRefreshAfterEdit(resp, 'update') : of(null)\n ),\n tap((entity) => {\n if (entity) {\n this.spEntitiesList()?.updateEntity(id, entity);\n }\n })\n );\n }\n\n /**\n * Initialize the columns for the mat-entities-list component. This is\n * called when the <sp-mat-entities-list> component has been properly\n * initialized.\n */\n private _initEntitiesList() {\n const spEntitiesList = this.spEntitiesList();\n if (spEntitiesList) {\n // Build contentColumnDefs using our component's content. Then add our own\n // 'action' column definition to it. Then set this as the value of\n // child SPMatEntityListComponent.contentColumnDef. This way we force\n // SPMatEntityListComponent to use our component's any project MatColumnDef\n // content in the final mat-table.\n const clientColumnDefs = this.clientColumnDefs;\n let contentColumnDefs = new Array<MatColumnDef>();\n if (clientColumnDefs.length) {\n // Note that we process any content projected matColumnDef first and\n // our own internal content later. And when we process our own internal\n // columns (for now only 'action'), it's not added if a column with that\n // name has already been defined via content projection. This allows the\n // clients to override even internal columns with their column defintion.\n clientColumnDefs.toArray().forEach((c) => contentColumnDefs.push(c));\n }\n this.componentColumns().forEach((ic) => {\n if (!contentColumnDefs.find((c) => c.name === ic.name)) {\n contentColumnDefs.push(ic);\n }\n });\n spEntitiesList.contentColumnDefs = contentColumnDefs;\n // This is a replication of SPMatEntityCrudList.ngAfterViewInit. That\n // code is skipped as we declare <sp-mat-entity-list> with\n // deferViewInit=true.\n spEntitiesList.buildColumns();\n spEntitiesList.setupSort();\n spEntitiesList.loadMoreEntities();\n }\n }\n}\n","import { HttpParams } from \"@angular/common/http\";\nimport { SPContextMenuItem } from \"@smallpearl/ngx-helper/mat-context-menu\";\nimport { Observable } from \"rxjs\";\n\nexport const ITEM_ACTION_UPDATE = '_update_';\nexport const ITEM_ACTION_DELETE = '_delete_';\n\n/**\n * SPMatEntityCrudCreateEditBridge implementer uses this interface to\n * communicate with the parent SPMatEntityCreateComponent. The bridge\n * component would use the hideCreateEdit() to close itself, when user cancels\n * the create/edit operation.\n */\nexport interface SPMatEntityCrudComponentBase<\n TEntity extends { [P in IdKey]: PropertyKey },\n IdKey extends string = 'id'\n> {\n /**\n * Wrappers around entityName & entityNamePlural properties.\n */\n getEntityName(): string;\n getEntityNamePlural(): string;\n /**\n * Wrapper around idKey property.\n */\n getIdKey(): string;\n /**\n * This method should return the entity URL for the given entity id.\n * The entity URL should include any additional query parameters that were\n * passed to the SPMatEntityCrudComponentBase component's `url` property.\n * @param id\n */\n getEntityUrl(id: TEntity[IdKey]|undefined): string;\n /**\n * FormViewHostComponent will call this to close the Create/Edit pane.\n * SPMatEntityCrudComponentBase implementor will destroy the client form\n * view and hide the Create/Edit form view pane and show the hidden\n * entity list view.\n * @returns\n */\n closeCreateEdit: (cancelled: boolean) => void;\n /**\n * Used internally by FormViewHostComponent to determine if the client form\n * view wants to intercept user's cancel the create/edit operation. Perhaps\n * with the Yes/No prompt 'Lose changes?'\n * @returns boolean indicating it's okay to cancel the create/edit operation.\n */\n canCancelEdit: () => boolean;\n /**\n * Client form view can register a callback that will be invoked by the\n * framework when user cancels the create/edit operation by clicking on the\n * close button on the top right.\n * @param callback\n * @returns\n */\n registerCanCancelEditCallback: (callback: () => boolean) => void;\n /**\n * Initiates update on the given entity.\n * @returns\n */\n triggerEntityUpdate: (entity: TEntity) => void;\n /**\n * Initiates entity delete.\n * @returns\n */\n triggerEntityDelete: (entity: TEntity) => void;\n /**\n * Called by client form-view host component to close a new entity.\n * @param entityValue The ReactiveForm.value object that the server expects\n * to create a new object.\n * @returns The new Entity object returned by the server. For typical REST\n * API, this would be of the same shape as the objects returned by the\n * REST's GET request.\n */\n create: (entityValue: any) => Observable<any>;\n /**\n * Called by client form-view host component to close a new entity.\n * @param id The id of the entity being edited.\n * @param entityValue The ReactiveForm.value object that the server expects\n * to update the new object.\n * @returns The new Entity object returned by the server. For typical REST\n * API, this would be of the same shape as the objects returned by the\n * REST's GET request.\n */\n update: (id: any, entityValue: any) => Observable<any>;\n /**\n * Load the entity with the given id from server.\n * @param id The id of the entity to load.\n * @param params Additional parameters for loading the entity.\n * @returns An observable of the loaded entity.\n */\n loadEntity: (id: any, params: string | HttpParams) => Observable<TEntity>;\n /**\n * Close the preview pane.\n * @returns\n */\n closePreview: () => void;\n /**\n * Returns the context menu items for the entity. This can be used to build\n * the context menu for an entity in its preview pane toolbar.\n * @returns\n */\n getItemActions(entity?: TEntity): SPContextMenuItem[];\n /**\n * Returns the class to be used for the preview pane content. This interface\n * is provided to allow the PreviewPaneComponent to access the client\n * configured class for the preview pane content.\n */\n getPreviewPaneContentClass(): string;\n\n getFormPaneWrapperClass(): string;\n\n getFormPaneContentClass(): string;\n\n getItemLabel(): string | Observable<string>;\n\n getItemLabelPlural(): string | Observable<string>;\n /*\n * Remove the entity with the given id from the list of entities.\n * This is typically called by the client when it peforms the delete\n * operation itself without using the MatEntityCrud's delete operation.\n *\n * @param id The id of the entity to remove.\n * @returns None\n **/\n removeEntity(id: TEntity[IdKey]): void;\n /**\n * Update the entity with the given id in the list of entities.\n * This is typically called by the client when it has performed an update\n * on the entity itself and want to reflect the resulting changed\n * entity in the list view.\n *\n * @param id The id of the entity to update.\n * @param data The updated entity.\n */\n updateEntity(id: TEntity[IdKey], data: TEntity): void;\n /**\n * Perform a custom action on the entity endpoint. The action is specified\n * by the verb argument, which will be used to derive the final URL. This\n * is keeping in line with DRF specification where viewsets can define\n * custom action methods, which translate into endpoints with the same name\n * ast he action method.\n * @param id id of the entity to perform the action on.\n * @param verb The action verb, which will be appended to the entity URL to\n * derive the final URL for the POST request.\n * @param addlParams additional query parameters to include in the request.\n * Called `additional` as these are in addition to the query params specified\n * in the CRUD's endpoint.\n * @param data the data to send with the request for the POST\n * @returns Observable<TEntity>\n */\n doEntityAction(\n id: TEntity[IdKey],\n verb: string,\n addlParams: HttpParams,\n data: any,\n busyWheelName: string\n ): Observable<any>;\n}\n","import {\n ChangeDetectionStrategy,\n Component,\n computed,\n input,\n InputSignal,\n OnDestroy,\n OnInit,\n} from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatToolbarModule } from '@angular/material/toolbar';\nimport { SPContextMenuItem } from '@smallpearl/ngx-helper/mat-context-menu';\nimport {\n ITEM_ACTION_DELETE,\n ITEM_ACTION_UPDATE,\n SPMatEntityCrudComponentBase,\n} from './mat-entity-crud-internal-types';\n\n/**\n * A preview pane container to provide a consistent UX for all preview panes.\n * It consits of a toolbar on the top and a container div below that takes up\n * the rest of the preview pane area.\n */\n@Component({\n imports: [MatToolbarModule, MatButtonModule, MatIconModule],\n selector: 'sp-mat-entity-crud-preview-pane',\n template: `\n <div class=\"preview-wrapper\">\n <mat-toolbar>\n <mat-toolbar-row>\n @if (title()) {\n <h2>{{ title() }}</h2>\n } @if (!hideUpdate()) {\n <button\n mat-icon-button\n aria-label=\"Edit\"\n (click)=\"onEdit()\"\n [disabled]=\"_disableUpdate()\"\n >\n <mat-icon>edit</mat-icon>\n </button>\n } @if (!hideDelete()) {\n <button\n mat-icon-button\n aria-label=\"Delete\"\n (click)=\"onDelete()\"\n [disabled]=\"_disableDelete()\"\n >\n <mat-icon>delete</mat-icon>\n </button>\n }\n <ng-content select=\"[previewToolbarContent]\"></ng-content>\n <span class=\"spacer\"></span>\n <button mat-icon-button aria-label=\"Close\" (click)=\"onClose()\">\n <mat-icon>close</mat-icon>\n </button>\n </mat-toolbar-row>\n </mat-toolbar>\n <div\n [class]=\"\n 'preview-content ' +\n entityCrudComponent().getPreviewPaneContentClass()\n \"\n >\n <ng-content select=\"[previewContent]\"></ng-content>\n </div>\n </div>\n `,\n styles: [\n `\n .preview-wrapper {\n display: flex;\n flex-direction: column;\n height: 100% !important;\n width: 100% !important;\n }\n mat-toolbar {\n background-color: var(--mat-sys-surface-variant);\n }\n .spacer {\n flex: 1 1 auto;\n }\n .preview-content {\n padding: 0.4em;\n flex-grow: 1;\n overflow: scroll;\n }\n `,\n ],\n changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SPMatEntityCrudPreviewPaneComponent<TEntity extends { [P in IdKey]: PropertyKey }, IdKey extends string = 'id'>\n implements OnInit, OnDestroy\n{\n entity = input.required<TEntity>();\n entityCrudComponent = input.required<SPMatEntityCrudComponentBase<TEntity, IdKey>>();\n title = input<string>();\n disableUpdate = input<boolean>(false);\n hideUpdate = input<boolean>(false);\n disableDelete = input<boolean>(false);\n hideDelete = input<boolean>(false);\n itemActions!: SPContextMenuItem[];\n\n _disableActionFactory = (role: string, signal?: InputSignal<boolean>) => {\n return computed(() => {\n if (signal && signal()) {\n return true;\n }\n const actionFn = this.itemActions.find((a) => a.role === role)?.disable;\n if (actionFn && actionFn(this.entity())) {\n return true;\n }\n return false;\n });\n };\n _disableUpdate = this._disableActionFactory(\n ITEM_ACTION_UPDATE,\n this.disableUpdate\n );\n _disableDelete = this._disableActionFactory(\n ITEM_ACTION_DELETE,\n this.disableDelete\n );\n\n ngOnInit() {\n this.itemActions = this.entityCrudComponent().getItemActions();\n }\n\n ngOnDestroy(): void {}\n\n onEdit() {\n this.entityCrudComponent().triggerEntityUpdate(this.entity());\n }\n\n onDelete() {\n this.entityCrudComponent().triggerEntityDelete(this.entity());\n }\n\n onClose() {\n this.entityCrudComponent().closePreview();\n }\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":["i2","i3","i6","i9","i10","i1"],"mappings":"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"}
|
|
@@ -130,8 +130,8 @@ import * as i0 from "@angular/core";
|
|
|
130
130
|
export declare abstract class SPMatEntityCrudFormBase<TFormGroup extends AbstractControl, TEntity extends {
|
|
131
131
|
[P in IdKey]: PropertyKey;
|
|
132
132
|
}, IdKey extends string = 'id'> implements OnInit, OnDestroy {
|
|
133
|
-
entity: import("@angular/core").InputSignal<TEntity | TEntity[IdKey]>;
|
|
134
|
-
bridge: import("@angular/core").InputSignal<SPMatEntityCrudCreateEditBridge>;
|
|
133
|
+
entity: import("@angular/core").InputSignal<TEntity | TEntity[IdKey] | undefined>;
|
|
134
|
+
bridge: import("@angular/core").InputSignal<SPMatEntityCrudCreateEditBridge | undefined>;
|
|
135
135
|
params: import("@angular/core").InputSignal<any>;
|
|
136
136
|
entityName: import("@angular/core").InputSignal<string | undefined>;
|
|
137
137
|
baseUrl: import("@angular/core").InputSignal<string | undefined>;
|
|
@@ -145,6 +145,7 @@ export declare abstract class SPMatEntityCrudFormBase<TFormGroup extends Abstrac
|
|
|
145
145
|
transloco: TranslocoService;
|
|
146
146
|
cdr: ChangeDetectorRef;
|
|
147
147
|
http: HttpClient;
|
|
148
|
+
mode: import("@angular/core").Signal<"standalone" | "bridge">;
|
|
148
149
|
canCancelEdit: () => boolean;
|
|
149
150
|
_canCancelEdit(): boolean;
|
|
150
151
|
ngOnInit(): void;
|
|
@@ -217,8 +218,29 @@ export declare abstract class SPMatEntityCrudFormBase<TFormGroup extends Abstrac
|
|
|
217
218
|
* @returns
|
|
218
219
|
*/
|
|
219
220
|
protected update(id: any, values: any): Observable<TEntity>;
|
|
221
|
+
/**
|
|
222
|
+
* Wrapper around entityName input to get the entity name. If `bridge` input
|
|
223
|
+
* is defined, then its `getEntityName()` method is used. This allows
|
|
224
|
+
* derived classes to override this method to provide custom logic to
|
|
225
|
+
* determine the entity name.
|
|
226
|
+
* @returns
|
|
227
|
+
*/
|
|
228
|
+
protected getEntityName(): string | undefined;
|
|
229
|
+
/**
|
|
230
|
+
* Returns the baseUrl. Derived classes can override this to provide custom
|
|
231
|
+
* logic to determine the baseUrl.
|
|
232
|
+
* @returns
|
|
233
|
+
*/
|
|
234
|
+
protected getBaseUrl(): string | undefined;
|
|
235
|
+
/**
|
|
236
|
+
* Returns the entity URL for the given entity id. If `bridge` input is
|
|
237
|
+
* defined, then its `getEntityUrl()` method is used. Otherwise, the URL is
|
|
238
|
+
* derived from `baseUrl` input.
|
|
239
|
+
* @param entityId
|
|
240
|
+
* @returns
|
|
241
|
+
*/
|
|
220
242
|
protected getEntityUrl(entityId: any): string;
|
|
221
243
|
protected getRequestContext(): HttpContext;
|
|
222
244
|
static ɵfac: i0.ɵɵFactoryDeclaration<SPMatEntityCrudFormBase<any, any, any>, never>;
|
|
223
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<SPMatEntityCrudFormBase<any, any, any>, "_#_sp-mat-entity-crud-form-base_#_", never, { "entity": { "alias": "entity"; "required":
|
|
245
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<SPMatEntityCrudFormBase<any, any, any>, "_#_sp-mat-entity-crud-form-base_#_", never, { "entity": { "alias": "entity"; "required": false; "isSignal": true; }; "bridge": { "alias": "bridge"; "required": false; "isSignal": true; }; "params": { "alias": "params"; "required": false; "isSignal": true; }; "entityName": { "alias": "entityName"; "required": false; "isSignal": true; }; "baseUrl": { "alias": "baseUrl"; "required": false; "isSignal": true; }; "httpReqContext": { "alias": "httpReqContext"; "required": false; "isSignal": true; }; "idKey": { "alias": "idKey"; "required": false; "isSignal": true; }; }, {}, never, never, false, never>;
|
|
224
246
|
}
|
package/package.json
CHANGED
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
"path": "src/assets/i18n/"
|
|
7
7
|
}
|
|
8
8
|
],
|
|
9
|
-
"version": "0.33.
|
|
9
|
+
"version": "0.33.30",
|
|
10
10
|
"peerDependencies": {
|
|
11
11
|
"@angular/common": "^19.1.0",
|
|
12
12
|
"@angular/core": "^19.1.0",
|
|
@@ -48,26 +48,26 @@
|
|
|
48
48
|
"types": "./entities/index.d.ts",
|
|
49
49
|
"default": "./fesm2022/smallpearl-ngx-helper-entities.mjs"
|
|
50
50
|
},
|
|
51
|
-
"./locale": {
|
|
52
|
-
"types": "./locale/index.d.ts",
|
|
53
|
-
"default": "./fesm2022/smallpearl-ngx-helper-locale.mjs"
|
|
54
|
-
},
|
|
55
51
|
"./hover-dropdown": {
|
|
56
52
|
"types": "./hover-dropdown/index.d.ts",
|
|
57
53
|
"default": "./fesm2022/smallpearl-ngx-helper-hover-dropdown.mjs"
|
|
58
54
|
},
|
|
59
|
-
"./forms": {
|
|
60
|
-
"types": "./forms/index.d.ts",
|
|
61
|
-
"default": "./fesm2022/smallpearl-ngx-helper-forms.mjs"
|
|
62
|
-
},
|
|
63
55
|
"./mat-busy-wheel": {
|
|
64
56
|
"types": "./mat-busy-wheel/index.d.ts",
|
|
65
57
|
"default": "./fesm2022/smallpearl-ngx-helper-mat-busy-wheel.mjs"
|
|
66
58
|
},
|
|
59
|
+
"./locale": {
|
|
60
|
+
"types": "./locale/index.d.ts",
|
|
61
|
+
"default": "./fesm2022/smallpearl-ngx-helper-locale.mjs"
|
|
62
|
+
},
|
|
67
63
|
"./entity-field": {
|
|
68
64
|
"types": "./entity-field/index.d.ts",
|
|
69
65
|
"default": "./fesm2022/smallpearl-ngx-helper-entity-field.mjs"
|
|
70
66
|
},
|
|
67
|
+
"./forms": {
|
|
68
|
+
"types": "./forms/index.d.ts",
|
|
69
|
+
"default": "./fesm2022/smallpearl-ngx-helper-forms.mjs"
|
|
70
|
+
},
|
|
71
71
|
"./mat-context-menu": {
|
|
72
72
|
"types": "./mat-context-menu/index.d.ts",
|
|
73
73
|
"default": "./fesm2022/smallpearl-ngx-helper-mat-context-menu.mjs"
|