2019-12-15 20:41:13 +02:00
|
|
|
const InteropService_Exporter_Base = require('lib/services/InteropService_Exporter_Base');
|
|
|
|
const { basename, friendlySafeFilename, rtrimSlashes } = require('lib/path-utils.js');
|
|
|
|
const BaseModel = require('lib/BaseModel');
|
|
|
|
const Folder = require('lib/models/Folder');
|
|
|
|
const Note = require('lib/models/Note');
|
|
|
|
const Setting = require('lib/models/Setting');
|
|
|
|
const Resource = require('lib/models/Resource');
|
|
|
|
const { shim } = require('lib/shim');
|
|
|
|
const dataurl = require('dataurl');
|
|
|
|
const { themeStyle } = require('../../theme.js');
|
|
|
|
const { dirname } = require('lib/path-utils.js');
|
2019-12-17 11:44:48 +02:00
|
|
|
const { escapeHtml } = require('lib/string-utils.js');
|
2019-12-29 19:58:40 +02:00
|
|
|
const markupLanguageUtils = require('lib/markupLanguageUtils');
|
2019-12-15 20:41:13 +02:00
|
|
|
|
|
|
|
class InteropService_Exporter_Html extends InteropService_Exporter_Base {
|
|
|
|
|
|
|
|
async init(path) {
|
|
|
|
if (this.metadata().target === 'file') {
|
|
|
|
this.destDir_ = dirname(path);
|
|
|
|
this.filePath_ = path;
|
|
|
|
} else {
|
|
|
|
this.destDir_ = path;
|
|
|
|
this.filePath_ = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.createdDirs_ = [];
|
|
|
|
this.resourceDir_ = this.destDir_ ? `${this.destDir_}/_resources` : null;
|
|
|
|
|
|
|
|
await shim.fsDriver().mkdir(this.destDir_);
|
2019-12-29 19:58:40 +02:00
|
|
|
this.markupToHtml_ = markupLanguageUtils.newMarkupToHtml();
|
2019-12-15 20:41:13 +02:00
|
|
|
this.resources_ = [];
|
|
|
|
this.style_ = themeStyle(Setting.THEME_LIGHT);
|
|
|
|
}
|
|
|
|
|
|
|
|
async makeDirPath_(item, pathPart = null) {
|
|
|
|
let output = '';
|
|
|
|
while (true) {
|
|
|
|
if (item.type_ === BaseModel.TYPE_FOLDER) {
|
|
|
|
if (pathPart) {
|
|
|
|
output = `${pathPart}/${output}`;
|
|
|
|
} else {
|
|
|
|
output = `${friendlySafeFilename(item.title, null, true)}/${output}`;
|
|
|
|
output = await shim.fsDriver().findUniqueFilename(output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!item.parent_id) return output;
|
|
|
|
item = await Folder.load(item.parent_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async processNoteResources_(item) {
|
|
|
|
const target = this.metadata().target;
|
|
|
|
const linkedResourceIds = await Note.linkedResourceIds(item.body);
|
|
|
|
const relativePath = target === 'directory' ? rtrimSlashes(await this.makeDirPath_(item, '..')) : '';
|
|
|
|
const resourcePaths = this.context() && this.context().resourcePaths ? this.context().resourcePaths : {};
|
|
|
|
|
|
|
|
let newBody = item.body;
|
|
|
|
|
|
|
|
for (let i = 0; i < linkedResourceIds.length; i++) {
|
|
|
|
const id = linkedResourceIds[i];
|
|
|
|
const resource = await Resource.load(id);
|
|
|
|
let resourceContent = '';
|
|
|
|
const isImage = Resource.isSupportedImageMimeType(resource.mime);
|
|
|
|
|
|
|
|
if (!isImage) {
|
|
|
|
resourceContent = `${relativePath ? `${relativePath}/` : ''}_resources/${basename(resourcePaths[id])}`;
|
|
|
|
} else {
|
|
|
|
const buffer = await shim.fsDriver().readFile(resourcePaths[id], 'Buffer');
|
|
|
|
|
|
|
|
resourceContent = dataurl.convert({
|
|
|
|
data: buffer,
|
|
|
|
mimetype: resource.mime,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
newBody = newBody.replace(new RegExp(`:/${id}`, 'g'), resourceContent);
|
|
|
|
}
|
|
|
|
|
|
|
|
return newBody;
|
|
|
|
}
|
|
|
|
|
|
|
|
async processItem(ItemClass, item) {
|
|
|
|
if ([BaseModel.TYPE_NOTE, BaseModel.TYPE_FOLDER].indexOf(item.type_) < 0) return;
|
|
|
|
|
|
|
|
let dirPath = '';
|
|
|
|
if (!this.filePath_) {
|
|
|
|
dirPath = `${this.destDir_}/${await this.makeDirPath_(item)}`;
|
|
|
|
|
|
|
|
if (this.createdDirs_.indexOf(dirPath) < 0) {
|
|
|
|
await shim.fsDriver().mkdir(dirPath);
|
|
|
|
this.createdDirs_.push(dirPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item.type_ === BaseModel.TYPE_NOTE) {
|
|
|
|
let noteFilePath = '';
|
|
|
|
|
|
|
|
if (this.filePath_) {
|
|
|
|
noteFilePath = this.filePath_;
|
|
|
|
} else {
|
|
|
|
noteFilePath = `${dirPath}/${friendlySafeFilename(item.title, null, true)}.html`;
|
|
|
|
noteFilePath = await shim.fsDriver().findUniqueFilename(noteFilePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
const bodyMd = await this.processNoteResources_(item);
|
2019-12-29 19:58:40 +02:00
|
|
|
const result = await this.markupToHtml_.render(item.markup_language, bodyMd, this.style_, { resources: this.resources_, plainResourceRendering: true });
|
2019-12-15 20:41:13 +02:00
|
|
|
const noteContent = [];
|
2019-12-17 11:44:48 +02:00
|
|
|
if (item.title) noteContent.push(`<div class="exported-note-title">${escapeHtml(item.title)}</div>`);
|
2019-12-15 20:41:13 +02:00
|
|
|
if (result.html) noteContent.push(result.html);
|
|
|
|
|
2019-12-30 21:44:15 +02:00
|
|
|
// We need to export all the plugin assets too and refer them from the header
|
|
|
|
// The source path is a bit hard-coded but shouldn't change.
|
|
|
|
// TODO: The joplin-renderer package should take care of creating the <link> and <script> tags to reduce duplicate code
|
|
|
|
// Calling app would then ensure that the CSS files, etc. are in the correct location.
|
|
|
|
const headers = [];
|
|
|
|
for (let i = 0; i < result.pluginAssets.length; i++) {
|
|
|
|
const asset = result.pluginAssets[i];
|
|
|
|
const filePath = `${dirname(dirname(__dirname))}/gui/note-viewer/pluginAssets/${asset.name}`;
|
|
|
|
const destPath = `${dirname(noteFilePath)}/pluginAssets/${asset.name}`;
|
|
|
|
await shim.fsDriver().mkdir(dirname(destPath));
|
|
|
|
await shim.fsDriver().copy(filePath, destPath);
|
|
|
|
|
|
|
|
if (asset.mime === 'text/css') {
|
|
|
|
headers.push(`<link rel="stylesheet" href="pluginAssets/${asset.name}">`);
|
|
|
|
} else if (asset.mime === 'application/javascript') {
|
|
|
|
// NOT TESTED!!
|
|
|
|
headers.push(`<script type="application/javascript" src="pluginAssets/${asset.name}"></script>`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-17 11:44:48 +02:00
|
|
|
const fullHtml = `
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta charset="UTF-8">
|
2019-12-30 21:44:15 +02:00
|
|
|
${headers.join('\n')}
|
2019-12-17 11:44:48 +02:00
|
|
|
<title>${escapeHtml(item.title)}</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<div class="exported-note">${noteContent.join('\n\n')}</div>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
`;
|
|
|
|
|
|
|
|
await shim.fsDriver().writeFile(noteFilePath, fullHtml, 'utf-8');
|
2019-12-15 20:41:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async processResource(resource, filePath) {
|
|
|
|
if (!Resource.isSupportedImageMimeType(resource.mime)) {
|
|
|
|
const destResourcePath = `${this.resourceDir_}/${basename(filePath)}`;
|
|
|
|
await shim.fsDriver().copy(filePath, destResourcePath);
|
|
|
|
this.resources_.push(resource);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async close() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = InteropService_Exporter_Html;
|