* Split permissions for reading documents from updating collection * fix: Admins should have collection read permission, tests * tsc * Add admin option to permission selector * Combine publish and create permissions, update -> createDocuments where appropriate * Plural -> singular * wip * Quick version of collection structure loading, will revisit * Remove documentIds method * stash * fixing tests to account for admin creation * Add self-hosted migration * fix: Allow groups to have admin permission * Prefetch collection documents * fix: Document explorer (move/publish) not working with async documents * fix: Cannot re-parent document to collection by drag and drop * fix: Cannot drag to import into collection item without admin permission * Remove unused isEditor getter
139 lines
3.0 KiB
TypeScript
139 lines
3.0 KiB
TypeScript
import { pick } from "lodash";
|
|
import { set, observable } from "mobx";
|
|
import Logger from "~/utils/Logger";
|
|
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) => {
|
|
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) {
|
|
Logger.warn("Checking dirty on model with no @Field decorators");
|
|
}
|
|
|
|
return (
|
|
JSON.stringify(this.persistedAttributes) !== JSON.stringify(attributes)
|
|
);
|
|
}
|
|
|
|
protected persistedAttributes: Partial<BaseModel> = {};
|
|
}
|