mirror of
https://github.com/frappe/books.git
synced 2025-01-15 03:44:57 +00:00
178 lines
4.0 KiB
Vue
178 lines
4.0 KiB
Vue
<script>
|
|
import frappe from 'frappejs';
|
|
import feather from 'feather-icons';
|
|
import Autocomplete from './Autocomplete';
|
|
import FeatherIcon from 'frappejs/ui/components/FeatherIcon';
|
|
import Form from '../Form/Form';
|
|
import { _ } from 'frappejs/utils';
|
|
|
|
export default {
|
|
extends: Autocomplete,
|
|
methods: {
|
|
async getList(query) {
|
|
let filters = this.docfield.getFilters
|
|
? this.docfield.getFilters(query, this.doc)
|
|
: null;
|
|
|
|
if (query) {
|
|
if (!filters) filters = {};
|
|
filters.keywords = ['like', query];
|
|
}
|
|
|
|
let target = this.getTarget();
|
|
let titleField = frappe.getMeta(target).titleField;
|
|
|
|
const list = await frappe.db.getAll({
|
|
doctype: target,
|
|
filters,
|
|
fields: ['name', titleField],
|
|
limit: 50
|
|
});
|
|
|
|
const plusIcon = feather.icons.plus.toSvg({
|
|
class: 'm-1',
|
|
width: 16,
|
|
height: 16
|
|
});
|
|
|
|
return list
|
|
.map(d => ({
|
|
label: d[titleField],
|
|
value: d.name
|
|
}))
|
|
.concat({
|
|
label: plusIcon + ' New ' + this.getTarget(),
|
|
filters,
|
|
value: '__newItem'
|
|
});
|
|
},
|
|
getChildrenElement(h) {
|
|
return [
|
|
this.onlyInput ? null : this.getLabelElement(h),
|
|
this.getInputGroupElement(h),
|
|
this.getDropdownElement(h)
|
|
];
|
|
},
|
|
getInputGroupElement(h) {
|
|
return h(
|
|
'div',
|
|
{
|
|
class: ['input-group']
|
|
},
|
|
[
|
|
this.getInputElement(h),
|
|
h(
|
|
'div',
|
|
{
|
|
class: ['input-group-append']
|
|
},
|
|
[this.getFollowLink(h)]
|
|
)
|
|
]
|
|
);
|
|
},
|
|
getFollowLink(h) {
|
|
const doctype = this.getTarget();
|
|
const name = this.value;
|
|
|
|
if (!name) {
|
|
return null;
|
|
}
|
|
|
|
const arrow = h(FeatherIcon, {
|
|
props: {
|
|
name: 'arrow-right'
|
|
},
|
|
class: ['text-muted'],
|
|
style: {
|
|
height: '16px'
|
|
}
|
|
});
|
|
|
|
return h(
|
|
'button',
|
|
{
|
|
class: ['btn btn-sm btn-outline-light border d-flex'],
|
|
attrs: {
|
|
type: 'button'
|
|
},
|
|
on: {
|
|
click: () => {
|
|
this.$router.push(`/edit/${doctype}/${name}`);
|
|
}
|
|
}
|
|
},
|
|
[arrow]
|
|
);
|
|
},
|
|
getTarget() {
|
|
return this.docfield.target;
|
|
},
|
|
sort() {
|
|
return (a, b) => {
|
|
a = a.toLowerCase();
|
|
b = b.toLowerCase();
|
|
|
|
if (a.value === '__newItem') {
|
|
return 1;
|
|
}
|
|
if (b.value === '___newItem') {
|
|
return -1;
|
|
}
|
|
if (a.value === b.value) {
|
|
return 0;
|
|
}
|
|
if (a.value < b.value) {
|
|
return -1;
|
|
}
|
|
if (a.value > b.value) {
|
|
return 1;
|
|
}
|
|
};
|
|
},
|
|
filter() {
|
|
return (suggestion, txt) => {
|
|
if (suggestion.value === '__newItem') {
|
|
return true;
|
|
}
|
|
};
|
|
},
|
|
onItemClick(item) {
|
|
if (item.value === '__newItem') {
|
|
this.openFormModal(item.filters);
|
|
} else {
|
|
this.handleChange(item.value);
|
|
}
|
|
},
|
|
async openFormModal(filters) {
|
|
const input = this.$refs.input;
|
|
const newDoc = await frappe.getNewDoc(this.getTarget());
|
|
let defaultValues = {};
|
|
if (filters) {
|
|
for (let key of Object.keys(filters)) {
|
|
defaultValues[key] = filters[key];
|
|
}
|
|
}
|
|
defaultValues.name = input.value !== '__newItem' ? input.value : null;
|
|
this.$formModal.open(newDoc, {
|
|
defaultValues,
|
|
onClose: () => {
|
|
// if new doc was not created
|
|
// then reset the input value
|
|
if (this.value === '__newItem') {
|
|
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>
|