mirror of
https://github.com/devcode-it/openstamanager.git
synced 2025-04-03 04:41:06 +02:00
168 lines
4.5 KiB
TypeScript
168 lines
4.5 KiB
TypeScript
import {mdiFloppy} from '@mdi/js';
|
|
import RecordDialog, {RecordDialogAttributes} from '@osm/Components/Dialogs/RecordDialog';
|
|
import MdIcon from '@osm/Components/MdIcon';
|
|
import Model from '@osm/Models/Record';
|
|
import {
|
|
VnodeCollection,
|
|
VnodeCollectionItem
|
|
} from '@osm/typings/jsx';
|
|
import {
|
|
isFormValid,
|
|
isVnode,
|
|
showSnackbar
|
|
} from '@osm/utils/misc';
|
|
import collect, {Collection} from 'collect.js';
|
|
import {
|
|
Children,
|
|
Vnode,
|
|
VnodeDOM
|
|
} from 'mithril';
|
|
import Stream from 'mithril/stream';
|
|
import {Form} from 'mithril-utilities';
|
|
import {Class} from 'type-fest';
|
|
import {JsonapiErrorDoc} from 'spraypaint';
|
|
import {ResponseError} from 'spraypaint/lib-esm/request';
|
|
|
|
export default abstract class AddEditRecordDialog<M extends Model> extends RecordDialog<M> {
|
|
// eslint-disable-next-line unicorn/no-null
|
|
protected formElement: HTMLFormElement | null = null;
|
|
protected abstract formState: Map<string, Stream<any>> | Record<string, Stream<any>>;
|
|
protected abstract modelType: Class<M>;
|
|
// Recommended: <= 3
|
|
protected numberOfColumns: number = 3;
|
|
protected record!: M;
|
|
protected formId?: string;
|
|
|
|
oninit(vnode: Vnode<RecordDialogAttributes<M>, this>) {
|
|
super.oninit(vnode);
|
|
|
|
if (!this.record) {
|
|
this.record = new this.modelType();
|
|
}
|
|
|
|
this.fillForm();
|
|
}
|
|
|
|
oncreate(vnode: VnodeDOM<RecordDialogAttributes<M>, this>) {
|
|
super.oncreate(vnode);
|
|
|
|
this.formElement = this.element.querySelector('form');
|
|
this.element.querySelector(`#saveBtn${this.formId!}`)?.setAttribute('form', this.formId!);
|
|
}
|
|
|
|
fillForm() {
|
|
for (const [key, value] of this.formStateAsMap) {
|
|
// @ts-ignore
|
|
value(this.record[key] ?? value());
|
|
}
|
|
}
|
|
|
|
contents(): Children {
|
|
return (
|
|
<>
|
|
{this.form()}
|
|
</>
|
|
);
|
|
}
|
|
|
|
headline() {
|
|
return <span slot="headline">{this.record.isNew() ? __('Nuovo record') : __('Modifica record')}</span>;
|
|
}
|
|
|
|
form(): Children {
|
|
this.formId ??= `form-${Date.now()}`;
|
|
return (
|
|
<Form id={this.formId} state={this.formState} onsubmit={this.onFormSubmit.bind(this)} additionalElementsSelector="md-filled-select,md-outlined-select">
|
|
{this.formContents()}
|
|
</Form>
|
|
);
|
|
}
|
|
|
|
formContents(): Children {
|
|
return (
|
|
<md-layout-grid>
|
|
{...this.formFields().values().all()}
|
|
</md-layout-grid>
|
|
);
|
|
}
|
|
|
|
protected formFields(): Collection<Children> {
|
|
return this.fields().map((field, key: string) => {
|
|
// TODO: Remove this cast when new collection library is done
|
|
if (isVnode<{name?: string, 'grid-span'?: number}>(field)) {
|
|
field.attrs.name ??= key;
|
|
field.attrs['grid-span'] ??= Math.floor(12 / this.numberOfColumns);
|
|
field.key = key;
|
|
}
|
|
return field;
|
|
});
|
|
}
|
|
|
|
abstract fields(): Collection<Children>;
|
|
|
|
onCancelButtonClicked(): void {
|
|
void this.close('cancel');
|
|
}
|
|
|
|
async onFormSubmit() {
|
|
if (isFormValid(this.formElement!) && await this.save()) {
|
|
void this.close();
|
|
}
|
|
}
|
|
|
|
actions(): VnodeCollection {
|
|
return collect<VnodeCollectionItem>({
|
|
cancelButton: (
|
|
<md-text-button type="button" onclick={this.onCancelButtonClicked.bind(this)}>
|
|
{__('Annulla')}
|
|
</md-text-button>
|
|
),
|
|
saveButton: (
|
|
<md-text-button id={`saveBtn${this.formId}`} type="submit">
|
|
{__('Salva')}
|
|
<MdIcon icon={mdiFloppy} slot="icon"/>
|
|
</md-text-button>
|
|
)
|
|
});
|
|
}
|
|
|
|
async save(): Promise<boolean> {
|
|
this.record.assignAttributes(this.modelAttributesFromFormState);
|
|
try {
|
|
const result = await this.record.save();
|
|
this.afterSave(result);
|
|
return result;
|
|
} catch (error) {
|
|
this.onSaveError(error as ResponseError);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
afterSave(result: boolean): void {
|
|
if (result) {
|
|
void showSnackbar(__('Record salvato con successo'));
|
|
}
|
|
}
|
|
|
|
async onSaveError(error: ResponseError) {
|
|
const {message} = (await error.response!.json()) as JsonapiErrorDoc;
|
|
void showSnackbar(message, false);
|
|
}
|
|
|
|
protected static createFormState<EK extends string, EV extends Stream<any>>(entries: Record<EK, EV>): Map<EK, EV> {
|
|
return new Map<EK, EV>(Object.entries(entries) as [EK, EV][]);
|
|
}
|
|
|
|
get modelAttributesFromFormState(): Record<string, unknown> {
|
|
const state: Record<string, unknown> = {};
|
|
for (const [key, value] of this.formStateAsMap) {
|
|
state[key] = value();
|
|
}
|
|
return state;
|
|
}
|
|
|
|
protected get formStateAsMap(): Map<string, Stream<any>> {
|
|
return this.formState instanceof Map ? this.formState : new Map(Object.entries(this.formState));
|
|
}
|
|
}
|