mirror of
https://github.com/frappe/books.git
synced 2025-01-22 22:58:28 +00:00
chore: fix all fixable eslint errors in src
This commit is contained in:
parent
4415c04a88
commit
e67e6ae257
@ -17,8 +17,8 @@ module.exports = {
|
||||
'@typescript-eslint/ban-ts-comment': 'off',
|
||||
'@typescript-eslint/no-var-requires': 'off',
|
||||
'@typescript-eslint/no-non-null-assertion': 'off',
|
||||
'@typescript-eslint/no-floating-promises': 'off',
|
||||
'@typescript-eslint/no-misused-promises': 'off',
|
||||
'@typescript-eslint/no-floating-promises': 'warn',
|
||||
'@typescript-eslint/no-misused-promises': 'warn',
|
||||
},
|
||||
parser: '@typescript-eslint/parser',
|
||||
parserOptions: { project: true, tsconfigRootDir: __dirname },
|
||||
@ -28,5 +28,5 @@ module.exports = {
|
||||
'plugin:@typescript-eslint/recommended',
|
||||
'plugin:@typescript-eslint/recommended-requiring-type-checking',
|
||||
],
|
||||
ignorePatterns: ['.eslintrc.js'],
|
||||
ignorePatterns: ['.eslintrc.js', 'tailwind.config.js'],
|
||||
};
|
||||
|
@ -9,7 +9,6 @@ import { fyo } from './initFyo';
|
||||
import router from './router';
|
||||
import { getErrorMessage, stringifyCircular } from './utils';
|
||||
import { DialogOptions, ToastOptions } from './utils/types';
|
||||
import { UnknownFunction } from 'utils/types';
|
||||
const { ipcRenderer } = require('electron');
|
||||
|
||||
function shouldNotStore(error: Error) {
|
||||
@ -40,6 +39,7 @@ export async function sendError(errorLogObj: ErrorLog) {
|
||||
};
|
||||
|
||||
if (fyo.store.isDevelopment) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log('sendError', body);
|
||||
}
|
||||
|
||||
@ -80,6 +80,7 @@ export async function handleError(
|
||||
notifyUser = true
|
||||
) {
|
||||
if (logToConsole) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.error(error);
|
||||
}
|
||||
|
||||
@ -93,7 +94,7 @@ export async function handleError(
|
||||
if (notifyUser) {
|
||||
const toastProps = getToastProps(errorLogObj);
|
||||
const { showToast } = await import('src/utils/interactive');
|
||||
await showToast(toastProps);
|
||||
showToast(toastProps);
|
||||
}
|
||||
}
|
||||
|
||||
@ -140,6 +141,7 @@ export async function handleErrorWithDialog(
|
||||
await showDialog(options);
|
||||
if (dontThrow) {
|
||||
if (fyo.store.isDevelopment) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.error(error);
|
||||
}
|
||||
return;
|
||||
@ -156,12 +158,14 @@ export async function showErrorDialog(title?: string, content?: string) {
|
||||
await ipcRenderer.invoke(IPC_ACTIONS.SHOW_ERROR, { title, content });
|
||||
}
|
||||
|
||||
// Wrapper Functions
|
||||
|
||||
export function getErrorHandled(func: UnknownFunction) {
|
||||
return async function errorHandled(...args: unknown[]) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
export function getErrorHandled<T extends (...args: any[]) => Promise<any>>(
|
||||
func: T
|
||||
) {
|
||||
type Return = ReturnType<T> extends Promise<infer P> ? P : true;
|
||||
return async function errorHandled(...args: Parameters<T>): Promise<Return> {
|
||||
try {
|
||||
return await func(...args);
|
||||
return (await func(...args)) as Return;
|
||||
} catch (error) {
|
||||
await handleError(false, error as Error, {
|
||||
functionName: func.name,
|
||||
@ -173,16 +177,19 @@ export function getErrorHandled(func: UnknownFunction) {
|
||||
};
|
||||
}
|
||||
|
||||
export function getErrorHandledSync(func: UnknownFunction) {
|
||||
return function errorHandledSync(...args: unknown[]) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
export function getErrorHandledSync<T extends (...args: any[]) => any>(
|
||||
func: T
|
||||
) {
|
||||
type Return = ReturnType<T> extends Promise<infer P> ? P : ReturnType<T>;
|
||||
return function errorHandledSync(...args: Parameters<T>) {
|
||||
try {
|
||||
return func(...args);
|
||||
return func(...args) as Return;
|
||||
} catch (error) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
handleError(false, error as Error, {
|
||||
functionName: func.name,
|
||||
functionArgs: args,
|
||||
}).then(() => {
|
||||
throw error;
|
||||
});
|
||||
}
|
||||
};
|
||||
|
@ -199,7 +199,7 @@ export class Importer {
|
||||
}[];
|
||||
|
||||
const cellErrors = [];
|
||||
for (const i in this.valueMatrix) {
|
||||
for (let i = 0; i < this.valueMatrix.length; i++) {
|
||||
const row = this.valueMatrix[i];
|
||||
for (const { tf, index } of assigned) {
|
||||
if (!row[index]?.error) {
|
||||
@ -278,14 +278,14 @@ export class Importer {
|
||||
return { dataMap, childTableMap };
|
||||
}
|
||||
|
||||
for (const i in this.valueMatrix) {
|
||||
for (let i = 0; i < this.valueMatrix.length; i++) {
|
||||
const row = this.valueMatrix[i];
|
||||
const name = row[nameIndex]?.value;
|
||||
if (typeof name !== 'string') {
|
||||
continue;
|
||||
}
|
||||
|
||||
for (const j in row) {
|
||||
for (let j = 0; j < row.length; j++) {
|
||||
const key = this.assignedTemplateFields[j];
|
||||
const tf = this.templateFieldsMap.get(key ?? '');
|
||||
if (!tf || !key) {
|
||||
@ -387,7 +387,7 @@ export class Importer {
|
||||
|
||||
pushToValueMatrixFromParsedRow(row: string[]) {
|
||||
const vmRow: ValueMatrix[number] = [];
|
||||
for (const i in row) {
|
||||
for (let i = 0; i < row.length; i++) {
|
||||
const rawValue = row[i];
|
||||
const index = Number(i);
|
||||
|
||||
@ -492,7 +492,7 @@ export class Importer {
|
||||
return false;
|
||||
}
|
||||
|
||||
for (const i in row) {
|
||||
for (let i = 0; i < row.length; i++) {
|
||||
const value = row[i];
|
||||
const tf = this.templateFieldsMap.get(value);
|
||||
let key: string | null = value;
|
||||
|
@ -6,7 +6,12 @@ import { App as VueApp, createApp } from 'vue';
|
||||
import App from './App.vue';
|
||||
import Badge from './components/Badge.vue';
|
||||
import FeatherIcon from './components/FeatherIcon.vue';
|
||||
import { getErrorHandled, handleError, sendError } from './errorHandling';
|
||||
import {
|
||||
getErrorHandled,
|
||||
getErrorHandledSync,
|
||||
handleError,
|
||||
sendError,
|
||||
} from './errorHandling';
|
||||
import { fyo } from './initFyo';
|
||||
import { outsideClickDirective } from './renderer/helpers';
|
||||
import registerIpcRendererListeners from './renderer/registerIpcRendererListeners';
|
||||
@ -14,6 +19,7 @@ import router from './router';
|
||||
import { stringifyCircular } from './utils';
|
||||
import { setLanguageMap } from './utils/language';
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
(async () => {
|
||||
const language = fyo.config.get('language') as string;
|
||||
if (language) {
|
||||
@ -21,8 +27,8 @@ import { setLanguageMap } from './utils/language';
|
||||
}
|
||||
fyo.store.language = language || 'English';
|
||||
|
||||
ipcRenderer.send = getErrorHandled(ipcRenderer.send);
|
||||
ipcRenderer.invoke = getErrorHandled(ipcRenderer.invoke);
|
||||
ipcRenderer.send = getErrorHandledSync(ipcRenderer.send.bind(ipcRenderer));
|
||||
ipcRenderer.invoke = getErrorHandled(ipcRenderer.invoke.bind(ipcRenderer));
|
||||
|
||||
registerIpcRendererListeners();
|
||||
const { isDevelopment, platform, version } = (await ipcRenderer.invoke(
|
||||
@ -69,6 +75,7 @@ import { setLanguageMap } from './utils/language';
|
||||
function setErrorHandlers(app: VueApp) {
|
||||
window.onerror = (message, source, lineno, colno, error) => {
|
||||
error = error ?? new Error('triggered in window.onerror');
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
handleError(true, error, { message, source, lineno, colno });
|
||||
};
|
||||
|
||||
@ -80,11 +87,13 @@ function setErrorHandlers(app: VueApp) {
|
||||
error = new Error(String(event.reason));
|
||||
}
|
||||
|
||||
// eslint-disable-next-line no-console
|
||||
handleError(true, error).catch((err) => console.error(err));
|
||||
};
|
||||
|
||||
window.addEventListener(CUSTOM_EVENTS.LOG_UNEXPECTED, (event) => {
|
||||
const details = (event as CustomEvent)?.detail as UnexpectedLogObject;
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
sendError(details);
|
||||
});
|
||||
|
||||
@ -100,7 +109,9 @@ function setErrorHandlers(app: VueApp) {
|
||||
more.props = stringifyCircular(vm.$props ?? {}, true, true);
|
||||
}
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
handleError(false, err as Error, more);
|
||||
// eslint-disable-next-line no-console
|
||||
console.error(err, vm, info);
|
||||
};
|
||||
}
|
||||
|
@ -6,7 +6,7 @@ import { IPC_CHANNELS } from 'utils/messages';
|
||||
export default function registerIpcRendererListeners() {
|
||||
ipcRenderer.on(
|
||||
IPC_CHANNELS.LOG_MAIN_PROCESS_ERROR,
|
||||
async (_, error, more) => {
|
||||
(_, error: unknown, more?: Record<string, unknown>) => {
|
||||
if (!(error instanceof Error)) {
|
||||
throw error;
|
||||
}
|
||||
@ -22,7 +22,8 @@ export default function registerIpcRendererListeners() {
|
||||
more.isMainProcess = true;
|
||||
more.notifyUser ??= true;
|
||||
|
||||
await handleError(true, error, more, more.notifyUser);
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
handleError(true, error, more, !!more.notifyUser);
|
||||
}
|
||||
);
|
||||
|
||||
@ -32,6 +33,7 @@ export default function registerIpcRendererListeners() {
|
||||
}
|
||||
|
||||
if (fyo.store.isDevelopment) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log(...stuff);
|
||||
}
|
||||
});
|
||||
@ -39,6 +41,7 @@ export default function registerIpcRendererListeners() {
|
||||
document.addEventListener('visibilitychange', () => {
|
||||
const { visibilityState } = document;
|
||||
if (visibilityState === 'visible' && !fyo.telemetry.started) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
||||
fyo.telemetry.start();
|
||||
}
|
||||
|
||||
|
@ -38,7 +38,7 @@ export class CreateCOA {
|
||||
const child = children[rootName];
|
||||
|
||||
if (rootAccount) {
|
||||
rootType = (child as COARootAccount).rootType as AccountRootType;
|
||||
rootType = (child as COARootAccount).rootType;
|
||||
}
|
||||
|
||||
const accountType = (child as COAChildAccount).accountType ?? '';
|
||||
@ -83,7 +83,7 @@ function identifyIsGroup(child: COARootAccount | COAChildAccount): boolean {
|
||||
return false;
|
||||
}
|
||||
|
||||
async function getCOA(chartOfAccounts: string) {
|
||||
async function getCOA(chartOfAccounts: string): Promise<COATree> {
|
||||
const coaList = getCOAList();
|
||||
const coa = coaList.find(({ name }) => name === chartOfAccounts);
|
||||
|
||||
@ -93,8 +93,9 @@ async function getCOA(chartOfAccounts: string) {
|
||||
}
|
||||
|
||||
try {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
||||
const countryCoa = (await import(`../../fixtures/verified/${conCode}.json`))
|
||||
.default;
|
||||
.default as { tree: COATree };
|
||||
return countryCoa.tree;
|
||||
} catch (e) {
|
||||
return getStandardCOA();
|
||||
|
@ -133,7 +133,7 @@ async function updateSystemSettings(
|
||||
const systemSettings = await fyo.doc.getDoc('SystemSettings');
|
||||
const instanceId = getRandomString();
|
||||
|
||||
systemSettings.setAndSync({
|
||||
await systemSettings.setAndSync({
|
||||
locale,
|
||||
currency,
|
||||
instanceId,
|
||||
@ -168,10 +168,8 @@ async function createCurrencyRecords(fyo: Fyo) {
|
||||
};
|
||||
|
||||
const doc = checkAndCreateDoc('Currency', docObject, fyo);
|
||||
if (doc) {
|
||||
promises.push(doc);
|
||||
queue.push(currency);
|
||||
}
|
||||
promises.push(doc);
|
||||
queue.push(currency);
|
||||
}
|
||||
return Promise.all(promises);
|
||||
}
|
||||
@ -256,7 +254,7 @@ async function checkAndCreateDoc(
|
||||
schemaName: string,
|
||||
docObject: DocValueMap,
|
||||
fyo: Fyo
|
||||
) {
|
||||
): Promise<Doc | undefined> {
|
||||
const canCreate = await checkIfExactRecordAbsent(schemaName, docObject, fyo);
|
||||
if (!canCreate) {
|
||||
return;
|
||||
@ -373,12 +371,12 @@ async function updateInventorySettings(fyo: Fyo) {
|
||||
}
|
||||
|
||||
const settingName = accountTypeDefaultMap[accountType]!;
|
||||
inventorySettings.set(settingName, accounts[0].name);
|
||||
await inventorySettings.set(settingName, accounts[0].name);
|
||||
}
|
||||
|
||||
const location = fyo.t`Stores`;
|
||||
if (await fyo.db.exists(ModelNameEnum.Location, location)) {
|
||||
inventorySettings.set('defaultLocation', location);
|
||||
await inventorySettings.set('defaultLocation', location);
|
||||
}
|
||||
|
||||
await inventorySettings.sync();
|
||||
|
11
src/shims-tsx.d.ts
vendored
11
src/shims-tsx.d.ts
vendored
@ -1,13 +1,12 @@
|
||||
import Vue, { VNode } from 'vue'
|
||||
import Vue, { VNode } from 'vue';
|
||||
|
||||
declare global {
|
||||
namespace JSX {
|
||||
// tslint:disable no-empty-interface
|
||||
type Element = VNode
|
||||
// tslint:disable no-empty-interface
|
||||
type ElementClass = Vue
|
||||
type Element = VNode;
|
||||
type ElementClass = Vue;
|
||||
interface IntrinsicElements {
|
||||
[elem: string]: any
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
[elem: string]: any;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -70,7 +70,7 @@ async function handleDirectoryDoesNotExist(dbPath: string) {
|
||||
|
||||
async function showDbErrorDialog(detail: string) {
|
||||
const { showDialog } = await import('src/utils/interactive');
|
||||
return await showDialog({
|
||||
return showDialog({
|
||||
type: 'error',
|
||||
title: t`Cannot Open File`,
|
||||
detail,
|
||||
|
@ -14,9 +14,7 @@ export function getGetStartedConfig() {
|
||||
icon: 'general',
|
||||
description: t`Set up your company information, email, country and fiscal year`,
|
||||
fieldname: 'companySetup',
|
||||
action() {
|
||||
openSettings(ModelNameEnum.AccountingSettings);
|
||||
},
|
||||
action: () => openSettings(ModelNameEnum.AccountingSettings),
|
||||
},
|
||||
{
|
||||
key: 'Print',
|
||||
@ -24,9 +22,7 @@ export function getGetStartedConfig() {
|
||||
icon: 'invoice',
|
||||
description: t`Customize your invoices by adding a logo and address details`,
|
||||
fieldname: 'printSetup',
|
||||
action() {
|
||||
openSettings(ModelNameEnum.PrintSettings);
|
||||
},
|
||||
action: () => openSettings(ModelNameEnum.PrintSettings),
|
||||
},
|
||||
{
|
||||
key: 'System',
|
||||
@ -34,9 +30,7 @@ export function getGetStartedConfig() {
|
||||
icon: 'system',
|
||||
description: t`Setup system defaults like date format and display precision`,
|
||||
fieldname: 'systemSetup',
|
||||
action() {
|
||||
openSettings(ModelNameEnum.SystemSettings);
|
||||
},
|
||||
action: () => openSettings(ModelNameEnum.SystemSettings),
|
||||
},
|
||||
],
|
||||
},
|
||||
@ -49,9 +43,7 @@ export function getGetStartedConfig() {
|
||||
label: t`Review Accounts`,
|
||||
icon: 'review-ac',
|
||||
description: t`Review your chart of accounts, add any account or tax heads as needed`,
|
||||
action: () => {
|
||||
routeTo('/chart-of-accounts');
|
||||
},
|
||||
action: () => routeTo('/chart-of-accounts'),
|
||||
fieldname: 'chartOfAccountsReviewed',
|
||||
documentation:
|
||||
'https://docs.frappebooks.com/setting-up/initial-entries.html#add-additional-bank-accounts',
|
||||
|
@ -16,11 +16,11 @@ export function stringifyCircular(
|
||||
obj: unknown,
|
||||
ignoreCircular = false,
|
||||
convertDocument = false
|
||||
) {
|
||||
): string {
|
||||
const cacheKey: string[] = [];
|
||||
const cacheValue: unknown[] = [];
|
||||
|
||||
return JSON.stringify(obj, (key, value) => {
|
||||
return JSON.stringify(obj, (key: string, value: unknown) => {
|
||||
if (typeof value !== 'object' || value === null) {
|
||||
cacheKey.push(key);
|
||||
cacheValue.push(value);
|
||||
|
@ -5,19 +5,13 @@ import { App, createApp, h } from 'vue';
|
||||
import { getColorClass } from './colors';
|
||||
import { DialogButton, DialogOptions, ToastOptions, ToastType } from './types';
|
||||
|
||||
type DialogReturn<DO extends DialogOptions> = DO['buttons'] extends {
|
||||
action: () => Promise<infer O> | infer O;
|
||||
}[]
|
||||
? O
|
||||
: void;
|
||||
|
||||
export async function showDialog<DO extends DialogOptions>(options: DO) {
|
||||
const preWrappedButtons: DialogButton[] = options.buttons ?? [
|
||||
{ label: t`Okay`, action: () => null, isEscape: true },
|
||||
];
|
||||
|
||||
return new Promise(async (resolve, reject) => {
|
||||
const buttons = preWrappedButtons!.map((config) => {
|
||||
const resultPromise = new Promise((resolve, reject) => {
|
||||
const buttons = preWrappedButtons.map((config) => {
|
||||
return {
|
||||
...config,
|
||||
action: async () => {
|
||||
@ -37,10 +31,12 @@ export async function showDialog<DO extends DialogOptions>(options: DO) {
|
||||
});
|
||||
|
||||
fragmentMountComponent(dialogApp);
|
||||
}) as DialogReturn<DO>;
|
||||
});
|
||||
|
||||
return await resultPromise;
|
||||
}
|
||||
|
||||
export async function showToast(options: ToastOptions) {
|
||||
export function showToast(options: ToastOptions) {
|
||||
const toastApp = createApp({
|
||||
render() {
|
||||
return h(Toast, { ...options });
|
||||
|
@ -4,9 +4,14 @@
|
||||
const { ipcRenderer } = require('electron');
|
||||
import { t } from 'fyo';
|
||||
import { BaseError } from 'fyo/utils/errors';
|
||||
import { BackendResponse } from 'utils/ipc/types';
|
||||
import type { BackendResponse } from 'utils/ipc/types';
|
||||
import { IPC_ACTIONS, IPC_MESSAGES } from 'utils/messages';
|
||||
import { SelectFileOptions, SelectFileReturn, TemplateFile } from 'utils/types';
|
||||
import type {
|
||||
LanguageMap,
|
||||
SelectFileOptions,
|
||||
SelectFileReturn,
|
||||
TemplateFile,
|
||||
} from 'utils/types';
|
||||
import { showDialog, showToast } from './interactive';
|
||||
import { setLanguageMap } from './language';
|
||||
import type { OpenDialogReturnValue } from 'electron';
|
||||
@ -15,22 +20,35 @@ export function reloadWindow() {
|
||||
return ipcRenderer.send(IPC_MESSAGES.RELOAD_MAIN_WINDOW);
|
||||
}
|
||||
|
||||
export async function getLanguageMap(code: string) {
|
||||
return (await ipcRenderer.invoke(IPC_ACTIONS.GET_LANGUAGE_MAP, code)) as {
|
||||
languageMap: LanguageMap;
|
||||
success: boolean;
|
||||
message: string;
|
||||
};
|
||||
}
|
||||
|
||||
export async function getSelectedFilePath(): Promise<OpenDialogReturnValue> {
|
||||
return await ipcRenderer.invoke(IPC_ACTIONS.GET_OPEN_FILEPATH, {
|
||||
title: this.t`Select file`,
|
||||
return (await ipcRenderer.invoke(IPC_ACTIONS.GET_OPEN_FILEPATH, {
|
||||
title: t`Select file`,
|
||||
properties: ['openFile'],
|
||||
filters: [{ name: 'SQLite DB File', extensions: ['db'] }],
|
||||
});
|
||||
})) as OpenDialogReturnValue;
|
||||
}
|
||||
|
||||
export async function getTemplates(): Promise<TemplateFile[]> {
|
||||
return await ipcRenderer.invoke(IPC_ACTIONS.GET_TEMPLATES);
|
||||
return (await ipcRenderer.invoke(
|
||||
IPC_ACTIONS.GET_TEMPLATES
|
||||
)) as TemplateFile[];
|
||||
}
|
||||
|
||||
export async function selectFile(
|
||||
options: SelectFileOptions
|
||||
): Promise<SelectFileReturn> {
|
||||
return await ipcRenderer.invoke(IPC_ACTIONS.SELECT_FILE, options);
|
||||
return (await ipcRenderer.invoke(
|
||||
IPC_ACTIONS.SELECT_FILE,
|
||||
options
|
||||
)) as SelectFileReturn;
|
||||
}
|
||||
|
||||
export async function checkForUpdates() {
|
||||
@ -38,7 +56,7 @@ export async function checkForUpdates() {
|
||||
await setLanguageMap();
|
||||
}
|
||||
|
||||
export async function openLink(link: string) {
|
||||
export function openLink(link: string) {
|
||||
ipcRenderer.send(IPC_MESSAGES.OPEN_EXTERNAL, link);
|
||||
}
|
||||
|
||||
@ -49,19 +67,19 @@ export async function deleteDb(filePath: string) {
|
||||
)) as BackendResponse;
|
||||
|
||||
if (error?.code === 'EBUSY') {
|
||||
showDialog({
|
||||
await showDialog({
|
||||
title: t`Delete Failed`,
|
||||
detail: t`Please restart and try again.`,
|
||||
type: 'error',
|
||||
});
|
||||
} else if (error?.code === 'ENOENT') {
|
||||
showDialog({
|
||||
await showDialog({
|
||||
title: t`Delete Failed`,
|
||||
detail: t`File ${filePath} does not exist.`,
|
||||
type: 'error',
|
||||
});
|
||||
} else if (error?.code === 'EPERM') {
|
||||
showDialog({
|
||||
await showDialog({
|
||||
title: t`Cannot Delete`,
|
||||
detail: t`Close Frappe Books and try manually.`,
|
||||
type: 'error',
|
||||
@ -87,14 +105,14 @@ export async function makePDF(
|
||||
savePath: string,
|
||||
width: number,
|
||||
height: number
|
||||
) {
|
||||
const success = await ipcRenderer.invoke(
|
||||
): Promise<void> {
|
||||
const success = (await ipcRenderer.invoke(
|
||||
IPC_ACTIONS.SAVE_HTML_AS_PDF,
|
||||
html,
|
||||
savePath,
|
||||
width,
|
||||
height
|
||||
);
|
||||
)) as boolean;
|
||||
|
||||
if (success) {
|
||||
showExportInFolder(t`Save as PDF Successful`, savePath);
|
||||
@ -108,8 +126,8 @@ export function showExportInFolder(message: string, filePath: string) {
|
||||
message,
|
||||
actionText: t`Open Folder`,
|
||||
type: 'success',
|
||||
action: async () => {
|
||||
await showItemInFolder(filePath);
|
||||
action: () => {
|
||||
showItemInFolder(filePath);
|
||||
},
|
||||
});
|
||||
}
|
||||
|
@ -1,9 +1,7 @@
|
||||
const { ipcRenderer } = require('electron');
|
||||
import { DEFAULT_LANGUAGE } from 'fyo/utils/consts';
|
||||
import { setLanguageMapOnTranslationString } from 'fyo/utils/translation';
|
||||
import { fyo } from 'src/initFyo';
|
||||
import { IPC_ACTIONS } from 'utils/messages';
|
||||
import { reloadWindow } from './ipcCalls';
|
||||
import { getLanguageMap, reloadWindow } from './ipcCalls';
|
||||
import { systemLanguageRef } from './refs';
|
||||
|
||||
// Language: Language Code in books/translations
|
||||
@ -65,17 +63,14 @@ function getLanguageCode(initLanguage: string, oldLanguage: string) {
|
||||
}
|
||||
|
||||
async function fetchAndSetLanguageMap(code: string) {
|
||||
const { success, message, languageMap } = await ipcRenderer.invoke(
|
||||
IPC_ACTIONS.GET_LANGUAGE_MAP,
|
||||
code
|
||||
);
|
||||
const { success, message, languageMap } = await getLanguageMap(code);
|
||||
|
||||
if (!success) {
|
||||
const { showToast } = await import('src/utils/interactive');
|
||||
showToast({ type: 'error', message });
|
||||
} else {
|
||||
setLanguageMapOnTranslationString(languageMap);
|
||||
fyo.db.translateSchemaMap(languageMap);
|
||||
await fyo.db.translateSchemaMap(languageMap);
|
||||
}
|
||||
|
||||
return success;
|
||||
|
@ -24,7 +24,7 @@ interface SearchItem {
|
||||
label: string;
|
||||
group: Exclude<SearchGroup, 'Docs'>;
|
||||
route?: string;
|
||||
action?: () => void;
|
||||
action?: () => void | Promise<void>;
|
||||
}
|
||||
|
||||
interface DocSearchItem extends Omit<SearchItem, 'group'> {
|
||||
@ -319,8 +319,8 @@ function getNonDocSearchList(fyo: Fyo) {
|
||||
.flat()
|
||||
.map((d) => {
|
||||
if (d.route && !d.action) {
|
||||
d.action = () => {
|
||||
routeTo(d.route!);
|
||||
d.action = async () => {
|
||||
await routeTo(d.route!);
|
||||
};
|
||||
}
|
||||
return d;
|
||||
|
@ -114,7 +114,7 @@ export interface DialogOptions {
|
||||
|
||||
export type DialogButton = {
|
||||
label: string;
|
||||
action: () => any;
|
||||
action: () => unknown;
|
||||
isPrimary?: boolean;
|
||||
isEscape?: boolean;
|
||||
};
|
||||
|
@ -7,6 +7,7 @@ import type { Doc } from 'fyo/model/doc';
|
||||
import { Action } from 'fyo/model/types';
|
||||
import { getActions } from 'fyo/utils';
|
||||
import { getDbError, LinkValidationError, ValueError } from 'fyo/utils/errors';
|
||||
import { Invoice } from 'models/baseModels/Invoice/Invoice';
|
||||
import { PurchaseInvoice } from 'models/baseModels/PurchaseInvoice/PurchaseInvoice';
|
||||
import { SalesInvoice } from 'models/baseModels/SalesInvoice/SalesInvoice';
|
||||
import { getLedgerLink } from 'models/helpers';
|
||||
@ -25,13 +26,12 @@ import { selectFile } from './ipcCalls';
|
||||
import { showSidebar } from './refs';
|
||||
import {
|
||||
ActionGroup,
|
||||
DialogButton,
|
||||
QuickEditOptions,
|
||||
SettingsTab,
|
||||
ToastOptions,
|
||||
UIGroupedFields,
|
||||
} from './types';
|
||||
import { Invoice } from 'models/baseModels/Invoice/Invoice';
|
||||
import { assertIsType } from 'utils/index';
|
||||
|
||||
export const toastDurationMap = { short: 2_500, long: 5_000 } as const;
|
||||
|
||||
@ -56,7 +56,7 @@ export async function openQuickEdit({
|
||||
showFields,
|
||||
hideFields,
|
||||
};
|
||||
router.push({ query });
|
||||
await router.push({ query });
|
||||
}
|
||||
|
||||
export async function openSettings(tab: SettingsTab) {
|
||||
@ -81,7 +81,7 @@ export async function deleteDocWithPrompt(doc: Doc) {
|
||||
detail = t`This action is permanent and will delete associated ledger entries.`;
|
||||
}
|
||||
|
||||
return await showDialog({
|
||||
return (await showDialog({
|
||||
title: t`Delete ${getDocReferenceLabel(doc)}?`,
|
||||
detail,
|
||||
type: 'warning',
|
||||
@ -93,13 +93,13 @@ export async function deleteDocWithPrompt(doc: Doc) {
|
||||
await doc.delete();
|
||||
} catch (err) {
|
||||
if (getDbError(err as Error) === LinkValidationError) {
|
||||
showDialog({
|
||||
await showDialog({
|
||||
title: t`Delete Failed`,
|
||||
detail: t`Cannot delete ${schemaLabel} "${doc.name!}" because of linked entries.`,
|
||||
type: 'error',
|
||||
});
|
||||
} else {
|
||||
handleErrorWithDialog(err as Error, doc);
|
||||
await handleErrorWithDialog(err as Error, doc);
|
||||
}
|
||||
|
||||
return false;
|
||||
@ -117,7 +117,7 @@ export async function deleteDocWithPrompt(doc: Doc) {
|
||||
isEscape: true,
|
||||
},
|
||||
],
|
||||
});
|
||||
})) as boolean;
|
||||
}
|
||||
|
||||
export async function cancelDocWithPrompt(doc: Doc) {
|
||||
@ -152,7 +152,7 @@ export async function cancelDocWithPrompt(doc: Doc) {
|
||||
}
|
||||
}
|
||||
|
||||
return await showDialog({
|
||||
return (await showDialog({
|
||||
title: t`Cancel ${getDocReferenceLabel(doc)}?`,
|
||||
detail,
|
||||
type: 'warning',
|
||||
@ -163,7 +163,7 @@ export async function cancelDocWithPrompt(doc: Doc) {
|
||||
try {
|
||||
await doc.cancel();
|
||||
} catch (err) {
|
||||
handleErrorWithDialog(err as Error, doc);
|
||||
await handleErrorWithDialog(err as Error, doc);
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -179,7 +179,7 @@ export async function cancelDocWithPrompt(doc: Doc) {
|
||||
isEscape: true,
|
||||
},
|
||||
],
|
||||
});
|
||||
})) as boolean;
|
||||
}
|
||||
|
||||
export function getActionsForDoc(doc?: Doc): Action[] {
|
||||
@ -280,7 +280,7 @@ function getDuplicateAction(doc: Doc): Action {
|
||||
const dupe = doc.duplicate();
|
||||
await openEdit(dupe);
|
||||
} catch (err) {
|
||||
handleErrorWithDialog(err as Error, doc);
|
||||
await handleErrorWithDialog(err as Error, doc);
|
||||
}
|
||||
},
|
||||
};
|
||||
@ -385,8 +385,8 @@ export function toggleSidebar(value?: boolean) {
|
||||
|
||||
export function focusOrSelectFormControl(
|
||||
doc: Doc,
|
||||
ref: any,
|
||||
clear= true
|
||||
ref: unknown,
|
||||
shouldClear = true
|
||||
) {
|
||||
if (!doc?.fyo) {
|
||||
return;
|
||||
@ -405,7 +405,15 @@ export function focusOrSelectFormControl(
|
||||
ref = ref[0];
|
||||
}
|
||||
|
||||
if (!clear && typeof ref?.select === 'function') {
|
||||
if (
|
||||
!ref ||
|
||||
typeof ref !== 'object' ||
|
||||
!assertIsType<Record<string, () => void>>(ref)
|
||||
) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!shouldClear && typeof ref?.select === 'function') {
|
||||
ref.select();
|
||||
return;
|
||||
}
|
||||
@ -429,7 +437,7 @@ export async function selectTextFile(filters?: SelectFileOptions['filters']) {
|
||||
const { success, canceled, filePath, data, name } = await selectFile(options);
|
||||
|
||||
if (canceled || !success) {
|
||||
await showToast({
|
||||
showToast({
|
||||
type: 'error',
|
||||
message: t`File selection failed`,
|
||||
});
|
||||
@ -438,7 +446,7 @@ export async function selectTextFile(filters?: SelectFileOptions['filters']) {
|
||||
|
||||
const text = new TextDecoder().decode(data);
|
||||
if (!text) {
|
||||
await showToast({
|
||||
showToast({
|
||||
type: 'error',
|
||||
message: t`Empty file selected`,
|
||||
});
|
||||
@ -528,7 +536,7 @@ async function syncWithoutDialog(doc: Doc): Promise<boolean> {
|
||||
try {
|
||||
await doc.sync();
|
||||
} catch (error) {
|
||||
handleErrorWithDialog(error, doc);
|
||||
await handleErrorWithDialog(error, doc);
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -563,14 +571,14 @@ async function showSubmitOrSyncDialog(doc: Doc, type: 'submit' | 'sync') {
|
||||
try {
|
||||
await doc[type]();
|
||||
} catch (error) {
|
||||
handleErrorWithDialog(error, doc);
|
||||
await handleErrorWithDialog(error, doc);
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
const buttons: DialogButton[] = [
|
||||
const buttons = [
|
||||
{
|
||||
label: t`Yes`,
|
||||
action: yesAction,
|
||||
@ -583,11 +591,13 @@ async function showSubmitOrSyncDialog(doc: Doc, type: 'submit' | 'sync') {
|
||||
},
|
||||
];
|
||||
|
||||
return await showDialog({
|
||||
const dialogOptions = {
|
||||
title,
|
||||
detail,
|
||||
buttons,
|
||||
});
|
||||
};
|
||||
|
||||
return (await showDialog(dialogOptions)) as boolean;
|
||||
}
|
||||
|
||||
function getDocSyncMessage(doc: Doc): string {
|
||||
|
@ -8,7 +8,7 @@ export function getValueMapFromList<T, K extends keyof T, V extends keyof T>(
|
||||
list: T[],
|
||||
key: K,
|
||||
valueKey: V,
|
||||
filterUndefined: boolean = true
|
||||
filterUndefined = true
|
||||
): Record<string, T[V]> {
|
||||
if (filterUndefined) {
|
||||
list = list.filter(
|
||||
@ -32,7 +32,7 @@ export function getRandomString(): string {
|
||||
return `${randomNumber}-${currentTime}`;
|
||||
}
|
||||
|
||||
export async function sleep(durationMilliseconds: number = 1000) {
|
||||
export async function sleep(durationMilliseconds = 1000) {
|
||||
return new Promise((r) => setTimeout(() => r(null), durationMilliseconds));
|
||||
}
|
||||
|
||||
@ -267,3 +267,9 @@ export function objectForEach<T extends object | unknown>(
|
||||
|
||||
return func(obj);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Asserts that `value` is of type T. Use with care.
|
||||
*/
|
||||
export const assertIsType = <T>(value: unknown): value is T => true;
|
||||
|
@ -3,7 +3,6 @@ import type { ConfigFile } from 'fyo/core/types';
|
||||
export type UnknownMap = Record<string, unknown>;
|
||||
export type Translation = { translation: string; context?: string };
|
||||
export type LanguageMap = Record<string, Translation>;
|
||||
export type UnknownFunction = (...args: unknown[]) => unknown;
|
||||
|
||||
export type CountryInfoMap = Record<string, CountryInfo | undefined>;
|
||||
export interface CountryInfo {
|
||||
|
Loading…
x
Reference in New Issue
Block a user