mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-08 13:06:15 +02:00
c63c6370b5
The goal is to make the command system more modular, so each command can be defined as a single object that includes a declaration (name, label, etc.) and a runtime (to execute the command, test if it should be enabled, etc.) Utility methods are provided to convert a command to a menu item or a toolbar button, thus reducing duplicated and boiler plate code across the codebase (often the menu item logic was duplicated in the toolbar button logic and vice versa). The goal is to make it easier to add new commands (and associated menu item and toolbar buttons) and to call them from anywhere. This is also useful for plugins, which can also easily define new commands. Could also allow creating a command palette.
173 lines
5.3 KiB
JavaScript
173 lines
5.3 KiB
JavaScript
const { _ } = require('lib/locale');
|
|
const { bridge } = require('electron').remote.require('./bridge');
|
|
const InteropService = require('lib/services/InteropService');
|
|
const CommandService = require('lib/services/CommandService').default;
|
|
const Setting = require('lib/models/Setting');
|
|
const Note = require('lib/models/Note.js');
|
|
const { friendlySafeFilename } = require('lib/path-utils');
|
|
const { time } = require('lib/time-utils.js');
|
|
const md5 = require('md5');
|
|
const url = require('url');
|
|
const { shim } = require('lib/shim');
|
|
|
|
class InteropServiceHelper {
|
|
|
|
static async exportNoteToHtmlFile(noteId, exportOptions) {
|
|
const tempFile = `${Setting.value('tempDir')}/${md5(Date.now() + Math.random())}.html`;
|
|
|
|
exportOptions = Object.assign({}, {
|
|
path: tempFile,
|
|
format: 'html',
|
|
target: 'file',
|
|
sourceNoteIds: [noteId],
|
|
customCss: '',
|
|
}, exportOptions);
|
|
|
|
const service = new InteropService();
|
|
|
|
const result = await service.export(exportOptions);
|
|
console.info('Export HTML result: ', result);
|
|
return tempFile;
|
|
}
|
|
|
|
static async exportNoteTo_(target, noteId, options = {}) {
|
|
let win = null;
|
|
let htmlFile = null;
|
|
|
|
const cleanup = () => {
|
|
if (win) win.destroy();
|
|
if (htmlFile) shim.fsDriver().remove(htmlFile);
|
|
};
|
|
|
|
try {
|
|
const exportOptions = {
|
|
customCss: options.customCss ? options.customCss : '',
|
|
};
|
|
|
|
htmlFile = await this.exportNoteToHtmlFile(noteId, exportOptions);
|
|
|
|
const windowOptions = {
|
|
show: false,
|
|
};
|
|
|
|
win = bridge().newBrowserWindow(windowOptions);
|
|
|
|
return new Promise((resolve, reject) => {
|
|
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.
|
|
setTimeout(async () => {
|
|
if (target === 'pdf') {
|
|
try {
|
|
const data = await win.webContents.printToPDF(options);
|
|
resolve(data);
|
|
} catch (error) {
|
|
reject(error);
|
|
} finally {
|
|
cleanup();
|
|
}
|
|
} else {
|
|
// 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.
|
|
|
|
win.webContents.print(options, (success, reason) => {
|
|
// 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();
|
|
});
|
|
}
|
|
}, 2000);
|
|
|
|
});
|
|
|
|
win.loadURL(url.format({
|
|
pathname: htmlFile,
|
|
protocol: 'file:',
|
|
slashes: true,
|
|
}));
|
|
});
|
|
} catch (error) {
|
|
cleanup();
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
static async exportNoteToPdf(noteId, options = {}) {
|
|
return this.exportNoteTo_('pdf', noteId, options);
|
|
}
|
|
|
|
static async printNote(noteId, options = {}) {
|
|
return this.exportNoteTo_('printer', noteId, options);
|
|
}
|
|
|
|
static async defaultFilename(noteId, fileExtension) {
|
|
// 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);
|
|
}
|
|
|
|
return `${filename}.${fileExtension}`;
|
|
}
|
|
|
|
static async export(dispatch, module, options = null) {
|
|
if (!options) options = {};
|
|
|
|
let path = null;
|
|
|
|
if (module.target === 'file') {
|
|
const noteId = options.sourceNoteIds && options.sourceNoteIds.length ? options.sourceNoteIds[0] : null;
|
|
path = bridge().showSaveDialog({
|
|
filters: [{ name: module.description, extensions: module.fileExtensions }],
|
|
defaultPath: await this.defaultFilename(noteId, module.fileExtensions[0]),
|
|
});
|
|
} else {
|
|
path = bridge().showOpenDialog({
|
|
properties: ['openDirectory', 'createDirectory'],
|
|
});
|
|
}
|
|
|
|
if (!path || (Array.isArray(path) && !path.length)) return;
|
|
|
|
if (Array.isArray(path)) path = path[0];
|
|
|
|
CommandService.instance().execute('showModalMessage', { message: _('Exporting to "%s" as "%s" format. Please wait...', path, module.format) });
|
|
|
|
const exportOptions = {};
|
|
exportOptions.path = path;
|
|
exportOptions.format = module.format;
|
|
exportOptions.modulePath = module.path;
|
|
exportOptions.target = module.target;
|
|
if (options.sourceFolderIds) exportOptions.sourceFolderIds = options.sourceFolderIds;
|
|
if (options.sourceNoteIds) exportOptions.sourceNoteIds = options.sourceNoteIds;
|
|
|
|
const service = new InteropService();
|
|
|
|
try {
|
|
const result = await service.export(exportOptions);
|
|
console.info('Export result: ', result);
|
|
} catch (error) {
|
|
console.error(error);
|
|
bridge().showErrorMessageBox(_('Could not export notes: %s', error.message));
|
|
}
|
|
|
|
CommandService.instance().execute('hideModalMessage');
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = InteropServiceHelper;
|