* fix: Logic error in toast fix: Remove useless component * fix: Logout not clearing all stores * Add icons to notification settings * Add eslint rule to enforce spaced comment * Add eslint rule for arrow-body-style * Add eslint rule to enforce self-closing components * Add menu to api key settings Fix: Deleting webhook subscription does not remove from UI Split webhook subscriptions into active and inactive Styling updates
139 lines
3.0 KiB
TypeScript
139 lines
3.0 KiB
TypeScript
import { pick } from "lodash";
|
|
import { set, observable } from "mobx";
|
|
import { getFieldsForModel } from "./decorators/Field";
|
|
|
|
export default abstract class BaseModel {
|
|
@observable
|
|
id: string;
|
|
|
|
@observable
|
|
isSaving: boolean;
|
|
|
|
@observable
|
|
isNew: boolean;
|
|
|
|
createdAt: string;
|
|
|
|
updatedAt: string;
|
|
|
|
store: any;
|
|
|
|
constructor(fields: Record<string, any>, store: any) {
|
|
this.updateFromJson(fields);
|
|
this.isNew = !this.id;
|
|
this.store = store;
|
|
}
|
|
|
|
save = async (
|
|
params?: Record<string, any>,
|
|
options?: Record<string, string | boolean | number | undefined>
|
|
) => {
|
|
this.isSaving = true;
|
|
|
|
try {
|
|
// ensure that the id is passed if the document has one
|
|
if (!params) {
|
|
params = this.toAPI();
|
|
}
|
|
|
|
const model = await this.store.save(
|
|
{
|
|
...params,
|
|
id: this.id,
|
|
},
|
|
{
|
|
...options,
|
|
isNew: this.isNew,
|
|
}
|
|
);
|
|
|
|
// if saving is successful set the new values on the model itself
|
|
set(this, { ...params, ...model, isNew: false });
|
|
|
|
this.persistedAttributes = this.toAPI();
|
|
|
|
return model;
|
|
} finally {
|
|
this.isSaving = false;
|
|
}
|
|
};
|
|
|
|
updateFromJson = (data: any) => {
|
|
// const isNew = !data.id && !this.id && this.isNew;
|
|
set(this, { ...data, isNew: false });
|
|
this.persistedAttributes = this.toAPI();
|
|
};
|
|
|
|
fetch = (options?: any) => this.store.fetch(this.id, options);
|
|
|
|
refresh = () =>
|
|
this.fetch({
|
|
force: true,
|
|
});
|
|
|
|
delete = async () => {
|
|
this.isSaving = true;
|
|
|
|
try {
|
|
return await this.store.delete(this);
|
|
} finally {
|
|
this.isSaving = false;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Returns a plain object representation of fields on the model for
|
|
* persistence to the server API
|
|
*
|
|
* @returns {Record<string, any>}
|
|
*/
|
|
toAPI = (): Record<string, any> => {
|
|
const fields = getFieldsForModel(this);
|
|
return pick(this, fields) || [];
|
|
};
|
|
|
|
/**
|
|
* Returns a plain object representation of all the properties on the model
|
|
* overrides the inbuilt toJSON method to avoid attempting to serialize store
|
|
*
|
|
* @returns {Record<string, any>}
|
|
*/
|
|
toJSON() {
|
|
const output: Partial<typeof this> = {};
|
|
|
|
for (const property in this) {
|
|
if (
|
|
// eslint-disable-next-line no-prototype-builtins
|
|
this.hasOwnProperty(property) &&
|
|
!["persistedAttributes", "store", "isSaving", "isNew"].includes(
|
|
property
|
|
)
|
|
) {
|
|
output[property] = this[property];
|
|
}
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
/**
|
|
* Returns a boolean indicating if the model has changed since it was last
|
|
* persisted to the server
|
|
*
|
|
* @returns boolean true if unsaved
|
|
*/
|
|
isDirty(): boolean {
|
|
const attributes = this.toAPI();
|
|
|
|
if (Object.keys(attributes).length === 0) {
|
|
console.warn("Checking dirty on model with no @Field decorators");
|
|
}
|
|
|
|
return (
|
|
JSON.stringify(this.persistedAttributes) !== JSON.stringify(attributes)
|
|
);
|
|
}
|
|
|
|
protected persistedAttributes: Partial<BaseModel> = {};
|
|
}
|