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

Merge with upstream

This commit is contained in:
Faris Ansari 2018-03-27 11:23:18 +05:30
commit ba32967956
19 changed files with 2707 additions and 2282 deletions

View File

@ -1,2 +1,2 @@
server: nodemon server.js server: nodemon start.js
watch: node_modules/.bin/rollup -c --watch watch: node_modules/.bin/rollup -c --watch

37
client/index.js Normal file
View File

@ -0,0 +1,37 @@
const GeneralLedgerView = require('../reports/GeneralLedgerView');
const frappe = require('frappejs');
module.exports = {
start() {
// require modules
frappe.registerModels(require('../models'), 'client');
frappe.registerView('List', 'ToDo', require('frappejs/models/doctype/ToDo/ToDoList.js'));
frappe.registerView('Form', 'FilterSelector', require('frappejs/models/doctype/FilterSelector/FilterSelectorForm.js'));
frappe.registerView('List', 'Account', require('../models/doctype/Account/AccountList.js'));
frappe.registerView('Form', 'Account', require('../models/doctype/Account/AccountForm.js'));
frappe.registerView('List', 'Invoice', require('../models/doctype/Invoice/InvoiceList.js'));
frappe.registerView('List', 'Customer', require('../models/doctype/Party/CustomerList.js'));
frappe.router.add('report/general-ledger', async (params) => {
if (!frappe.views.generalLedger) {
frappe.views.generalLedger = new GeneralLedgerView();
}
await frappe.views.generalLedger.show(params);
})
frappe.desk.menu.addItem('ToDo', '#list/ToDo');
frappe.desk.menu.addItem('Accounts', '#list/Account');
frappe.desk.menu.addItem('Items', '#list/Item');
frappe.desk.menu.addItem('Customers', '#list/Customer');
frappe.desk.menu.addItem('Invoice', '#list/Invoice');
frappe.desk.menu.addItem('Settings', () => frappe.desk.showFormModal('SystemSettings'));
frappe.router.default = '#list/ToDo';
frappe.router.show(window.location.hash);
}
}

View File

@ -32,5 +32,11 @@ module.exports = {
"Expense" "Expense"
] ]
} }
] ],
events: {
validate: (doc) => {
}
}
} }

View File

@ -8,7 +8,7 @@ module.exports = {
keywordFields: [ keywordFields: [
'account', 'account',
'party', 'party',
'reference_name' 'referenceName'
], ],
fields: [ fields: [
{ {

View File

@ -99,17 +99,21 @@ module.exports = {
{ fields: [ "terms" ] }, { fields: [ "terms" ] },
], ],
formEvents: { links: [
refresh: (form) => { {
if (!form.ledgerLink) { label: 'Ledger Entries',
form.ledgerLink = form.container.addLink('Ledger Entries', () => { condition: (form) => form.doc.submitted,
frappe.flags.filters = { action:(form) => {
reference_type: 'Invoice', return {
reference_name: form.doc.name route: ['table', 'AccountingLedgerEntry'],
params: {
filters: {
referenceType: 'Invoice',
referenceName: form.doc.name
}
} }
frappe.router.setRoute('table', 'AccountingLedgerEntry'); };
});
} }
} }
} ]
} }

View File

@ -23,5 +23,23 @@ module.exports = {
"label": "Supplier", "label": "Supplier",
"fieldtype": "Check" "fieldtype": "Check"
} }
],
links: [
{
label: 'Invoices',
condition: (form) => form.doc.customer,
action: (form) => {
return {
route: ['table', 'Invoice'],
params: {
filters: {
customer: form.doc.name,
}
}
};
}
}
] ]
} }

19
reports/GeneralLedger.js Normal file
View File

@ -0,0 +1,19 @@
const frappe = require('frappejs');
module.exports = class GeneralLedger {
async run(params) {
const filters = {};
if (params.account) filters.account = params.account;
if (params.party) filters.party = params.party;
if (params.from_date) filters.date = ('>=', params.from_date);
if (params.to_date) filters.date = ('<=', params.to_date);
let data = await frappe.db.getAll({
doctype: 'AccountingLedgerEntry',
fields: ['date', 'account', 'party', 'referenceType', 'referenceName', 'debit', 'credit'],
filters: filters
});
return data;
}
}

View File

@ -0,0 +1,28 @@
const ReportPage = require('frappejs/client/desk/reportpage');
const frappe = require('frappejs');
module.exports = class GeneralLedgerView extends ReportPage {
constructor() {
super({title: frappe._('General Ledger')});
this.addFilter({fieldtype: 'Link', target: 'Account', label: 'Account'});
this.addFilter({fieldtype: 'Link', target: 'Party', label: 'Party'});
this.addFilter({fieldtype: 'Date', label: 'From Date'});
this.addFilter({fieldtype: 'Date', label: 'To Date'});
this.method = 'general-ledger';
}
getColumns() {
return [
{label: 'Date', fieldtype: 'Date'},
{label: 'Account', fieldtype: 'Link'},
{label: 'Party', fieldtype: 'Link'},
{label: 'Description', fieldtype: 'Data'},
{label: 'Debit', fieldtype: 'Currency'},
{label: 'Credit', fieldtype: 'Currency'},
{label: 'Balance', fieldtype: 'Currency'}
]
}
}

View File

@ -1,15 +0,0 @@
const server = require('frappejs/server');
const frappe = require('frappejs');
server.start({
backend: 'sqlite',
connectionParams: {dbPath: 'test.db'},
static: './',
models: require('./models')
}).then(() => {
// set server-side modules
frappe.models.Invoice.documentClass = require('./models/doctype/Invoice/InvoiceServer.js');
frappe.metaCache = {};
frappe.syncDoc(require('./fixtures/invoicePrint'));
});

28
server/index.js Normal file
View File

@ -0,0 +1,28 @@
const server = require('frappejs/server');
const frappe = require('frappejs');
const GeneralLedger = require('../reports/GeneralLedger')
module.exports = {
async start() {
await server.start({
backend: 'sqlite',
connectionParams: {dbPath: 'test.db'},
staticPath: './www',
models: require('../models')
})
// set server-side modules
frappe.models.Invoice.documentClass = require('../models/doctype/Invoice/InvoiceServer.js');
frappe.metaCache = {};
frappe.syncDoc(require('../fixtures/invoicePrint'));
frappe.registerMethod({
method: 'general-ledger',
handler: async (args) => {
const generalLedger = new GeneralLedger();
return await generalLedger.run(args);
}
});
}
}

View File

@ -1,33 +0,0 @@
const client = require('frappejs/client');
// start server
client.start({
columns: 3,
server: 'localhost:8000'
}).then(() => {
// require modules
frappe.registerModels(require('../models'), 'client');
frappe.registerView('List', 'ToDo', require('frappejs/models/doctype/ToDo/ToDoList.js'));
frappe.registerView('Form', 'FilterSelector', require('frappejs/models/doctype/FilterSelector/FilterSelectorForm.js'));
frappe.registerView('List', 'Account', require('../models/doctype/Account/AccountList.js'));
frappe.registerView('Form', 'Account', require('../models/doctype/Account/AccountForm.js'));
frappe.registerView('List', 'Invoice', require('../models/doctype/Invoice/InvoiceList.js'));
frappe.registerView('List', 'Customer', require('../models/doctype/Party/CustomerList.js'));
frappe.desk.menu.addItem('ToDo', '#list/ToDo');
frappe.desk.menu.addItem('Accounts', '#list/Account');
frappe.desk.menu.addItem('Items', '#list/Item');
frappe.desk.menu.addItem('Customers', '#list/Customer');
frappe.desk.menu.addItem('Invoice', '#list/Invoice');
frappe.desk.menu.addItem('Settings', () => frappe.desk.showFormModal('SystemSettings'));
frappe.router.default = '#list/ToDo';
frappe.router.show(window.location.hash);
});
module.exports = false;

2
start.js Normal file
View File

@ -0,0 +1,2 @@
const server = require('./server');
server.start();

View File

@ -7395,6 +7395,8 @@ html {
background-color: lightyellow; } background-color: lightyellow; }
.form-body .alert { .form-body .alert {
margin-top: 1rem; } margin-top: 1rem; }
.form-inline .form-group {
margin-right: 0.5rem; }
.list-search { .list-search {
padding: 1rem 2rem; } padding: 1rem 2rem; }
.list-body .list-row { .list-body .list-row {

View File

@ -1,9 +1,16 @@
var desk = (function () { var desk = (function () {
'use strict'; 'use strict';
Array.prototype.equals = function( array ) {
return this.length == array.length &&
this.every( function(item,i) { return item == array[i] } );
};
var utils = { var utils = {
slug(text) { slug(str) {
return text.toLowerCase().replace(/ /g, '_'); return str.replace(/(?:^\w|[A-Z]|\b\w)/g, function(letter, index) {
return index == 0 ? letter.toLowerCase() : letter.toUpperCase();
}).replace(/\s+/g, '');
}, },
getRandomString() { getRandomString() {
@ -40,7 +47,27 @@ var utils = {
: match; : match;
} }
}); });
} },
getQueryString(params) {
if (!params) return '';
let parts = [];
for (let key in params) {
if (key!=null && params[key]!=null) {
parts.push(encodeURIComponent(key) + '=' + encodeURIComponent(params[key]));
}
}
return parts.join('&');
},
asyncHandler(fn) {
return (req, res, next) => Promise.resolve(fn(req, res, next))
.catch((err) => {
console.log(err);
// handle error
res.status(err.status_code || 500).send({error: err.message});
});
},
}; };
@ -4548,7 +4575,7 @@ if (typeof undefined === 'function' && undefined.amd) {
} }
}).call(commonjsGlobal); }).call(commonjsGlobal);
//# sourceMappingURL=showdown.js.map
}); });
var moment = createCommonjsModule(function (module, exports) { var moment = createCommonjsModule(function (module, exports) {
@ -9106,6 +9133,9 @@ var frappejs = {
this.forms = {}; this.forms = {};
this.views = {}; this.views = {};
this.flags = {}; this.flags = {};
this.methods = {};
// temp params while calling routes
this.params = {};
}, },
registerLibs(common) { registerLibs(common) {
@ -9130,6 +9160,25 @@ var frappejs = {
this.views[view][name] = module; this.views[view][name] = module;
}, },
registerMethod({method, handler}) {
this.methods[method] = handler;
if (this.app) {
// add to router if client-server
this.app.post(`/api/method/${method}`, this.asyncHandler(async function(request, response) {
const data = await handler(request.body);
response.json(data);
}));
}
},
call({method, type, args}) {
if (this.methods[method]) {
return this.methods[method](args);
} else {
throw `${method} not found`;
}
},
addToCache(doc) { addToCache(doc) {
if (!this.docs) return; if (!this.docs) return;
@ -10088,7 +10137,7 @@ var http = class HTTPClient extends observable {
async getAll({ doctype, fields, filters, start, limit, sort_by, order }) { async getAll({ doctype, fields, filters, start, limit, sort_by, order }) {
let url = this.getURL('/api/resource', doctype); let url = this.getURL('/api/resource', doctype);
url = url + "?" + this.getQueryString({ url = url + "?" + frappejs.getQueryString({
fields: JSON.stringify(fields), fields: JSON.stringify(fields),
filters: JSON.stringify(filters), filters: JSON.stringify(filters),
start: start, start: start,
@ -10157,13 +10206,6 @@ var http = class HTTPClient extends observable {
return this.protocol + '://' + this.server + parts.join('/'); return this.protocol + '://' + this.server + parts.join('/');
} }
getQueryString(params) {
return Object.keys(params)
.map(k => params[k] != null ? encodeURIComponent(k) + '=' + encodeURIComponent(params[k]) : null)
.filter(v => v)
.join('&');
}
getHeaders() { getHeaders() {
return { return {
'Accept': 'application/json', 'Accept': 'application/json',
@ -23004,7 +23046,7 @@ Popper.placements = placements;
Popper.Defaults = Defaults; Popper.Defaults = Defaults;
//# sourceMappingURL=popper.js.map
var popper = Object.freeze({ var popper = Object.freeze({
@ -26903,7 +26945,7 @@ exports.Tooltip = Tooltip;
Object.defineProperty(exports, '__esModule', { value: true }); Object.defineProperty(exports, '__esModule', { value: true });
}))); })));
//# sourceMappingURL=bootstrap.js.map
}); });
unwrapExports(bootstrap); unwrapExports(bootstrap);
@ -27200,6 +27242,10 @@ var page = class Page extends observable {
return link; return link;
} }
clearLinks() {
frappejs.ui.empty(this.linksElement);
}
hide() { hide() {
this.parent.activePage = null; this.parent.activePage = null;
this.wrapper.classList.add('hide'); this.wrapper.classList.add('hide');
@ -27235,6 +27281,8 @@ var page = class Page extends observable {
} }
this.parent.activePage = this; this.parent.activePage = this;
frappejs.desk.toggleCenter(this.fullPage ? false : true);
} }
renderError(title, message) { renderError(title, message) {
@ -27672,6 +27720,9 @@ class BaseControl {
makeLabel(labelClass = null) { makeLabel(labelClass = null) {
this.labelElement = frappejs.ui.add('label', labelClass, this.inputContainer, this.label); this.labelElement = frappejs.ui.add('label', labelClass, this.inputContainer, this.label);
this.labelElement.setAttribute('for', this.id); this.labelElement.setAttribute('for', this.id);
if (this.inline) {
this.labelElement.classList.add("sr-only");
}
} }
makeInput(inputClass='form-control') { makeInput(inputClass='form-control') {
@ -27685,6 +27736,9 @@ class BaseControl {
if (!this.onlyInput) { if (!this.onlyInput) {
this.makeDescription(); this.makeDescription();
} }
if (this.placeholder) {
this.input.setAttribute('placeholder', this.placeholder);
}
} }
@ -39742,6 +39796,11 @@ var htmlmixed = createCommonjsModule(function (module, exports) {
}); });
}); });
// const frappe = require('frappejs');
// eslint-disable-line
// eslint-disable-line
class CodeControl extends base { class CodeControl extends base {
makeInput() { makeInput() {
if (!this.options) { if (!this.options) {
@ -47881,8 +47940,12 @@ class DataTable {
this.options || {}, options this.options || {}, options
); );
this.options.headerDropdown this.options.headerDropdown =
.push(...(options.headerDropdown || [])); DEFAULT_OPTIONS.headerDropdown
.concat(
this.options.headerDropdown || [],
options.headerDropdown || []
);
// custom user events // custom user events
this.events = Object.assign( this.events = Object.assign(
@ -48076,6 +48139,9 @@ DataTable.__version__ = packageJson.version;
module.exports = DataTable; module.exports = DataTable;
}); });
// eslint-disable-line
var modal = class Modal extends observable { var modal = class Modal extends observable {
constructor({ title, body, primary, secondary }) { constructor({ title, body, primary, secondary }) {
super(); super();
@ -48152,8 +48218,35 @@ var modal = class Modal extends observable {
} }
}; };
var utils$3 = {
convertFieldsToDatatableColumns(fields, layout = 'fixed') {
return fields.map(field => {
if (!field.width) {
if (layout==='ratio') {
field.width = 1;
} else if (layout==='fixed') {
field.width = 120;
}
}
return {
id: field.fieldname || frappejs.slug(field.label),
field: field,
content: field.label,
editable: true,
sortable: false,
resizable: true,
dropdown: false,
width: field.width,
align: ['Int', 'Float', 'Currency'].includes(field.fieldtype) ? 'right' : 'left',
format: (value) => frappejs.format(value, field)
}
});
}
};
var modelTable = class ModelTable { var modelTable = class ModelTable {
constructor({doctype, parent, layout, parentControl, getRowDoc, constructor({doctype, parent, layout, parentControl, getRowData,
isDisabled, getTableData}) { isDisabled, getTableData}) {
Object.assign(this, arguments[0]); Object.assign(this, arguments[0]);
this.meta = frappejs.getMeta(this.doctype); this.meta = frappejs.getMeta(this.doctype);
@ -48175,27 +48268,7 @@ var modelTable = class ModelTable {
} }
getColumns() { getColumns() {
return this.getTableFields().map(field => { return utils$3.convertFieldsToDatatableColumns(this.getTableFields(), this.layout);
if (!field.width) {
if (this.layout==='ratio') {
field.width = 1;
} else if (this.layout==='fixed') {
field.width = 120;
}
}
return {
id: field.fieldname,
field: field,
content: field.label,
editable: true,
sortable: false,
resizable: true,
dropdown: false,
width: field.width,
align: ['Int', 'Float', 'Currency'].includes(field.fieldtype) ? 'right' : 'left',
format: (value) => frappejs.format(value, field)
}
});
} }
getTableFields() { getTableFields() {
@ -48224,17 +48297,16 @@ var modelTable = class ModelTable {
control.skipChangeEvent = true; control.skipChangeEvent = true;
return { return {
initValue: (value, rowIndex, column) => { initValue: async (value, rowIndex, column) => {
let doc = this.getRowDoc(rowIndex);
column.activeControl = control; column.activeControl = control;
control.parentControl = this.parentControl; control.parentControl = this.parentControl;
control.doc = doc; control.doc = await this.getRowData(rowIndex);
control.setFocus(); control.setFocus();
control.setInputValue(control.doc[column.id]); control.setInputValue(control.doc[column.id]);
return control; return control;
}, },
setValue: async (value, rowIndex, column) => { setValue: async (value, rowIndex, column) => {
control.handleChange(); await this.setValue(control);
}, },
getValue: () => { getValue: () => {
return control.getInputValue(); return control.getInputValue();
@ -48243,6 +48315,10 @@ var modelTable = class ModelTable {
} }
async setValue(control) {
await control.handleChange();
}
getControlModal(field) { getControlModal(field) {
this.modal = new modal({ this.modal = new modal({
title: frappejs._('Edit {0}', field.label), title: frappejs._('Edit {0}', field.label),
@ -48305,9 +48381,9 @@ class TableControl extends base {
parent: this.wrapper.querySelector('.datatable-wrapper'), parent: this.wrapper.querySelector('.datatable-wrapper'),
parentControl: this, parentControl: this,
layout: this.layout || 'ratio', layout: this.layout || 'ratio',
getRowDoc: (rowIndex) => this.doc[this.fieldname][rowIndex], getTableData: () => this.getTableData(),
getRowData: (rowIndex) => this.doc[this.fieldname][rowIndex],
isDisabled: () => this.isDisabled(), isDisabled: () => this.isDisabled(),
getTableData: () => this.getTableData()
}); });
this.setupToolbar(); this.setupToolbar();
} }
@ -48439,6 +48515,7 @@ var form = class BaseForm extends observable {
this.controls = {}; this.controls = {};
this.controlList = []; this.controlList = [];
this.sections = []; this.sections = [];
this.links = [];
this.meta = frappejs.getMeta(this.doctype); this.meta = frappejs.getMeta(this.doctype);
if (this.setup) { if (this.setup) {
@ -48622,6 +48699,49 @@ var form = class BaseForm extends observable {
} }
} }
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) {
this.container.clearLinks();
for(let link of links) {
// make the link
this.container.addLink(link.label, () => {
let options = link.action(this);
if (options) {
if (options.params) {
// set route parameters
frappejs.params = options.params;
}
if (options.route) {
// go to the given route
frappejs.router.setRoute(...options.route);
}
}
});
}
}
async bindEvents(doc) { async bindEvents(doc) {
if (this.doc && this.docListener) { if (this.doc && this.docListener) {
// stop listening to the old doc // stop listening to the old doc
@ -48679,6 +48799,7 @@ var form = class BaseForm extends observable {
control.refresh(); control.refresh();
} }
this.trigger('refresh', this); this.trigger('refresh', this);
this.setLinks();
} }
async submit() { async submit() {
@ -56545,7 +56666,7 @@ module.exports = installCompat;
/***/ }) /***/ })
/******/ ]); /******/ ]);
}); });
//# sourceMappingURL=nunjucks.js.map
}); });
unwrapExports(nunjucks); unwrapExports(nunjucks);
@ -56666,16 +56787,23 @@ var tablepage = class TablePage extends page {
this.filterSelector.reset(this.doctype); this.filterSelector.reset(this.doctype);
} }
if (frappejs.flags.filters) { if (frappejs.params.filters) {
this.filterSelector.setFilters(frappejs.flags.filters); this.filterSelector.setFilters(frappejs.params.filters);
frappejs.flags.filters = null;
} }
frappejs.params = null;
if (!this.modelTable) { if (!this.modelTable) {
this.modelTable = new modelTable({ this.modelTable = new modelTable({
doctype: this.doctype, doctype: this.doctype,
parent: this.tableWrapper, parent: this.tableWrapper,
layout: 'fluid' layout: 'fluid',
getRowData: async (rowIndex) => {
return await frappejs.getDoc(this.doctype, this.data[rowIndex].name);
},
setValue: async (control) => {
await control.handleChange();
await control.doc.update();
}
}); });
} }
@ -56684,14 +56812,18 @@ var tablepage = class TablePage extends page {
async run() { async run() {
this.displayFilters(); this.displayFilters();
const data = await frappejs.db.getAll({ this.data = await frappejs.db.getAll({
doctype: this.doctype, doctype: this.doctype,
fields: ['*'], fields: ['*'],
filters: this.filterSelector.getFilters(), filters: this.filterSelector.getFilters(),
start: this.start, start: this.start,
limit: 500 limit: 500
}); });
this.modelTable.refresh(data); this.modelTable.refresh(this.data);
}
displayFilters() {
this.fitlerButton.textContent = this.filterSelector.getText();
} }
displayFilters() { displayFilters() {
@ -56739,6 +56871,10 @@ var menu = class DeskMenu {
} }
}; };
// const Search = require('./search');
const views = {}; const views = {};
views.Form = formpage; views.Form = formpage;
views.List = listpage; views.List = listpage;
@ -56867,7 +57003,6 @@ var desk = class Desk {
if (!this.pages[view]) this.pages[view] = {}; if (!this.pages[view]) this.pages[view] = {};
if (!this.pages[view][doctype]) this.pages[view][doctype] = new views[view](doctype); if (!this.pages[view][doctype]) this.pages[view][doctype] = new views[view](doctype);
const page$$1 = this.pages[view][doctype]; const page$$1 = this.pages[view][doctype];
this.toggleCenter(page$$1.fullPage ? false : true);
await page$$1.show(params); await page$$1.show(params);
} }
@ -57347,6 +57482,25 @@ var ToDo = {
"label": "Description", "label": "Description",
"fieldtype": "Text" "fieldtype": "Text"
} }
],
links: [
{
label: 'Close',
condition: (form) => form.doc.status !== 'Closed',
action: async (form) => {
await form.doc.set('status', 'Closed');
await form.doc.update();
}
},
{
label: 'Re-Open',
condition: (form) => form.doc.status !== 'Open',
action: async (form) => {
await form.doc.set('status', 'Open');
await form.doc.update();
}
}
] ]
}; };
@ -57427,6 +57581,8 @@ var client = {
frappejs.registerModels(models, 'client'); frappejs.registerModels(models, 'client');
frappejs.fetch = window.fetch.bind(); frappejs.fetch = window.fetch.bind();
this.setCall();
frappejs.db = await new http({server: server}); frappejs.db = await new http({server: server});
this.socket = io.connect('http://localhost:8000'); // eslint-disable-line this.socket = io.connect('http://localhost:8000'); // eslint-disable-line
frappejs.db.bindSocketClient(this.socket); frappejs.db.bindSocketClient(this.socket);
@ -57437,6 +57593,122 @@ var client = {
frappejs.desk = new desk(columns); frappejs.desk = new desk(columns);
await frappejs.login(); await frappejs.login();
},
setCall() {
frappejs.call = async (method, args) => {
let url = `/api/method/${method}`;
let response = await fetch(url, {
method: 'POST',
headers: {
'Accept': 'application/json',
'Content-Type': 'application/json'
},
body: JSON.stringify(args || {})
});
return await response.json();
};
}
};
// baseclass for report
// `url` url for report
// `getColumns` return columns
var reportpage = class ReportPage extends page {
constructor({title, }) {
super({title: title, hasRoute: true});
this.fullPage = true;
this.filterWrapper = frappejs.ui.add('div', 'filter-toolbar form-inline', this.body);
this.tableWrapper = frappejs.ui.add('div', 'table-page-wrapper', this.body);
this.btnNew = this.addButton(frappejs._('Refresh'), 'btn-primary', async () => {
await this.run();
});
this.filters = {};
}
getColumns() {
// overrride
}
addFilter(field) {
if (field.fieldname) {
field.fieldname = frappejs.slug(field.label);
}
field.placeholder = field.label;
field.inline = true;
this.filters[field.fieldname] = controls$1.makeControl({field: field, form: this, parent: this.filterWrapper});
return this.filters[field.fieldname];
}
getFilterValues() {
const values = {};
for (let fieldname in this.filters) {
let control = this.filters[fieldname];
values[fieldname] = control.getInputValue();
if (control.required && !values[fieldname]) {
frappejs.ui.showAlert({message: frappejs._('{0} is mandatory', control.label), color: 'red'});
return false;
}
}
return values;
}
async show(params) {
super.show();
await this.run();
}
async run() {
if (!this.datatable) {
this.makeDataTable();
}
const filterValues = this.getFilterValues();
if (filterValues === false) return;
let data = await frappejs.call(this.method, filterValues);
this.datatable.refresh(data);
}
makeDataTable() {
this.datatable = new frappeDatatable_cjs(this.tableWrapper, {
columns: utils$3.convertFieldsToDatatableColumns(this.getColumns(), this.layout),
data: [],
layout: this.layout || 'fluid',
});
}
};
var GeneralLedgerView_1 = class GeneralLedgerView extends reportpage {
constructor() {
super({title: frappejs._('General Ledger')});
this.addFilter({fieldtype: 'Link', target: 'Account', label: 'Account'});
this.addFilter({fieldtype: 'Link', target: 'Party', label: 'Party'});
this.addFilter({fieldtype: 'Date', label: 'From Date'});
this.addFilter({fieldtype: 'Date', label: 'To Date'});
this.method = 'general-ledger';
}
getColumns() {
return [
{label: 'Date', fieldtype: 'Date'},
{label: 'Account', fieldtype: 'Link'},
{label: 'Party', fieldtype: 'Link'},
{label: 'Description', fieldtype: 'Data'},
{label: 'Debit', fieldtype: 'Currency'},
{label: 'Credit', fieldtype: 'Currency'},
{label: 'Balance', fieldtype: 'Currency'}
]
} }
}; };
@ -57486,7 +57758,13 @@ var Account = {
"Expense" "Expense"
] ]
} }
] ],
events: {
validate: (doc) => {
}
}
}; };
var AccountingLedgerEntry = { var AccountingLedgerEntry = {
@ -57499,7 +57777,7 @@ var AccountingLedgerEntry = {
keywordFields: [ keywordFields: [
'account', 'account',
'party', 'party',
'reference_name' 'referenceName'
], ],
fields: [ fields: [
{ {
@ -57561,7 +57839,8 @@ var AccountingLedgerEntry = {
] ]
}; };
var Party = { var Party = createCommonjsModule(function (module) {
module.exports = {
"name": "Party", "name": "Party",
"doctype": "DocType", "doctype": "DocType",
"isSingle": 0, "isSingle": 0,
@ -57586,8 +57865,29 @@ var Party = {
"label": "Supplier", "label": "Supplier",
"fieldtype": "Check" "fieldtype": "Check"
} }
],
links: [
{
label: 'Invoices',
condition: (form) => form.doc.customer,
action: (form) => {
return {
route: ['table', 'Invoice'],
params: {
filters: {
customer: form.doc.name,
}
}
};
}
}
] ]
}; };
});
var Party_1 = Party.links;
var Item = { var Item = {
name: "Item", name: "Item",
@ -57839,24 +58139,28 @@ module.exports = {
{ fields: [ "terms" ] }, { fields: [ "terms" ] },
], ],
formEvents: { links: [
refresh: (form) => { {
if (!form.ledgerLink) { label: 'Ledger Entries',
form.ledgerLink = form.container.addLink('Ledger Entries', () => { condition: (form) => form.doc.submitted,
frappejs.flags.filters = { action:(form) => {
reference_type: 'Invoice', return {
reference_name: form.doc.name route: ['table', 'AccountingLedgerEntry'],
}; params: {
frappejs.router.setRoute('table', 'AccountingLedgerEntry'); filters: {
}); referenceType: 'Invoice',
referenceName: form.doc.name
}
}
};
} }
} }
} ]
}; };
}); });
var Invoice_1 = Invoice.layout; var Invoice_1 = Invoice.layout;
var Invoice_2 = Invoice.formEvents; var Invoice_2 = Invoice.links;
var InvoiceItem = { var InvoiceItem = {
name: "InvoiceItem", name: "InvoiceItem",
@ -58098,38 +58402,51 @@ var CustomerList_1 = class CustomerList extends list {
} }
}; };
var client$2 = {
start() {
// require modules
frappejs.registerModels(models$2, 'client');
frappejs.registerView('List', 'ToDo', ToDoList_1);
frappejs.registerView('Form', 'FilterSelector', FilterSelectorForm_1);
frappejs.registerView('List', 'Account', AccountList_1);
frappejs.registerView('Form', 'Account', AccountForm_1);
frappejs.registerView('List', 'Invoice', InvoiceList_1);
frappejs.registerView('List', 'Customer', CustomerList_1);
frappejs.router.add('report/general-ledger', async (params) => {
if (!frappejs.views.generalLedger) {
frappejs.views.generalLedger = new GeneralLedgerView_1();
}
await frappejs.views.generalLedger.show(params);
});
frappejs.desk.menu.addItem('ToDo', '#list/ToDo');
frappejs.desk.menu.addItem('Accounts', '#list/Account');
frappejs.desk.menu.addItem('Items', '#list/Item');
frappejs.desk.menu.addItem('Customers', '#list/Customer');
frappejs.desk.menu.addItem('Invoice', '#list/Invoice');
frappejs.desk.menu.addItem('Settings', () => frappejs.desk.showFormModal('SystemSettings'));
frappejs.router.default = '#list/ToDo';
frappejs.router.show(window.location.hash);
}
};
// start server // start server
client.start({ client.start({
columns: 3, columns: 3,
server: 'localhost:8000' server: 'localhost:8000'
}).then(() => { }).then(() => {
client$2.start();
// require modules
frappe.registerModels(models$2, 'client');
frappe.registerView('List', 'ToDo', ToDoList_1);
frappe.registerView('Form', 'FilterSelector', FilterSelectorForm_1);
frappe.registerView('List', 'Account', AccountList_1);
frappe.registerView('Form', 'Account', AccountForm_1);
frappe.registerView('List', 'Invoice', InvoiceList_1);
frappe.registerView('List', 'Customer', CustomerList_1);
frappe.desk.menu.addItem('ToDo', '#list/ToDo');
frappe.desk.menu.addItem('Accounts', '#list/Account');
frappe.desk.menu.addItem('Items', '#list/Item');
frappe.desk.menu.addItem('Customers', '#list/Customer');
frappe.desk.menu.addItem('Invoice', '#list/Invoice');
frappe.desk.menu.addItem('Settings', () => frappe.desk.showFormModal('SystemSettings'));
frappe.router.default = '#list/ToDo';
frappe.router.show(window.location.hash);
}); });
var src = false; var www = false;
return src; return www;
}()); }());

12
www/index.js Normal file
View File

@ -0,0 +1,12 @@
const client = require('frappejs/client');
const appClient = require('../client');
// start server
client.start({
columns: 3,
server: 'localhost:8000'
}).then(() => {
appClient.start();
});
module.exports = false;