2020-10-09 18:35:46 +01:00
|
|
|
import InteropService from 'lib/services/interop/InteropService';
|
|
|
|
import CommandService from 'lib/services/CommandService';
|
|
|
|
import shim from 'lib/shim';
|
|
|
|
import { ExportOptions, FileSystemItem, Module } from 'lib/services/interop/types';
|
|
|
|
|
|
|
|
import { _ } from 'lib/locale';
|
|
|
|
const bridge = require('electron').remote.require('./bridge').default;
|
|
|
|
const Setting = require('lib/models/Setting').default;
|
2020-04-02 17:24:33 -06:00
|
|
|
const Note = require('lib/models/Note.js');
|
|
|
|
const { friendlySafeFilename } = require('lib/path-utils');
|
2020-06-17 11:35:13 -06:00
|
|
|
const { time } = require('lib/time-utils.js');
|
2019-12-17 09:44:48 +00:00
|
|
|
const md5 = require('md5');
|
|
|
|
const url = require('url');
|
2018-03-01 20:14:06 +00:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
interface ExportNoteOptions {
|
|
|
|
customCss?: string,
|
|
|
|
sourceNoteIds?: string[],
|
|
|
|
sourceFolderIds?: string[],
|
|
|
|
printBackground?: boolean,
|
|
|
|
pageSize?: string,
|
|
|
|
landscape?: boolean,
|
|
|
|
}
|
|
|
|
|
|
|
|
export default class InteropServiceHelper {
|
2018-03-01 20:14:06 +00:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
private static async exportNoteToHtmlFile(noteId:string, exportOptions:ExportNoteOptions) {
|
2019-12-17 09:44:48 +00:00
|
|
|
const tempFile = `${Setting.value('tempDir')}/${md5(Date.now() + Math.random())}.html`;
|
2020-01-24 21:46:48 +00:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
const fullExportOptions:ExportOptions = Object.assign({}, {
|
2020-01-24 21:46:48 +00:00
|
|
|
path: tempFile,
|
|
|
|
format: 'html',
|
2020-10-09 18:35:46 +01:00
|
|
|
target: FileSystemItem.File,
|
2020-01-24 21:46:48 +00:00
|
|
|
sourceNoteIds: [noteId],
|
|
|
|
customCss: '',
|
|
|
|
}, exportOptions);
|
2019-12-17 09:44:48 +00:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
const service = InteropService.instance();
|
2019-12-17 09:44:48 +00:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
const result = await service.export(fullExportOptions);
|
2019-12-17 09:44:48 +00:00
|
|
|
console.info('Export HTML result: ', result);
|
|
|
|
return tempFile;
|
|
|
|
}
|
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
private static async exportNoteTo_(target:string, noteId:string, options:ExportNoteOptions = {}) {
|
|
|
|
let win:any = null;
|
|
|
|
let htmlFile:string = null;
|
2019-12-17 09:44:48 +00:00
|
|
|
|
|
|
|
const cleanup = () => {
|
|
|
|
if (win) win.destroy();
|
|
|
|
if (htmlFile) shim.fsDriver().remove(htmlFile);
|
|
|
|
};
|
|
|
|
|
|
|
|
try {
|
2020-01-24 21:46:48 +00:00
|
|
|
const exportOptions = {
|
|
|
|
customCss: options.customCss ? options.customCss : '',
|
|
|
|
};
|
|
|
|
|
|
|
|
htmlFile = await this.exportNoteToHtmlFile(noteId, exportOptions);
|
2019-12-17 09:44:48 +00:00
|
|
|
|
|
|
|
const windowOptions = {
|
2019-12-30 20:44:15 +01:00
|
|
|
show: false,
|
2019-12-17 09:44:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
win = bridge().newBrowserWindow(windowOptions);
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
2020-04-14 23:58:23 +01:00
|
|
|
win.webContents.on('did-finish-load', () => {
|
|
|
|
|
|
|
|
// did-finish-load will trigger when most assets are done loading, probably
|
|
|
|
// images, JavaScript and CSS. However it seems it might trigger *before*
|
|
|
|
// all fonts are loaded, which will break for example Katex rendering.
|
|
|
|
// So we need to add an additional timer to make sure fonts are loaded
|
|
|
|
// as it doesn't seem there's any easy way to figure that out.
|
2020-10-09 18:35:46 +01:00
|
|
|
shim.setTimeout(async () => {
|
2020-04-14 23:58:23 +01:00
|
|
|
if (target === 'pdf') {
|
|
|
|
try {
|
|
|
|
const data = await win.webContents.printToPDF(options);
|
|
|
|
resolve(data);
|
|
|
|
} catch (error) {
|
|
|
|
reject(error);
|
|
|
|
} finally {
|
|
|
|
cleanup();
|
|
|
|
}
|
|
|
|
} else {
|
2020-05-03 18:44:49 +01:00
|
|
|
// TODO: it is crashing at this point :(
|
|
|
|
// Appears to be a Chromium bug: https://github.com/electron/electron/issues/19946
|
|
|
|
// Maybe can be fixed by doing everything from main process?
|
|
|
|
// i.e. creating a function `print()` that takes the `htmlFile` variable as input.
|
2020-05-02 16:41:07 +01:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
win.webContents.print(options, (success:boolean, reason:string) => {
|
2020-04-14 23:58:23 +01:00
|
|
|
// TODO: This is correct but broken in Electron 4. Need to upgrade to 5+
|
|
|
|
// It calls the callback right away with "false" even if the document hasn't be print yet.
|
|
|
|
|
|
|
|
cleanup();
|
|
|
|
if (!success && reason !== 'cancelled') reject(new Error(`Could not print: ${reason}`));
|
|
|
|
resolve();
|
|
|
|
});
|
2019-12-17 11:08:55 +00:00
|
|
|
}
|
2020-04-14 23:58:23 +01:00
|
|
|
}, 2000);
|
|
|
|
|
2019-12-17 09:44:48 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
win.loadURL(url.format({
|
|
|
|
pathname: htmlFile,
|
|
|
|
protocol: 'file:',
|
|
|
|
slashes: true,
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
cleanup();
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
public static async exportNoteToPdf(noteId:string, options:ExportNoteOptions = {}) {
|
2019-12-17 09:44:48 +00:00
|
|
|
return this.exportNoteTo_('pdf', noteId, options);
|
|
|
|
}
|
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
public static async printNote(noteId:string, options:ExportNoteOptions = {}) {
|
2019-12-17 09:44:48 +00:00
|
|
|
return this.exportNoteTo_('printer', noteId, options);
|
|
|
|
}
|
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
public static async defaultFilename(noteId:string, fileExtension:string) {
|
2020-06-17 11:35:13 -06:00
|
|
|
// Default filename is just the date
|
|
|
|
const date = time.formatMsToLocal(new Date().getTime(), time.dateFormat());
|
|
|
|
let filename = friendlySafeFilename(`${date}`, 100);
|
|
|
|
|
|
|
|
if (noteId) {
|
|
|
|
const note = await Note.load(noteId);
|
|
|
|
// In a rare case the passed note will be null, use the id for filename
|
|
|
|
filename = friendlySafeFilename(note ? note.title : noteId, 100);
|
|
|
|
}
|
|
|
|
|
2020-05-03 18:44:49 +01:00
|
|
|
return `${filename}.${fileExtension}`;
|
2020-04-02 17:24:33 -06:00
|
|
|
}
|
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
public static async export(_dispatch:Function, module:Module, options:ExportNoteOptions = null) {
|
2018-03-01 20:14:06 +00:00
|
|
|
if (!options) options = {};
|
|
|
|
|
|
|
|
let path = null;
|
|
|
|
|
|
|
|
if (module.target === 'file') {
|
2020-05-03 18:44:49 +01:00
|
|
|
const noteId = options.sourceNoteIds && options.sourceNoteIds.length ? options.sourceNoteIds[0] : null;
|
2018-03-01 20:14:06 +00:00
|
|
|
path = bridge().showSaveDialog({
|
2020-02-04 22:09:34 +00:00
|
|
|
filters: [{ name: module.description, extensions: module.fileExtensions }],
|
2020-05-03 18:44:49 +01:00
|
|
|
defaultPath: await this.defaultFilename(noteId, module.fileExtensions[0]),
|
2018-03-01 20:14:06 +00:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
path = bridge().showOpenDialog({
|
|
|
|
properties: ['openDirectory', 'createDirectory'],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!path || (Array.isArray(path) && !path.length)) return;
|
|
|
|
|
|
|
|
if (Array.isArray(path)) path = path[0];
|
|
|
|
|
2020-07-03 22:32:39 +01:00
|
|
|
CommandService.instance().execute('showModalMessage', { message: _('Exporting to "%s" as "%s" format. Please wait...', path, module.format) });
|
2018-03-01 20:14:06 +00:00
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
const exportOptions:ExportOptions = {};
|
2018-03-01 20:14:06 +00:00
|
|
|
exportOptions.path = path;
|
|
|
|
exportOptions.format = module.format;
|
2019-12-15 18:41:13 +00:00
|
|
|
exportOptions.modulePath = module.path;
|
|
|
|
exportOptions.target = module.target;
|
2018-03-01 20:14:06 +00:00
|
|
|
if (options.sourceFolderIds) exportOptions.sourceFolderIds = options.sourceFolderIds;
|
|
|
|
if (options.sourceNoteIds) exportOptions.sourceNoteIds = options.sourceNoteIds;
|
|
|
|
|
2020-10-09 18:35:46 +01:00
|
|
|
const service = InteropService.instance();
|
2018-03-01 20:14:06 +00:00
|
|
|
|
2019-10-11 20:20:12 +02:00
|
|
|
try {
|
|
|
|
const result = await service.export(exportOptions);
|
|
|
|
console.info('Export result: ', result);
|
|
|
|
} catch (error) {
|
2019-12-17 00:40:25 +00:00
|
|
|
console.error(error);
|
2019-10-11 20:20:12 +02:00
|
|
|
bridge().showErrorMessageBox(_('Could not export notes: %s', error.message));
|
|
|
|
}
|
2018-03-01 20:14:06 +00:00
|
|
|
|
2020-07-03 22:32:39 +01:00
|
|
|
CommandService.instance().execute('hideModalMessage');
|
2018-03-01 20:14:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|