2
0
mirror of https://github.com/frappe/books.git synced 2025-01-25 16:18:33 +00:00

Merge pull request #424 from frappe/minor-fixes-two

fix: a bunch of minor issues
This commit is contained in:
Alan 2022-07-26 19:24:42 +05:30 committed by GitHub
commit 8c7088d752
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 123 additions and 43 deletions

View File

@ -394,7 +394,7 @@ export class Doc extends Observable<DocValue | Doc[]> {
} }
if (filterComputed) { if (filterComputed) {
fields = this.schema.fields.filter((f) => !f.computed); fields = fields.filter((f) => !f.computed);
} }
const data: DocValueMap = {}; const data: DocValueMap = {};

View File

@ -1,5 +1,5 @@
import { app, dialog, ipcMain } from 'electron'; import { app, dialog, ipcMain } from 'electron';
import { autoUpdater } from 'electron-updater'; import { autoUpdater, UpdateInfo } from 'electron-updater';
import fs from 'fs/promises'; import fs from 'fs/promises';
import path from 'path'; import path from 'path';
import databaseManager from '../backend/database/manager'; import databaseManager from '../backend/database/manager';
@ -15,6 +15,40 @@ import {
} from './helpers'; } from './helpers';
import { saveHtmlAsPdf } from './saveHtmlAsPdf'; import { saveHtmlAsPdf } from './saveHtmlAsPdf';
autoUpdater.autoDownload = false;
autoUpdater.on('error', (error) => {
dialog.showErrorBox(
'Update Error: ',
error == null ? 'unknown' : (error.stack || error).toString()
);
});
autoUpdater.on('update-available', async (info: UpdateInfo) => {
const currentVersion = app.getVersion();
const nextVersion = info.version;
const isCurrentBeta = currentVersion.includes('beta');
const isNextBeta = nextVersion.includes('beta');
let downloadUpdate = true;
if (!isCurrentBeta && isNextBeta) {
const option = await dialog.showMessageBox({
type: 'info',
title: `Update Frappe Books?`,
message: `Download version ${nextVersion}?`,
buttons: ['Yes', 'No'],
});
downloadUpdate = option.response === 0;
}
if (!downloadUpdate) {
return;
}
await autoUpdater.downloadUpdate();
});
export default function registerIpcMainActionListeners(main: Main) { export default function registerIpcMainActionListeners(main: Main) {
ipcMain.handle(IPC_ACTIONS.GET_OPEN_FILEPATH, async (event, options) => { ipcMain.handle(IPC_ACTIONS.GET_OPEN_FILEPATH, async (event, options) => {
return await dialog.showOpenDialog(main.mainWindow!, options); return await dialog.showOpenDialog(main.mainWindow!, options);
@ -51,9 +85,9 @@ export default function registerIpcMainActionListeners(main: Main) {
sendError(bodyJson); sendError(bodyJson);
}); });
ipcMain.handle(IPC_ACTIONS.CHECK_FOR_UPDATES, () => { ipcMain.handle(IPC_ACTIONS.CHECK_FOR_UPDATES, async () => {
if (!main.isDevelopment && !main.checkedForUpdate) { if (!main.isDevelopment && !main.checkedForUpdate) {
autoUpdater.checkForUpdates(); await autoUpdater.checkForUpdates();
main.checkedForUpdate = true; main.checkedForUpdate = true;
} }
}); });

View File

@ -539,7 +539,9 @@ function pruneAccountTree(accountTree: AccountTree) {
} }
for (const root of Object.keys(accountTree)) { for (const root of Object.keys(accountTree)) {
accountTree[root].children = getPrunedChildren(accountTree[root].children!); accountTree[root].children = getPrunedChildren(
accountTree[root].children ?? []
);
} }
} }

View File

@ -74,21 +74,65 @@ export class ProfitAndLoss extends AccountReport {
async getReportDataFromRows( async getReportDataFromRows(
incomeRows: ReportData, incomeRows: ReportData,
expenseRows: ReportData, expenseRows: ReportData,
incomeRoot: AccountTreeNode, incomeRoot: AccountTreeNode | undefined,
expenseRoot: AccountTreeNode expenseRoot: AccountTreeNode | undefined
): Promise<ReportData> { ): Promise<ReportData> {
if (incomeRoot && !expenseRoot) {
return await this.getIncomeOrExpenseRows(
incomeRoot,
incomeRows,
t`Total Income (Credit)`
);
}
if (expenseRoot && !incomeRoot) {
return await this.getIncomeOrExpenseRows(
expenseRoot,
expenseRows,
t`Total Income (Credit)`
);
}
if (!incomeRoot || !expenseRoot) { if (!incomeRoot || !expenseRoot) {
return []; return [];
} }
return await this.getIncomeAndExpenseRows(
incomeRows,
expenseRows,
incomeRoot,
expenseRoot
);
}
async getIncomeOrExpenseRows(
root: AccountTreeNode,
rows: ReportData,
totalRowName: string
): Promise<ReportData> {
const total = await this.getTotalNode(root, totalRowName);
const totalRow = this.getRowFromAccountListNode(total);
return [rows, totalRow].flat();
}
async getIncomeAndExpenseRows(
incomeRows: ReportData,
expenseRows: ReportData,
incomeRoot: AccountTreeNode,
expenseRoot: AccountTreeNode
) {
const totalIncome = await this.getTotalNode( const totalIncome = await this.getTotalNode(
incomeRoot, incomeRoot,
t`Total Income (Credit)` t`Total Income (Credit)`
); );
const totalIncomeRow = this.getRowFromAccountListNode(totalIncome);
const totalExpense = await this.getTotalNode( const totalExpense = await this.getTotalNode(
expenseRoot, expenseRoot,
t`Total Expense (Debit)` t`Total Expense (Debit)`
); );
const totalExpenseRow = this.getRowFromAccountListNode(totalExpense);
const totalValueMap: ValueMap = new Map(); const totalValueMap: ValueMap = new Map();
for (const key of totalIncome.valueMap!.keys()) { for (const key of totalIncome.valueMap!.keys()) {
@ -103,9 +147,6 @@ export class ProfitAndLoss extends AccountReport {
level: 0, level: 0,
} as AccountListNode; } as AccountListNode;
const totalIncomeRow = this.getRowFromAccountListNode(totalIncome);
const totalExpenseRow = this.getRowFromAccountListNode(totalExpense);
const totalProfitRow = this.getRowFromAccountListNode(totalProfit); const totalProfitRow = this.getRowFromAccountListNode(totalProfit);
totalProfitRow.cells.forEach((c) => { totalProfitRow.cells.forEach((c) => {
c.bold = true; c.bold = true;

View File

@ -175,7 +175,6 @@ export default {
this.filters = this.filters.filter((f) => f !== filter); this.filters = this.filters.filter((f) => f !== filter);
}, },
setFilter(filters, implicit) { setFilter(filters, implicit) {
console.log(filters);
this.filters = []; this.filters = [];
Object.keys(filters).map((fieldname) => { Object.keys(filters).map((fieldname) => {

View File

@ -1,15 +1,10 @@
<template> <template>
<div <div
class=" class="px-4 flex justify-between items-center h-row-largest flex-shrink-0"
px-4 :class="[
flex border ? 'border-b' : '',
justify-between platform !== 'Windows' ? 'window-drag' : '',
window-drag ]"
items-center
h-row-largest
flex-shrink-0
"
:class="border ? 'border-b' : ''"
> >
<h1 class="text-xl font-semibold select-none" v-if="title"> <h1 class="text-xl font-semibold select-none" v-if="title">
{{ title }} {{ title }}

View File

@ -121,9 +121,11 @@ export default {
}, },
setActiveTab() { setActiveTab() {
const { tab } = this.$route.query; const { tab } = this.$route.query;
const index = this.tabs.findIndex((i) => i.key === tab || 'Invoice'); const index = this.tabs.findIndex((i) => i.key === tab);
if (index !== -1) { if (index !== -1) {
this.activeTab = index; this.activeTab = index;
} else {
this.activeTab = 0;
} }
}, },
getIconComponent(tab) { getIconComponent(tab) {

View File

@ -1,5 +1,8 @@
<template> <template>
<div class="flex-1 bg-gray-25 flex justify-center items-center window-drag"> <div
class="flex-1 bg-gray-25 flex justify-center items-center window-drag"
:class="{ 'window-drag': platform !== 'Windows' }"
>
<!-- Setup Wizard Slide --> <!-- Setup Wizard Slide -->
<Slide <Slide
:primary-disabled="!valuesFilled || loading" :primary-disabled="!valuesFilled || loading"

View File

@ -1,35 +1,39 @@
import { Directive } from 'vue'; import { Directive } from 'vue';
const instances: OutsideClickCallback[] = [];
type OutsideClickCallback = (e: Event) => void; type OutsideClickCallback = (e: Event) => void;
const instanceMap: Map<HTMLElement, OutsideClickCallback> = new Map();
export const outsideClickDirective: Directive< export const outsideClickDirective: Directive<
HTMLElement, HTMLElement,
OutsideClickCallback OutsideClickCallback
> = { > = {
beforeMount(el, binding) { beforeMount(el, binding) {
el.dataset.outsideClickIndex = String(instances.length); const clickHandler = function (e: Event) {
onDocumentClick(e, el, binding.value);
const fn = binding.value;
const click = function (e: Event) {
onDocumentClick(e, el, fn);
}; };
document.addEventListener('click', click); removeHandlerIfPresent(el);
instances.push(click); instanceMap.set(el, clickHandler);
document.addEventListener('click', clickHandler);
}, },
unmounted(el) { unmounted(el) {
const index = parseInt(el.dataset.outsideClickIndex ?? '0'); removeHandlerIfPresent(el);
const handler = instances[index];
document.addEventListener('click', handler);
instances.splice(index, 1);
}, },
}; };
function onDocumentClick(e: Event, el: HTMLElement, fn: OutsideClickCallback) { function onDocumentClick(e: Event, el: HTMLElement, fn: OutsideClickCallback) {
const target = e.target; const target = e.target as Node;
if (el !== target && !el.contains(target)) {
fn?.(e);
}
}
if (el !== target && !el.contains(target as Node)) { function removeHandlerIfPresent(el: HTMLElement) {
fn(e); const clickHandler = instanceMap.get(el);
if (!clickHandler) {
return;
} }
instanceMap.delete(el);
document.removeEventListener('click', clickHandler);
} }

View File

@ -94,7 +94,7 @@ export function getGetStartedConfig() {
key: 'Add Customers', key: 'Add Customers',
label: t`Add Customers`, label: t`Add Customers`,
icon: 'customer', icon: 'customer',
description: t`Add a few customers to create your first invoice`, description: t`Add a few customers to create your first sales invoice`,
action: () => routeTo(`/list/Party/role/Customer/${t`Customers`}`), action: () => routeTo(`/list/Party/role/Customer/${t`Customers`}`),
fieldname: 'customerCreated', fieldname: 'customerCreated',
documentation: documentation:
@ -104,7 +104,7 @@ export function getGetStartedConfig() {
key: 'Create Sales Invoice', key: 'Create Sales Invoice',
label: t`Create Sales Invoice`, label: t`Create Sales Invoice`,
icon: 'sales-invoice', icon: 'sales-invoice',
description: t`Create your first invoice and mail it to your customer`, description: t`Create your first sales invoice for the created customer`,
action: () => routeTo('/list/SalesInvoice'), action: () => routeTo('/list/SalesInvoice'),
fieldname: 'invoiceCreated', fieldname: 'invoiceCreated',
documentation: documentation:
@ -129,7 +129,7 @@ export function getGetStartedConfig() {
key: 'Add Suppliers', key: 'Add Suppliers',
label: t`Add Suppliers`, label: t`Add Suppliers`,
icon: 'supplier', icon: 'supplier',
description: t`Add a few suppliers to create your first bill`, description: t`Add a few suppliers to create your first purchase invoice`,
action: () => routeTo(`/list/Party/role/Supplier/${t`Suppliers`}`), action: () => routeTo(`/list/Party/role/Supplier/${t`Suppliers`}`),
fieldname: 'supplierCreated', fieldname: 'supplierCreated',
}, },
@ -137,7 +137,7 @@ export function getGetStartedConfig() {
key: 'Create Purchase Invoice', key: 'Create Purchase Invoice',
label: t`Create Purchase Invoice`, label: t`Create Purchase Invoice`,
icon: 'purchase-invoice', icon: 'purchase-invoice',
description: t`Create your first bill and mail it to your supplier`, description: t`Create your first purchase invoice from the created supplier`,
action: () => routeTo('/list/PurchaseInvoice'), action: () => routeTo('/list/PurchaseInvoice'),
fieldname: 'billCreated', fieldname: 'billCreated',
documentation: documentation: