2
0
mirror of https://github.com/frappe/books.git synced 2024-11-10 15:50:56 +00:00
books/client/view/form.js

211 lines
6.2 KiB
JavaScript
Raw Normal View History

2018-01-16 06:09:17 +00:00
const frappe = require('frappejs');
2018-01-12 12:25:07 +00:00
const controls = require('./controls');
2018-01-31 10:13:33 +00:00
const Observable = require('frappejs/utils/observable');
const keyboard = require('frappejs/client/ui/keyboard');
2018-01-12 12:25:07 +00:00
2018-01-31 10:13:33 +00:00
module.exports = class BaseForm extends Observable {
constructor({doctype, parent, submit_label='Submit', container}) {
2018-01-31 10:13:33 +00:00
super();
2018-02-08 09:38:47 +00:00
Object.assign(this, arguments[0]);
2018-01-12 12:25:07 +00:00
this.controls = {};
2018-02-08 09:38:47 +00:00
this.controlList = [];
2018-01-12 12:25:07 +00:00
this.meta = frappe.getMeta(this.doctype);
2018-01-24 11:52:05 +00:00
if (this.setup) {
this.setup();
}
2018-01-12 12:25:07 +00:00
this.make();
}
make() {
if (this.body || !this.parent) {
return;
}
this.body = frappe.ui.add('div', 'form-body', this.parent);
2018-02-08 09:38:47 +00:00
this.makeToolbar();
2018-01-12 12:25:07 +00:00
2018-02-09 12:55:55 +00:00
this.form = frappe.ui.add('form', 'form-container', this.body);
this.form.onValidate = true;
this.makeControls();
this.bindKeyboard();
2018-02-09 12:55:55 +00:00
}
makeControls() {
2018-02-06 17:14:07 +00:00
for(let field of this.meta.fields) {
2018-02-09 12:55:55 +00:00
if (!field.hidden && controls.getControlClass(field.fieldtype)) {
2018-02-08 09:38:47 +00:00
let control = controls.makeControl({field: field, form: this});
this.controlList.push(control);
2018-02-06 17:14:07 +00:00
this.controls[field.fieldname] = control;
2018-01-12 12:25:07 +00:00
}
}
}
2018-02-08 09:38:47 +00:00
makeToolbar() {
if (this.actions.includes('submit')) {
2018-02-14 12:50:56 +00:00
this.container.addButton(frappe._("Save"), 'primary', async (event) => {
await this.submit();
})
}
2018-02-14 12:50:56 +00:00
if (!this.meta.isSingle && this.actions.includes('delete')) {
let menu = this.container.getDropdown(frappe._('Menu'));
menu.addItem(frappe._("Delete"), async (e) => {
2018-02-20 09:53:38 +00:00
await this.delete();
});
}
2018-02-14 12:50:56 +00:00
if (!this.meta.isSingle && this.actions.includes('duplicate')) {
let menu = this.container.getDropdown(frappe._('Menu'));
menu.addItem(frappe._('Duplicate'), async () => {
let newDoc = await frappe.getDuplicate(this.doc);
console.log(newDoc);
await frappe.router.setRoute('edit', newDoc.doctype, newDoc.name);
newDoc.set('name', '');
});
}
if (this.meta.settings && this.actions.includes('settings')) {
let menu = this.container.getDropdown(frappe._('Menu'));
menu.addItem(frappe._('Settings...'), () => {
frappe.desk.showFormModal(this.meta.settings, this.meta.settings);
2018-02-14 12:50:56 +00:00
});
}
}
2018-01-23 08:00:29 +00:00
bindKeyboard() {
keyboard.bindKey(this.form, 'ctrl+s', (e) => {
if (document.activeElement) {
document.activeElement.blur();
}
e.preventDefault();
this.submit();
2018-01-12 12:25:07 +00:00
});
}
async setDoc(doctype, name) {
this.doc = await frappe.getDoc(doctype, name);
this.bindEvents(this.doc);
if (this.doc._notInserted && !this.doc._nameCleared) {
this.doc._nameCleared = true;
// flag so that name is cleared only once
await this.doc.set('name', '');
}
2018-02-15 09:53:28 +00:00
this.setTitle();
}
setTitle() {
const doctypeLabel = this.doc.meta.label || this.doc.meta.name;
2018-02-15 09:53:28 +00:00
if (this.doc.meta.isSingle || !this.doc.meta.showTitle) {
this.container.setTitle(doctypeLabel);
2018-02-15 09:53:28 +00:00
} else if (this.doc._notInserted) {
this.container.setTitle(frappe._('New {0}', doctypeLabel));
2018-02-15 09:53:28 +00:00
} else {
this.container.setTitle(`${doctypeLabel} ${this.doc.name}`);
2018-02-15 09:53:28 +00:00
}
}
async bindEvents(doc) {
2018-01-12 12:25:07 +00:00
if (this.doc) {
2018-02-13 11:54:57 +00:00
// clear listeners of outgoing doc
this.doc.clearListeners();
2018-01-12 12:25:07 +00:00
}
2018-02-08 09:38:47 +00:00
this.clearAlert();
2018-01-12 12:25:07 +00:00
this.doc = doc;
2018-02-08 09:38:47 +00:00
for (let control of this.controlList) {
2018-01-12 12:25:07 +00:00
control.bind(this.doc);
}
2018-02-13 11:54:57 +00:00
this.setupChangeListener();
2018-02-09 12:55:55 +00:00
this.trigger('use', {doc:doc});
}
2018-02-13 11:54:57 +00:00
setupChangeListener() {
// refresh value in control
2018-02-13 11:54:57 +00:00
this.doc.on('change', (params) => {
2018-02-08 11:45:32 +00:00
if (params.fieldname) {
// only single value changed
let control = this.controls[params.fieldname];
if (control && control.getInputValue() !== control.format(params.fieldname)) {
2018-02-19 06:31:07 +00:00
control.refresh();
2018-02-08 11:45:32 +00:00
}
} else {
// multiple values changed
this.refresh();
}
2018-02-09 12:55:55 +00:00
this.form.classList.remove('was-validated');
});
2018-01-12 12:25:07 +00:00
}
2018-02-12 09:42:26 +00:00
checkValidity() {
let validity = this.form.checkValidity();
if (validity) {
for (let control of this.controlList) {
// check validity in table
if (control.fieldtype==='Table') {
validity = control.checkValidity();
if (!validity) {
break;
}
}
}
}
return validity;
}
2018-01-12 12:25:07 +00:00
async submit() {
2018-02-12 09:42:26 +00:00
if (!this.checkValidity()) {
2018-02-09 12:55:55 +00:00
this.form.classList.add('was-validated');
return;
}
2018-01-12 12:25:07 +00:00
try {
2018-02-09 12:55:55 +00:00
if (this.doc._notInserted) {
2018-01-12 12:25:07 +00:00
await this.doc.insert();
} else {
await this.doc.update();
}
await this.refresh();
2018-02-08 09:38:47 +00:00
this.showAlert('Saved', 'success');
2018-01-12 12:25:07 +00:00
} catch (e) {
2018-02-08 09:38:47 +00:00
this.showAlert('Failed', 'danger');
2018-02-09 12:55:55 +00:00
return;
2018-01-12 12:25:07 +00:00
}
2018-01-31 10:13:33 +00:00
await this.trigger('submit');
2018-01-12 12:25:07 +00:00
}
2018-02-20 09:53:38 +00:00
async delete() {
try {
await this.doc.delete();
this.showAlert('Deleted', 'success');
this.trigger('delete');
} catch (e) {
this.showAlert(e, 'danger');
}
}
2018-01-12 12:25:07 +00:00
refresh() {
2018-02-08 09:38:47 +00:00
for(let control of this.controlList) {
2018-01-12 12:25:07 +00:00
control.refresh();
}
}
2018-02-08 09:38:47 +00:00
showAlert(message, type, clear_after = 5) {
this.clearAlert();
2018-01-31 10:13:33 +00:00
this.alert = frappe.ui.add('div', `alert alert-${type}`, this.body);
this.alert.textContent = message;
setTimeout(() => {
2018-02-08 09:38:47 +00:00
this.clearAlert();
2018-01-31 10:13:33 +00:00
}, clear_after * 1000);
}
2018-02-08 09:38:47 +00:00
clearAlert() {
2018-01-31 10:13:33 +00:00
if (this.alert) {
frappe.ui.remove(this.alert);
this.alert = null;
}
}
}