2
0
mirror of https://github.com/frappe/books.git synced 2025-01-23 15:18:24 +00:00

feat: Use InvoiceForm for PurchaseInvoice as well

- Button with icon prop
This commit is contained in:
Faris Ansari 2019-10-14 02:49:28 +05:30
parent 1537ea4609
commit bd5d812d77
13 changed files with 80 additions and 42 deletions

View File

@ -26,7 +26,7 @@ module.exports = {
getFilters: (query, doc) => {
return {
isGroup: 0,
accountType: doc.customer === 1 ? 'Receivable' : 'Payable'
accountType: doc.customer ? 'Receivable' : 'Payable'
};
}
},

View File

@ -17,7 +17,7 @@ module.exports = {
fieldname: 'date',
label: 'Date',
fieldtype: 'Date',
defaultValue: new Date().toISOString()
default: new Date().toISOString().slice(0, 10)
},
{
fieldname: 'supplier',

View File

@ -4,6 +4,7 @@ import indicators from 'frappejs/ui/constants/indicators';
export default {
doctype: 'PurchaseInvoice',
title: _('Purchase Invoice'),
formRoute: name => `/edit/PurchaseInvoice/${name}`,
columns: [
'supplier',
{

View File

@ -45,7 +45,7 @@ module.exports = class PurchaseInvoiceServer extends PurchaseInvoice {
await entries.validateEntries();
}
async afterSubmit() {
async beforeSubmit() {
const entries = await this.getPosting();
await entries.post();
await frappe.db.setValue(

View File

@ -6,6 +6,13 @@ module.exports = {
isChild: 1,
keywordFields: [],
layout: 'ratio',
tableFields: [
'item',
'tax',
'quantity',
'rate',
'amount'
],
fields: [
{
fieldname: 'item',

View File

@ -20,7 +20,7 @@ module.exports = {
fieldname: 'date',
label: 'Date',
fieldtype: 'Date',
defaultValue: new Date().toISOString()
default: new Date().toISOString().slice(0, 10)
},
{
fieldname: 'customer',

View File

@ -57,7 +57,7 @@ module.exports = class SalesInvoiceServer extends SalesInvoice {
await entries.validateEntries();
}
async afterSubmit() {
async beforeSubmit() {
const entries = await this.getPosting();
await entries.post();
await frappe.db.setValue(

View File

@ -1,6 +1,6 @@
<template>
<button
class="px-4 py-2 focus:outline-none rounded-6px"
class="focus:outline-none rounded-6px"
:style="style"
v-bind="$attrs"
v-on="$listeners"
@ -15,11 +15,16 @@ export default {
type: {
type: String,
default: 'secondary'
},
icon: {
type: Boolean,
default: false
}
},
computed: {
style() {
return {
padding: this.icon ? '9px 15px' : '6px 24px',
'background-image':
this.type === 'primary'
? 'linear-gradient(180deg, #4AC3F8 0%, #2490EF 100%)'

View File

@ -2,7 +2,7 @@
<div class="flex">
<Sidebar class="w-56" />
<div class="flex flex-1 overflow-y-hidden bg-white">
<keep-alive exclude="ListView">
<keep-alive include="InvoiceForm">
<router-view class="flex-1" :key="$route.fullPath" />
</keep-alive>
<div class="flex" v-if="$route.query.edit && $route.query.doctype && $route.query.name">

View File

@ -3,11 +3,26 @@
<PageHeader>
<a class="cursor-pointer font-semibold" slot="title" @click="$router.go(-1)">{{ _('Back') }}</a>
<template slot="actions">
<Button class="text-gray-900 text-xs">Customise</Button>
<Button v-if="doc._notInserted || doc._dirty" type="primary" class="text-white text-xs ml-2" @click="onSaveClick">{{ _('Save') }}</Button>
<Button class="text-gray-900 text-xs">{{ _('Customise') }}</Button>
<Button
v-if="doc._notInserted || doc._dirty"
type="primary"
class="text-white text-xs ml-2"
@click="onSaveClick"
>{{ _('Save') }}</Button>
<Button
v-if="!doc._dirty && !doc._notInserted && !doc.submitted"
type="primary"
class="text-white text-xs ml-2"
@click="onSubmitClick"
>{{ _('Submit') }}</Button>
</template>
</PageHeader>
<div class="flex justify-center flex-1 mb-8 mt-6" v-if="meta">
<div
class="flex justify-center flex-1 mb-8 mt-6"
v-if="meta"
:class="doc.submitted && 'pointer-events-none'"
>
<div class="border rounded shadow h-full flex flex-col justify-between" style="width: 600px">
<div>
<div class="px-6 pt-6">
@ -53,11 +68,11 @@
</div>
<div class="w-1/3">
<FormControl
:df="meta.getField('customer')"
:value="doc.customer"
:placeholder="'Customer'"
@change="value => doc.set('customer', value)"
@new-doc="doc => doc.set('customer', doc.name)"
:df="meta.getField(partyField.fieldname)"
:value="doc[partyField.fieldname]"
:placeholder="partyField.label"
@change="value => doc.set(partyField.fieldname, value)"
@new-doc="doc => doc.set(partyField.fieldname, doc.name)"
input-class="bg-gray-100 rounded-lg p-2 text-right"
/>
<div
@ -86,7 +101,9 @@
<div>{{ tax.account }} ({{ tax.rate }}%)</div>
<div>{{ tax.amount }}</div>
</div>
<div class="flex pl-2 justify-between py-3 border-t text-green-600 font-semibold text-base">
<div
class="flex pl-2 justify-between py-3 border-t text-green-600 font-semibold text-base"
>
<div>Grand Total</div>
<div>{{ doc.grandTotal }}</div>
</div>
@ -105,7 +122,7 @@ import AddIcon from '@/components/Icons/Add';
export default {
name: 'InvoiceForm',
props: ['name'],
props: ['doctype', 'name'],
components: {
PageHeader,
Button,
@ -115,7 +132,7 @@ export default {
},
provide() {
return {
doctype: 'SalesInvoice',
doctype: this.doctype,
name: this.name
};
},
@ -134,12 +151,19 @@ export default {
},
itemTableColumnRatio() {
return [0.3].concat(this.itemTableFields.map(_ => 1));
}
},
partyField() {
let fieldname = {
SalesInvoice: 'customer',
PurchaseInvoice: 'supplier'
}[this.doctype];
return this.meta.getField(fieldname);
},
},
async mounted() {
this.meta = frappe.getMeta('SalesInvoice');
this.itemsMeta = frappe.getMeta('SalesInvoiceItem');
this.doc = await frappe.getDoc('SalesInvoice', this.name);
this.meta = frappe.getMeta(this.doctype);
this.itemsMeta = frappe.getMeta(`${this.doctype}Item`);
this.doc = await frappe.getDoc(this.doctype, this.name);
window.si = this.doc;
},
methods: {
@ -154,6 +178,9 @@ export default {
} else {
this.doc.update();
}
},
async onSubmitClick() {
await this.doc.submit();
}
}
};

View File

@ -26,7 +26,7 @@
</Row>
</div>
<div class="flex items-center justify-center">
<Button :class="start == 0 && 'text-gray-600'" :disabled="start == 0" @click="prevPage">
<Button :icon="true" :class="start == 0 && 'text-gray-600'" :disabled="start == 0" @click="prevPage">
<svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"
@ -45,7 +45,7 @@
<span class="text-gray-600">of</span>
<span class="font-medium">{{ totalCount }}</span>
</div>
<Button :class="start + pageLength >= totalCount && 'text-gray-600'" :disabled="start + pageLength >= totalCount" @click="nextPage">
<Button :icon="true" :class="start + pageLength >= totalCount && 'text-gray-600'" :disabled="start + pageLength >= totalCount" @click="nextPage">
<svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"

View File

@ -1,10 +1,9 @@
<template>
<div class="flex">
<div class="flex flex-col flex-1">
<div class="flex flex-col">
<PageHeader>
<h1 slot="title" class="text-xl font-bold" v-if="title">{{ title }}</h1>
<template slot="actions">
<Button type="primary" @click="makeNewDoc">
<Button :icon="true" type="primary" @click="makeNewDoc">
<Add class="w-3 h-3 stroke-current text-white" />
</Button>
<SearchBar class="ml-2" />
@ -14,7 +13,6 @@
<List :listConfig="listConfig" :filters="filters" class="flex-1" />
</div>
</div>
</div>
</template>
<script>
import frappe from 'frappejs';
@ -80,7 +78,7 @@ export default {
title: this.doctype,
doctype: this.doctype,
columns: frappe.getMeta(this.doctype).getKeywordFields()
}
};
}
},
title() {

View File

@ -1,14 +1,14 @@
<template>
<div class="border-l h-full">
<div class="flex justify-end px-4 pt-4">
<Button @click="routeToList">
<Button :icon="true" @click="routeToList">
<XIcon class="w-3 h-3 stroke-current text-gray-700" />
</Button>
<Button @click="insertDoc" type="primary" v-if="doc._notInserted" class="ml-2 flex">
<Button :icon="true" @click="insertDoc" type="primary" v-if="doc._notInserted" class="ml-2 flex">
<feather-icon name="check" class="text-white" />
</Button>
</div>
<div class="px-1 pt-2 pb-4 border-b flex items-center justify-between">
<div class="pl-1 pr-4 pt-2 pb-4 border-b flex items-center justify-between">
<FormControl
ref="titleControl"
v-if="titleDocField"