2
0
mirror of https://github.com/frappe/books.git synced 2025-01-25 16:18:33 +00:00
- Simplify modal plugin
- Support multiple stacked modals in a modal container
- Add formModal plugin
This commit is contained in:
Faris Ansari 2018-07-10 19:05:43 +05:30
parent 4fe78a2878
commit 8f2c48c3df
11 changed files with 345 additions and 274 deletions

View File

@ -22,7 +22,7 @@
"eslint": "^4.19.1", "eslint": "^4.19.1",
"express": "^4.16.2", "express": "^4.16.2",
"flatpickr": "^4.3.2", "flatpickr": "^4.3.2",
"frappe-datatable": "^1.1.1", "frappe-datatable": "^1.1.2",
"jquery": "^3.3.1", "jquery": "^3.3.1",
"jwt-simple": "^0.5.1", "jwt-simple": "^0.5.1",
"luxon": "^1.0.0", "luxon": "^1.0.0",

View File

@ -2,10 +2,7 @@
<div class="frappe-form"> <div class="frappe-form">
<form-actions <form-actions
v-if="shouldRenderForm" v-if="shouldRenderForm"
:doctype="doctype" :doc="doc"
:name="name"
:title="formTitle"
:isDirty="isDirty"
@save="save" @save="save"
/> />
<div class="p-3"> <div class="p-3">
@ -38,7 +35,6 @@ export default {
docLoaded: false, docLoaded: false,
notFound: false, notFound: false,
invalid: false, invalid: false,
isDirty: false,
invalidFields: [] invalidFields: []
} }
}, },
@ -48,21 +44,12 @@ export default {
}, },
shouldRenderForm() { shouldRenderForm() {
return this.name && this.docLoaded; return this.name && this.docLoaded;
},
formTitle() {
if (this.doc._notInserted) {
return _('New {0}', _(this.doctype));
}
return this.doc[this.meta.titleField];
} }
}, },
async created() { async created() {
if (!this.name) return; if (!this.name) return;
try { try {
this.doc = await frappe.getDoc(this.doctype, this.name); this.doc = await frappe.getDoc(this.doctype, this.name);
this.doc.on('change', () => {
this.isDirty = this.doc._dirty;
});
if (this.doc._notInserted && this.meta.fields.map(df => df.fieldname).includes('name')) { if (this.doc._notInserted && this.meta.fields.map(df => df.fieldname).includes('name')) {
// For a user editable name field, // For a user editable name field,

View File

@ -1,11 +1,38 @@
<template> <template>
<div class="frappe-form-actions d-flex justify-content-between align-items-center p-3 border-bottom"> <div class="frappe-form-actions d-flex justify-content-between align-items-center p-3 border-bottom">
<h5 class="m-0">{{ title || name }}</h5> <h5 class="m-0">{{ title }}</h5>
<f-button primary :disabled="!isDirty" @click="$emit('save')">{{ _('Save') }}</f-button> <f-button primary :disabled="!isDirty" @click="$emit('save')">{{ _('Save') }}</f-button>
</div> </div>
</template> </template>
<script> <script>
import frappe from 'frappejs';
export default { export default {
props: ['doctype', 'name', 'title', 'isDirty'] props: ['doc'],
data() {
return {
isDirty: false
}
},
created() {
this.doc.on('change', () => {
this.isDirty = this.doc._dirty;
});
},
computed: {
meta() {
return frappe.getMeta(this.doc.doctype);
},
title() {
const _ = this._;
if (this.doc._notInserted) {
return _('New {0}', _(this.doc.doctype));
}
const titleField = this.meta.titleField || 'name';
return this.doc[titleField];
}
}
} }
</script> </script>

View File

@ -7,7 +7,7 @@
<div class="col" v-for="(column, j) in section.columns" :key="j"> <div class="col" v-for="(column, j) in section.columns" :key="j">
<frappe-control <frappe-control
v-for="fieldname in column.fields" v-for="fieldname in column.fields"
v-if="fieldIsNotHidden(fieldname)" v-if="shouldRenderField(fieldname)"
:key="fieldname" :key="fieldname"
:docfield="getDocField(fieldname)" :docfield="getDocField(fieldname)"
:value="$data[fieldname]" :value="$data[fieldname]"
@ -18,7 +18,7 @@
<div v-if="!layout"> <div v-if="!layout">
<frappe-control <frappe-control
v-for="docfield in fields" v-for="docfield in fields"
v-if="!docfield.hidden" v-if="shouldRenderField(docfield.fieldname)"
:key="docfield.fieldname" :key="docfield.fieldname"
:docfield="docfield" :docfield="docfield"
:value="$data[docfield.fieldname]" :value="$data[docfield.fieldname]"
@ -60,8 +60,18 @@ export default {
getDocField(fieldname) { getDocField(fieldname) {
return this.fields.find(df => df.fieldname === fieldname); return this.fields.find(df => df.fieldname === fieldname);
}, },
fieldIsNotHidden(fieldname) { shouldRenderField(fieldname) {
return !Boolean(this.getDocField(fieldname).hidden); const hidden = Boolean(this.getDocField(fieldname).hidden);
if (hidden) {
return false;
}
if (fieldname === 'name' && !this.doc._notInserted) {
return false;
}
return true;
}, },
updateDoc(fieldname, value) { updateDoc(fieldname, value) {
this.doc.set(fieldname, value); this.doc.set(fieldname, value);

View File

@ -1,70 +0,0 @@
<template>
<div>
<div :class="['modal fade', modalClasses]" :style="{display: show ? 'block' : ''}" id="frappe-modal"
tabindex="-1" role="dialog" aria-labelledby="frappe-modal-label" aria-hidden="true">
<div class="modal-dialog" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="frappe-modal-label">{{ title }}</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close" @click="closeModal">
<span aria-hidden="true">&times;</span>
</button>
</div>
<div class="modal-body modal-height">
<component :is="bodyComponent" v-bind="bodyProps"/>
</div>
<div class="modal-footer">
<component :is="footerComponent" v-bind="footerProps"/>
<f-button secondary @click="closeModal">Close</f-button>
</div>
</div>
</div>
</div>
<div class="modal-backdrop show" v-show="show" @click="closeModal"></div>
</div>
</template>
<script>
export default {
props: {
show: Boolean,
title: {
type: String,
default: "Modal Title"
},
bodyComponent: {
type: Object,
default: null
},
bodyProps: {
type: Object,
default: null
},
footerComponent: {
type: Object,
default: null
},
footerProps: {
type: Object,
default: null
},
},
computed: {
modalClasses() {
return {
show: this.show
};
}
},
methods: {
closeModal() {
this.$emit('close-modal');
}
}
};
</script>
<style scoped>
.modal-height {
max-height: 80vh;
overflow: auto;
}
</style>

View File

@ -0,0 +1,51 @@
<template>
<div :class="['modal fade show d-block']"
tabindex="-1" role="dialog" aria-labelledby="frappe-modal-label" aria-hidden="true">
<div class="modal-dialog" role="document">
<div class="modal-content shadow">
<div class="modal-header">
<h5 class="modal-title">{{ title }}</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close" @click="closeModal">
<span aria-hidden="true">&times;</span>
</button>
</div>
<div class="modal-body modal-height">
<component :is="component" v-bind="props" v-on="events"/>
</div>
<div class="modal-footer">
<f-button secondary @click="closeModal">{{ _('Close') }}</f-button>
</div>
</div>
</div>
</div>
</template>
<script>
export default {
props: {
title: {
type: String,
default: "Modal Title"
},
component: {
type: Object
},
props: {
type: Object
},
events: {
type: Object
}
},
methods: {
closeModal() {
this.$emit('close-modal');
}
}
};
</script>
<style scoped>
.modal-height {
max-height: 80vh;
overflow: auto;
}
</style>

View File

@ -0,0 +1,64 @@
<template>
<div class="modal-container">
<modal
:key="modal.id"
v-for="modal in modals"
v-bind="modal"
@close-modal="onModalClose(modal.id)"
></modal>
<div class="modal-backdrop show" v-show="modals.length"></div>
</div>
</template>
<script>
import Modal from './Modal';
import Plugin from './plugin';
export default {
name: 'ModalContainer',
components: {
Modal
},
data() {
return {
currentId: 0,
modals: []
}
},
created() {
Plugin.modalContainer = this;
Plugin.event.$on('hide', (id) => {
if (!id) {
console.warn(`id not provided in $modal.hide method, the last modal in the stack will be hidden`);
}
this.onModalClose(id);
});
},
methods: {
add(component, props, events) {
this.currentId++;
this.modals.push({
id: this.currentId,
component,
props,
events
});
return this.currentId;
},
removeModal(id) {
if (!id) {
id = this.currentId;
}
this.currentId--;
this.modals = this.modals.filter(modal => modal.id !== id);
},
onModalClose(id) {
if (id) {
const modal = this.modals.find(modal => modal.id === id);
modal.props.onClose && modal.props.onClose();
}
this.removeModal(id);
}
}
}
</script>

View File

@ -0,0 +1,25 @@
import ModalContainer from './ModalContainer';
const Plugin = {
install (Vue) {
this.event = new Vue();
Vue.prototype.$modal = {
show(...args) {
Plugin.modalContainer.add(...args);
},
hide(id) {
Plugin.event.$emit('hide', id);
}
}
// create modal container
const div = document.createElement('div');
document.body.appendChild(div);
new Vue({ render: h => h(ModalContainer) }).$mount(div);
}
}
export default Plugin;

View File

@ -19,7 +19,7 @@ export default {
const list = await frappe.db.getAll({ const list = await frappe.db.getAll({
doctype: this.getTarget(), doctype: this.getTarget(),
filters: { filters: {
keywords: ["like", query] keywords: ['like', query]
}, },
fields: ['name'], fields: ['name'],
limit: 50 limit: 50
@ -39,19 +39,23 @@ export default {
.concat({ .concat({
label: plusIcon + ' New ' + this.getTarget(), label: plusIcon + ' New ' + this.getTarget(),
value: '__newItem' value: '__newItem'
}) });
}, },
getWrapperElement(h) { getWrapperElement(h) {
return h('div', { return h(
'div',
{
class: ['form-group', ...this.wrapperClass], class: ['form-group', ...this.wrapperClass],
attrs: { attrs: {
'data-fieldname': this.docfield.fieldname 'data-fieldname': this.docfield.fieldname
} }
}, [ },
[
this.getLabelElement(h), this.getLabelElement(h),
this.getInputElement(h), this.getInputElement(h),
this.getFollowLink(h) this.getFollowLink(h)
]); ]
);
}, },
getFollowLink(h) { getFollowLink(h) {
const doctype = this.getTarget(); const doctype = this.getTarget();
@ -84,11 +88,22 @@ export default {
}, },
sort() { sort() {
return (a, b) => { return (a, b) => {
if (a.value === '__newitem' || b.value === '__newitem') { if (a.value === '__newItem') {
return 1;
}
if (b.value === '___newItem') {
return -1; return -1;
} }
return a.value > b.value; if (a.value === b.value) {
return 0;
} }
if (a.value < b.value) {
return -1;
}
if (a.value > b.value) {
return 1;
}
};
}, },
filter() { filter() {
return (suggestion, txt) => { return (suggestion, txt) => {
@ -96,46 +111,42 @@ export default {
return true; return true;
} }
return Awesomplete.FILTER_CONTAINS(suggestion, txt); return Awesomplete.FILTER_CONTAINS(suggestion, txt);
} };
}, },
bindEvents() { bindEvents() {
const input = this.$refs.input; const input = this.$refs.input;
input.addEventListener('awesomplete-select', async (e) => { input.addEventListener('awesomplete-select', async e => {
// new item action // new item action
if (e.text && e.text.value === '__newItem') { if (e.text && e.text.value === '__newItem') {
e.preventDefault(); e.preventDefault();
const newDoc = await frappe.getNewDoc(this.getTarget()); const newDoc = await frappe.getNewDoc(this.getTarget());
this.$modal.show({ this.$formModal.open(
title: _('New {0}', _(newDoc.doctype)), newDoc,
bodyComponent: Form, {
bodyProps: {
doctype: newDoc.doctype,
name: newDoc.name,
defaultValues: { defaultValues: {
name: input.value name: input.value !== '__newItem' ? input.value : null
}
}, },
}); onClose: () => {
newDoc.on('afterInsert', (data) => {
// if new doc was created
// then set the name of the doc in input
this.handleChange(newDoc.name);
this.$modal.hide();
});
this.$modal.observable().once('modal.hide', () => {
// if new doc was not created // if new doc was not created
// then reset the input value // then reset the input value
if (this.value === '__newItem') { if (this.value === '__newItem') {
this.handleChange(''); this.handleChange('');
} }
})
}
})
} }
} }
);
newDoc.on('afterInsert', data => {
// if new doc was created
// then set the name of the doc in input
this.handleChange(newDoc.name);
this.$formModal.close();
});
} }
});
}
}
};
</script> </script>

27
ui/plugins/formModal.js Normal file
View File

@ -0,0 +1,27 @@
import Form from '../components/Form/Form';
export default function installFormModal(Vue) {
Vue.mixin({
computed: {
$formModal() {
const open = (doc, options = {}) => {
const { defaultValues = null, onClose = null } = options;
this.$modal.show(Form, {
doctype: doc.doctype,
name: doc.name,
defaultValues,
onClose
});
}
const close = () => this.$modal.hide();
return {
open,
close
}
}
}
})
}

View File

@ -1,61 +0,0 @@
import Observable from 'frappejs/utils/observable';
const Bus = new Observable();
export default {
// enable use of this.$modal in every component
// this also keeps only one modal in the DOM at any time
// which is the recommended way by bootstrap
install (Vue) {
Vue.prototype.$modal = {
show(options) {
Bus.trigger('showModal', options);
},
hide() {
Bus.trigger('hideModal');
},
observable() {
return Bus;
}
}
Vue.mixin({
data() {
return {
registered: false,
modalVisible: false,
modalOptions: {},
modalListeners: {}
}
},
watch: {
modalVisible(value) {
if (value === true) {
Bus.trigger('modal.show');
} else {
Bus.trigger('modal.hide');
}
}
},
created: function () {
if (this.registered) return;
Bus.on('showModal', (options = {}) => {
this.modalVisible = true;
this.modalOptions = options;
});
Bus.on('hideModal', () => {
this.modalVisible = false;
this.modalOptions = {};
});
this.registered = true;
}
});
}
}