2021-11-24 14:38:13 +05:30
|
|
|
import Avatar from '@/components/Avatar';
|
|
|
|
import router from '@/router';
|
2021-11-04 15:03:51 +05:30
|
|
|
import { ipcRenderer } from 'electron';
|
2021-11-24 14:38:13 +05:30
|
|
|
import frappe from 'frappejs';
|
|
|
|
import { _ } from 'frappejs/utils';
|
|
|
|
import { IPC_ACTIONS, IPC_MESSAGES } from './messages';
|
2020-01-01 13:41:57 +05:30
|
|
|
|
2021-08-18 12:31:05 +05:30
|
|
|
export async function showMessageDialog({
|
|
|
|
message,
|
|
|
|
description,
|
2021-11-04 15:03:51 +05:30
|
|
|
buttons = [],
|
2021-08-18 12:31:05 +05:30
|
|
|
}) {
|
2021-11-04 15:03:51 +05:30
|
|
|
const options = {
|
|
|
|
message,
|
|
|
|
detail: description,
|
|
|
|
buttons: buttons.map((a) => a.label),
|
|
|
|
};
|
|
|
|
|
|
|
|
const { response } = await ipcRenderer.invoke(
|
|
|
|
IPC_ACTIONS.GET_DIALOG_RESPONSE,
|
|
|
|
options
|
2019-11-28 00:09:16 +05:30
|
|
|
);
|
2021-08-18 12:02:45 +05:30
|
|
|
|
|
|
|
let button = buttons[response];
|
|
|
|
if (button && button.action) {
|
|
|
|
button.action();
|
|
|
|
}
|
2019-11-28 00:09:16 +05:30
|
|
|
}
|
2019-12-03 13:45:12 +05:30
|
|
|
|
2021-12-10 13:36:38 +05:30
|
|
|
export async function showErrorDialog({ title, content }) {
|
|
|
|
// To be used for show stopper errors
|
|
|
|
title = title ?? 'Error';
|
|
|
|
content =
|
|
|
|
content ??
|
|
|
|
'Something has gone terribly wrong. Please check the console and raise an issue.';
|
|
|
|
|
|
|
|
await ipcRenderer.invoke(IPC_ACTIONS.SHOW_ERROR, { title, content });
|
|
|
|
}
|
|
|
|
|
2019-12-03 13:45:12 +05:30
|
|
|
export function deleteDocWithPrompt(doc) {
|
2021-11-04 15:03:51 +05:30
|
|
|
return new Promise((resolve) => {
|
2019-12-03 13:45:12 +05:30
|
|
|
showMessageDialog({
|
|
|
|
message: _('Are you sure you want to delete {0} "{1}"?', [
|
|
|
|
doc.doctype,
|
2021-11-04 15:03:51 +05:30
|
|
|
doc.name,
|
2019-12-03 13:45:12 +05:30
|
|
|
]),
|
|
|
|
description: _('This action is permanent'),
|
|
|
|
buttons: [
|
|
|
|
{
|
|
|
|
label: _('Delete'),
|
|
|
|
action: () => {
|
|
|
|
doc
|
|
|
|
.delete()
|
|
|
|
.then(() => resolve(true))
|
2021-11-04 15:03:51 +05:30
|
|
|
.catch((e) => {
|
2019-12-23 16:07:30 +05:30
|
|
|
handleErrorWithDialog(e, doc);
|
2019-12-03 13:45:12 +05:30
|
|
|
});
|
2021-11-04 15:03:51 +05:30
|
|
|
},
|
2019-12-03 13:45:12 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
label: _('Cancel'),
|
|
|
|
action() {
|
|
|
|
resolve(false);
|
2021-11-04 15:03:51 +05:30
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
2019-12-03 13:45:12 +05:30
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2019-12-03 15:53:54 +05:30
|
|
|
|
2021-11-21 19:08:04 +05:30
|
|
|
export function cancelDocWithPrompt(doc) {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
showMessageDialog({
|
|
|
|
message: _('Are you sure you want to cancel {0} "{1}"?', [
|
|
|
|
doc.doctype,
|
|
|
|
doc.name,
|
|
|
|
]),
|
|
|
|
description: _('This action is permanent'),
|
|
|
|
buttons: [
|
|
|
|
{
|
|
|
|
label: _('Yes'),
|
|
|
|
async action() {
|
|
|
|
const entryDoc = await frappe.getDoc(doc.doctype, doc.name);
|
|
|
|
entryDoc.cancelled = 1;
|
|
|
|
await entryDoc.update();
|
|
|
|
entryDoc
|
|
|
|
.revert()
|
|
|
|
.then(() => resolve(true))
|
|
|
|
.catch((e) => {
|
|
|
|
handleErrorWithDialog(e, doc);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: _('No'),
|
|
|
|
action() {
|
|
|
|
resolve(false);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-12-03 15:53:54 +05:30
|
|
|
export function partyWithAvatar(party) {
|
|
|
|
return {
|
|
|
|
data() {
|
|
|
|
return {
|
|
|
|
imageURL: null,
|
2021-11-04 15:03:51 +05:30
|
|
|
label: null,
|
2019-12-03 15:53:54 +05:30
|
|
|
};
|
|
|
|
},
|
|
|
|
components: {
|
2021-11-04 15:03:51 +05:30
|
|
|
Avatar,
|
2019-12-03 15:53:54 +05:30
|
|
|
},
|
|
|
|
async mounted() {
|
|
|
|
this.imageURL = await frappe.db.getValue('Party', party, 'image');
|
|
|
|
this.label = party;
|
|
|
|
},
|
|
|
|
template: `
|
|
|
|
<div class="flex items-center" v-if="label">
|
|
|
|
<Avatar class="flex-shrink-0" :imageURL="imageURL" :label="label" size="sm" />
|
|
|
|
<span class="ml-2 truncate">{{ label }}</span>
|
|
|
|
</div>
|
2021-11-04 15:03:51 +05:30
|
|
|
`,
|
2019-12-03 15:53:54 +05:30
|
|
|
};
|
|
|
|
}
|
2019-12-03 18:40:21 +05:30
|
|
|
|
|
|
|
export function openQuickEdit({ doctype, name, hideFields, defaults = {} }) {
|
|
|
|
let currentRoute = router.currentRoute;
|
2019-12-04 22:56:17 +05:30
|
|
|
let query = currentRoute.query;
|
|
|
|
let method = 'push';
|
|
|
|
if (query.edit && query.doctype === doctype) {
|
|
|
|
// replace the current route if we are
|
|
|
|
// editing another document of the same doctype
|
|
|
|
method = 'replace';
|
|
|
|
}
|
2021-11-24 14:38:13 +05:30
|
|
|
if (query.name === name) return;
|
2019-12-04 22:56:17 +05:30
|
|
|
router[method]({
|
2019-12-03 18:40:21 +05:30
|
|
|
query: {
|
|
|
|
edit: 1,
|
|
|
|
doctype,
|
|
|
|
name,
|
|
|
|
hideFields,
|
|
|
|
values: defaults,
|
2021-11-04 15:03:51 +05:30
|
|
|
lastRoute: currentRoute,
|
|
|
|
},
|
2019-12-03 18:40:21 +05:30
|
|
|
});
|
|
|
|
}
|
2019-12-07 00:13:37 +05:30
|
|
|
|
2020-01-29 16:31:45 +05:30
|
|
|
export function getErrorMessage(e, doc) {
|
2019-12-27 15:53:13 +05:30
|
|
|
let errorMessage = e.message || _('An error occurred');
|
2021-11-06 01:11:39 +05:30
|
|
|
const { doctype, name } = doc;
|
|
|
|
const canElaborate = doctype && name;
|
|
|
|
if (e.type === frappe.errors.LinkValidationError && canElaborate) {
|
2019-12-23 16:07:30 +05:30
|
|
|
errorMessage = _('{0} {1} is linked with existing records.', [
|
2021-11-06 01:11:39 +05:30
|
|
|
doctype,
|
|
|
|
name,
|
2019-12-23 16:07:30 +05:30
|
|
|
]);
|
2021-11-06 01:11:39 +05:30
|
|
|
} else if (e.type === frappe.errors.DuplicateEntryError && canElaborate) {
|
|
|
|
errorMessage = _('{0} {1} already exists.', [doctype, name]);
|
2019-12-10 14:55:11 +05:30
|
|
|
}
|
2020-01-29 16:31:45 +05:30
|
|
|
return errorMessage;
|
|
|
|
}
|
2019-12-23 16:07:30 +05:30
|
|
|
|
2020-01-29 16:31:45 +05:30
|
|
|
export function handleErrorWithDialog(e, doc) {
|
|
|
|
let errorMessage = getErrorMessage(e, doc);
|
|
|
|
showMessageDialog({ message: errorMessage });
|
2019-12-10 14:55:11 +05:30
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2021-11-04 15:03:51 +05:30
|
|
|
export async function makePDF(html, savePath) {
|
|
|
|
ipcRenderer.invoke(IPC_ACTIONS.SAVE_HTML_AS_PDF, html, savePath);
|
2019-12-07 00:13:37 +05:30
|
|
|
}
|
2019-12-20 12:14:31 +05:30
|
|
|
|
2021-12-09 15:55:37 +05:30
|
|
|
export async function makeJSON(data, savePath) {
|
|
|
|
ipcRenderer.invoke(IPC_ACTIONS.SAVE_REPORT_AS_JSON, data, savePath);
|
|
|
|
}
|
|
|
|
|
2019-12-20 12:14:31 +05:30
|
|
|
export function getActionsForDocument(doc) {
|
|
|
|
if (!doc) return [];
|
|
|
|
|
|
|
|
let deleteAction = {
|
|
|
|
component: {
|
2021-11-04 15:03:51 +05:30
|
|
|
template: `<span class="text-red-700">{{ _('Delete') }}</span>`,
|
2019-12-20 12:14:31 +05:30
|
|
|
},
|
2021-11-24 14:38:13 +05:30
|
|
|
condition: (doc) =>
|
|
|
|
!doc.isNew() && !doc.submitted && !doc.meta.isSingle && !doc.cancelled,
|
2019-12-20 12:14:31 +05:30
|
|
|
action: () =>
|
2021-11-04 15:03:51 +05:30
|
|
|
deleteDocWithPrompt(doc).then((res) => {
|
2019-12-20 12:14:31 +05:30
|
|
|
if (res) {
|
2021-11-21 21:45:27 +05:30
|
|
|
routeTo(`/list/${doc.doctype}`);
|
2019-12-20 12:14:31 +05:30
|
|
|
}
|
2021-11-04 15:03:51 +05:30
|
|
|
}),
|
2019-12-20 12:14:31 +05:30
|
|
|
};
|
|
|
|
|
2021-11-21 19:08:04 +05:30
|
|
|
let cancelAction = {
|
|
|
|
component: {
|
|
|
|
template: `<span class="text-red-700">{{ _('Cancel') }}</span>`,
|
|
|
|
},
|
|
|
|
condition: (doc) => doc.submitted && !doc.cancelled,
|
|
|
|
action: () => {
|
|
|
|
cancelDocWithPrompt(doc).then((res) => {
|
|
|
|
if (res) {
|
|
|
|
router.push(`/list/${doc.doctype}`);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
let actions = [...(doc.meta.actions || []), deleteAction, cancelAction]
|
2021-11-04 15:03:51 +05:30
|
|
|
.filter((d) => (d.condition ? d.condition(doc) : true))
|
|
|
|
.map((d) => {
|
2019-12-20 12:14:31 +05:30
|
|
|
return {
|
|
|
|
label: d.label,
|
|
|
|
component: d.component,
|
2021-11-04 15:03:51 +05:30
|
|
|
action: d.action.bind(this, doc, router),
|
2019-12-20 12:14:31 +05:30
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
return actions;
|
|
|
|
}
|
2020-01-02 22:36:57 +05:30
|
|
|
|
2021-11-04 15:03:51 +05:30
|
|
|
export async function runWindowAction(name) {
|
|
|
|
switch (name) {
|
|
|
|
case 'close':
|
|
|
|
ipcRenderer.send(IPC_MESSAGES.CLOSE_CURRENT_WINDOW);
|
|
|
|
break;
|
|
|
|
case 'minimize':
|
|
|
|
ipcRenderer.send(IPC_MESSAGES.MINIMIZE_CURRENT_WINDOW);
|
|
|
|
break;
|
|
|
|
case 'maximize':
|
|
|
|
const maximizing = await ipcRenderer.invoke(
|
|
|
|
IPC_ACTIONS.TOGGLE_MAXIMIZE_CURRENT_WINDOW
|
|
|
|
);
|
|
|
|
name = maximizing ? name : 'unmaximize';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return name;
|
2021-11-11 13:41:33 +05:30
|
|
|
}
|
|
|
|
|
2021-11-11 15:05:34 +05:30
|
|
|
export const statusColor = {
|
|
|
|
Draft: 'gray',
|
|
|
|
Unpaid: 'orange',
|
|
|
|
Paid: 'green',
|
2021-11-21 19:08:04 +05:30
|
|
|
Cancelled: 'red',
|
2021-11-11 15:05:34 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
export function getInvoiceStatus(doc) {
|
2021-11-11 13:41:33 +05:30
|
|
|
let status = 'Unpaid';
|
|
|
|
if (!doc.submitted) {
|
|
|
|
status = 'Draft';
|
|
|
|
}
|
|
|
|
if (doc.submitted === 1 && doc.outstandingAmount === 0.0) {
|
|
|
|
status = 'Paid';
|
|
|
|
}
|
2021-11-21 19:08:04 +05:30
|
|
|
if (doc.cancelled === 1) {
|
|
|
|
status = 'Cancelled';
|
|
|
|
}
|
2021-11-11 15:05:34 +05:30
|
|
|
return status;
|
2021-11-11 13:41:33 +05:30
|
|
|
}
|
2021-11-21 21:45:27 +05:30
|
|
|
|
|
|
|
export function routeTo(route) {
|
2021-11-29 16:17:07 +05:30
|
|
|
let routeOptions = route;
|
2021-12-01 18:38:47 +05:30
|
|
|
if (typeof route === 'string' && route === router.currentRoute.fullPath) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-29 16:17:07 +05:30
|
|
|
if (typeof route === 'string') {
|
|
|
|
routeOptions = { path: route };
|
|
|
|
}
|
|
|
|
|
2021-12-01 18:38:47 +05:30
|
|
|
router.push(routeOptions);
|
2021-11-21 21:45:27 +05:30
|
|
|
}
|
2021-11-24 14:38:13 +05:30
|
|
|
|
2021-11-26 12:51:47 +05:30
|
|
|
export function fuzzyMatch(keyword, candidate) {
|
|
|
|
const keywordLetters = [...keyword];
|
|
|
|
const candidateLetters = [...candidate];
|
|
|
|
|
|
|
|
let keywordLetter = keywordLetters.shift();
|
|
|
|
let candidateLetter = candidateLetters.shift();
|
|
|
|
|
|
|
|
let isMatch = true;
|
|
|
|
let distance = 0;
|
|
|
|
|
|
|
|
while (keywordLetter && candidateLetter) {
|
|
|
|
if (keywordLetter.toLowerCase() === candidateLetter.toLowerCase()) {
|
|
|
|
keywordLetter = keywordLetters.shift();
|
|
|
|
} else {
|
|
|
|
distance += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
candidateLetter = candidateLetters.shift();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (keywordLetter !== undefined) {
|
|
|
|
distance = -1;
|
|
|
|
isMatch = false;
|
|
|
|
} else {
|
|
|
|
distance += candidateLetters.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return { isMatch, distance };
|
|
|
|
}
|
2021-11-29 16:17:07 +05:30
|
|
|
|
|
|
|
export function openSettings(tab) {
|
|
|
|
routeTo({ path: '/settings', query: { tab } });
|
|
|
|
}
|
2021-12-21 13:24:13 +05:30
|
|
|
|
|
|
|
export async function getSavePath(name, extention) {
|
|
|
|
let { canceled, filePath } = await ipcRenderer.invoke(
|
|
|
|
IPC_ACTIONS.GET_SAVE_FILEPATH,
|
|
|
|
{
|
|
|
|
title: _('Select Folder'),
|
|
|
|
defaultPath: `${name}.${extention}`,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
if (filePath && !filePath.endsWith(extention)) {
|
|
|
|
filePath = filePath + extention;
|
|
|
|
}
|
|
|
|
|
|
|
|
return { canceled, filePath };
|
|
|
|
}
|