2
0
mirror of https://github.com/frappe/books.git synced 2025-01-27 09:08:24 +00:00
books/client/view/form.js

335 lines
9.9 KiB
JavaScript
Raw Normal View History

2018-01-16 11:39:17 +05:30
const frappe = require('frappejs');
2018-01-12 17:55:07 +05:30
const controls = require('./controls');
const FormLayout = require('./formLayout');
2018-01-31 15:43:33 +05:30
const Observable = require('frappejs/utils/observable');
const keyboard = require('frappejs/client/ui/keyboard');
const utils = require('frappejs/client/ui/utils');
2018-01-12 17:55:07 +05:30
2018-01-31 15:43:33 +05:30
module.exports = class BaseForm extends Observable {
2018-03-27 19:25:26 +05:30
constructor({doctype, parent, submit_label='Submit', container, meta, inline=false}) {
2018-01-31 15:43:33 +05:30
super();
2018-02-08 15:08:47 +05:30
Object.assign(this, arguments[0]);
2018-03-26 14:23:46 +05:30
this.links = [];
2018-01-12 17:55:07 +05:30
2018-04-16 18:34:16 +05:30
if (!this.meta) {
this.meta = frappe.getMeta(this.doctype);
}
2018-03-27 19:25:26 +05:30
2018-01-24 17:22:05 +05:30
if (this.setup) {
this.setup();
}
2018-03-27 19:25:26 +05:30
2018-01-12 17:55:07 +05:30
this.make();
2018-03-08 19:01:22 +05:30
this.bindFormEvents();
2018-03-27 19:25:26 +05:30
if (this.doc) {
// bootstrapped with a doc
this.bindEvents(this.doc);
}
2018-01-12 17:55:07 +05:30
}
make() {
if (this.body || !this.parent) {
return;
}
2018-03-27 19:25:26 +05:30
if (this.inline) {
this.body = this.parent
} else {
this.body = frappe.ui.add('div', 'form-body', this.parent);
}
if (this.actions) {
this.makeToolbar();
}
2018-01-12 17:55:07 +05:30
2018-02-09 18:25:55 +05:30
this.form = frappe.ui.add('form', 'form-container', this.body);
2018-03-27 19:25:26 +05:30
if (this.inline) {
this.form.classList.add('form-inline');
}
2018-02-09 18:25:55 +05:30
this.form.onValidate = true;
this.formLayout = new FormLayout({
fields: this.meta.fields,
layout: this.meta.layout
});
this.form.appendChild(this.formLayout.form);
this.bindKeyboard();
2018-02-09 18:25:55 +05:30
}
2018-03-08 19:01:22 +05:30
bindFormEvents() {
if (this.meta.formEvents) {
for (let key in this.meta.formEvents) {
this.on(key, this.meta.formEvents[key]);
}
}
}
2018-02-08 15:08:47 +05:30
makeToolbar() {
if (this.actions.includes('save')) {
this.makeSaveButton();
if (this.meta.isSubmittable) {
this.makeSubmitButton();
2018-03-05 22:15:21 +05:30
this.makeRevertButton();
}
}
if (this.meta.print && this.actions.includes('print')) {
let menu = this.container.getDropdown(frappe._('Menu'));
menu.addItem(frappe._("Print"), async (e) => {
await frappe.router.setRoute('print', this.doctype, this.doc.name);
});
}
2018-02-14 18:20:56 +05:30
if (!this.meta.isSingle && this.actions.includes('delete')) {
let menu = this.container.getDropdown(frappe._('Menu'));
menu.addItem(frappe._("Delete"), async (e) => {
2018-02-20 15:23:38 +05:30
await this.delete();
});
}
2018-02-14 18:20:56 +05:30
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);
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 18:20:56 +05:30
});
}
}
2018-01-23 13:30:29 +05:30
makeSaveButton() {
this.saveButton = this.container.addButton(frappe._("Save"), 'primary', async (event) => {
await this.save();
});
this.on('change', () => {
const show = this.doc._dirty && !this.doc.submitted;
this.saveButton.classList.toggle('hide', !show);
});
}
makeSubmitButton() {
this.submitButton = this.container.addButton(frappe._("Submit"), 'primary', async (event) => {
await this.submit();
});
this.on('change', () => {
const show = this.meta.isSubmittable && !this.doc._dirty && !this.doc.submitted;
this.submitButton.classList.toggle('hide', !show);
});
2018-03-05 22:15:21 +05:30
}
2018-03-05 22:15:21 +05:30
makeRevertButton() {
this.revertButton = this.container.addButton(frappe._("Revert"), 'secondary', async (event) => {
await this.revert();
});
this.on('change', () => {
const show = this.meta.isSubmittable && !this.doc._dirty && this.doc.submitted;
this.revertButton.classList.toggle('hide', !show);
});
}
bindKeyboard() {
keyboard.bindKey(this.form, 'ctrl+s', (e) => {
if (document.activeElement) {
document.activeElement.blur();
}
e.preventDefault();
if (this.doc._notInserted || this.doc._dirty) {
this.save();
} else {
if (this.meta.isSubmittable && !this.doc.submitted) this.submit();
}
2018-01-12 17:55:07 +05:30
});
}
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 15:23:28 +05:30
this.setTitle();
frappe._curFrm = this;
2018-02-15 15:23:28 +05:30
}
setTitle() {
2018-03-27 19:25:26 +05:30
if (!this.container) return;
const doctypeLabel = this.doc.meta.label || this.doc.meta.name;
2018-03-05 22:15:21 +05:30
if (this.doc.meta.isSingle || this.doc.meta.naming === 'random') {
this.container.setTitle(doctypeLabel);
2018-02-15 15:23:28 +05:30
} else if (this.doc._notInserted) {
this.container.setTitle(frappe._('New {0}', doctypeLabel));
2018-02-15 15:23:28 +05:30
} else {
2018-02-22 13:06:28 +05:30
this.container.setTitle(this.doc.name);
2018-02-15 15:23:28 +05:30
}
2018-03-05 22:15:21 +05:30
if (this.doc.submitted) {
2018-03-07 16:07:58 +05:30
// this.container.addTitleBadge('✓', frappe._('Submitted'));
2018-03-05 22:15:21 +05:30
}
}
2018-03-26 14:23:46 +05:30
setLinks(label, options) {
// set links to helpful reports as identified by this.meta.links
if (this.meta.links) {
let links = this.getLinks();
if (!links.equals(this.links)) {
this.refreshLinks(links);
this.links = links;
}
}
}
getLinks() {
let links = [];
for (let link of this.meta.links) {
if (link.condition(this)) {
links.push(link);
}
}
return links;
}
refreshLinks(links) {
2018-04-18 12:18:24 +05:30
if (!(this.container && this.container.clearLinks)) return;
2018-03-27 19:25:26 +05:30
2018-03-26 14:23:46 +05:30
this.container.clearLinks();
for(let link of links) {
// make the link
2018-03-30 22:28:00 +05:30
utils.addButton(link.label, this.container.linksElement, () => {
2018-03-26 14:23:46 +05:30
let options = link.action(this);
if (options) {
if (options.params) {
// set route parameters
frappe.params = options.params;
}
if (options.route) {
// go to the given route
frappe.router.setRoute(...options.route);
}
}
});
}
}
async bindEvents(doc) {
if (this.doc && this.docListener) {
// stop listening to the old doc
this.doc.off(this.docListener);
2018-01-12 17:55:07 +05:30
}
this.doc = doc;
for (let control of this.formLayout.controlList) {
2018-01-12 17:55:07 +05:30
control.bind(this.doc);
}
2018-03-08 19:01:22 +05:30
this.refresh();
this.setupDocListener();
2018-02-09 18:25:55 +05:30
this.trigger('use', {doc:doc});
}
setupDocListener() {
// refresh value in control
this.docListener = (params) => {
2018-02-08 17:15:32 +05:30
if (params.fieldname) {
// only single value changed
let control = this.formLayout.controls[params.fieldname];
2018-02-08 17:15:32 +05:30
if (control && control.getInputValue() !== control.format(params.fieldname)) {
2018-02-19 12:01:07 +05:30
control.refresh();
2018-02-08 17:15:32 +05:30
}
} else {
// multiple values changed
this.refresh();
}
this.trigger('change');
2018-02-09 18:25:55 +05:30
this.form.classList.remove('was-validated');
};
this.doc.on('change', this.docListener);
this.trigger('change');
2018-01-12 17:55:07 +05:30
}
2018-02-12 15:12:26 +05:30
checkValidity() {
let validity = this.form.checkValidity();
if (validity) {
for (let control of this.formLayout.controlList) {
2018-02-12 15:12:26 +05:30
// check validity in table
if (control.fieldtype==='Table') {
validity = control.checkValidity();
if (!validity) {
break;
}
}
}
}
return validity;
}
refresh() {
this.formLayout.refresh();
2018-03-08 19:01:22 +05:30
this.trigger('refresh', this);
2018-03-26 14:23:46 +05:30
this.setLinks();
}
2018-01-12 17:55:07 +05:30
async submit() {
this.doc.submitted = 1;
await this.save();
}
2018-03-05 22:15:21 +05:30
async revert() {
this.doc.submitted = 0;
await this.save();
}
async save() {
2018-02-12 15:12:26 +05:30
if (!this.checkValidity()) {
2018-02-09 18:25:55 +05:30
this.form.classList.add('was-validated');
return;
}
2018-01-12 17:55:07 +05:30
try {
2018-03-05 22:15:21 +05:30
let oldName = this.doc.name;
2018-02-09 18:25:55 +05:30
if (this.doc._notInserted) {
2018-01-12 17:55:07 +05:30
await this.doc.insert();
} else {
await this.doc.update();
}
frappe.ui.showAlert({message: frappe._('Saved'), color: 'green'});
2018-03-05 22:15:21 +05:30
if (oldName !== this.doc.name) {
frappe.router.setRoute('edit', this.doctype, this.doc.name);
return;
}
this.refresh();
this.trigger('change');
2018-01-12 17:55:07 +05:30
} catch (e) {
frappe.ui.showAlert({message: frappe._('Failed'), color: 'red'});
2018-02-09 18:25:55 +05:30
return;
2018-01-12 17:55:07 +05:30
}
await this.trigger('save');
2018-01-12 17:55:07 +05:30
}
2018-02-20 15:23:38 +05:30
async delete() {
try {
await this.doc.delete();
frappe.ui.showAlert({message: frappe._('Deleted'), color: 'green'});
2018-02-20 15:23:38 +05:30
this.trigger('delete');
} catch (e) {
frappe.ui.showAlert({message: e, color: 'red'});
2018-01-31 15:43:33 +05:30
}
}
}