trilium/src/services/image.js

138 lines
4.3 KiB
JavaScript
Raw Normal View History

2018-02-11 13:18:59 +08:00
"use strict";
const repository = require('./repository');
const log = require('./log');
const protectedSessionService = require('./protected_session');
2018-11-08 18:08:16 +08:00
const noteService = require('./notes');
const optionService = require('./options');
2018-02-11 13:18:59 +08:00
const imagemin = require('imagemin');
const imageminMozJpeg = require('imagemin-mozjpeg');
const imageminPngQuant = require('imagemin-pngquant');
const imageminGifLossy = require('imagemin-giflossy');
const jimp = require('jimp');
const imageType = require('image-type');
const sanitizeFilename = require('sanitize-filename');
2019-11-09 18:58:52 +08:00
const noteRevisionService = require('./note_revisions.js');
2018-02-11 13:18:59 +08:00
2019-11-09 05:34:30 +08:00
async function processImage(uploadBuffer, originalName, shrinkImageSwitch) {
const origImageFormat = imageType(uploadBuffer);
2019-07-11 02:38:27 +08:00
if (origImageFormat.ext === "webp") {
// JIMP does not support webp at the moment: https://github.com/oliver-moran/jimp/issues/144
shrinkImageSwitch = false;
}
2019-11-09 05:34:30 +08:00
const finalImageBuffer = shrinkImageSwitch ? await shrinkImage(uploadBuffer, originalName) : uploadBuffer;
2018-02-11 13:18:59 +08:00
2019-02-24 19:24:28 +08:00
const imageFormat = imageType(finalImageBuffer);
2018-02-11 13:18:59 +08:00
2019-11-09 05:34:30 +08:00
return {
buffer: finalImageBuffer,
imageFormat
};
}
async function updateImage(noteId, uploadBuffer, originalName) {
const {buffer, imageFormat} = await processImage(uploadBuffer, originalName, true);
const note = await repository.getNote(noteId);
2019-11-09 18:58:52 +08:00
await noteRevisionService.createNoteRevision(note);
2019-11-09 06:09:57 +08:00
2019-11-09 05:34:30 +08:00
note.mime = 'image/' + imageFormat.ext.toLowerCase();
await note.setContent(buffer);
await note.setLabel('originalFileName', originalName);
2019-11-09 06:09:57 +08:00
2019-11-09 18:58:52 +08:00
await noteRevisionService.protectNoteRevisions(note);
2019-11-09 05:34:30 +08:00
}
async function saveImage(parentNoteId, uploadBuffer, originalName, shrinkImageSwitch) {
const {buffer, imageFormat} = await processImage(uploadBuffer, originalName, shrinkImageSwitch);
2019-07-11 05:01:30 +08:00
const fileName = sanitizeFilename(originalName);
2018-02-11 13:18:59 +08:00
2019-11-09 05:34:30 +08:00
const parentNote = await repository.getNote(parentNoteId);
2019-11-16 18:09:52 +08:00
const {note} = await noteService.createNewNote({
parentNoteId,
title: fileName,
content: buffer,
2018-11-08 18:08:16 +08:00
type: 'image',
mime: 'image/' + imageFormat.ext.toLowerCase(),
2019-11-16 18:09:52 +08:00
isProtected: parentNote.isProtected && protectedSessionService.isProtectedSessionAvailable()
2018-11-08 18:08:16 +08:00
});
2018-02-11 13:18:59 +08:00
2019-11-16 18:09:52 +08:00
await note.addLabel('originalFileName', originalName);
return {
fileName,
2019-02-26 04:22:57 +08:00
note,
2018-11-08 18:08:16 +08:00
noteId: note.noteId,
2019-11-09 06:09:57 +08:00
url: `api/images/${note.noteId}/${fileName}`
};
2018-02-11 13:18:59 +08:00
}
2019-03-04 03:41:03 +08:00
async function shrinkImage(buffer, originalName) {
2019-02-24 19:24:28 +08:00
const resizedImage = await resize(buffer);
let finalImageBuffer;
try {
finalImageBuffer = await optimize(resizedImage);
} catch (e) {
2019-07-11 02:38:27 +08:00
log.error("Failed to optimize image '" + originalName + "'\nStack: " + e.stack);
2019-02-24 19:24:28 +08:00
finalImageBuffer = resizedImage;
}
// if resizing & shrinking did not help with size then save the original
// (can happen when e.g. resizing PNG into JPEG)
if (finalImageBuffer.byteLength >= buffer.byteLength) {
finalImageBuffer = buffer;
}
2019-02-24 19:24:28 +08:00
return finalImageBuffer;
}
2018-02-11 13:18:59 +08:00
async function resize(buffer) {
const imageMaxWidthHeight = await optionService.getOptionInt('imageMaxWidthHeight');
2018-02-11 13:18:59 +08:00
const image = await jimp.read(buffer);
if (image.bitmap.width > image.bitmap.height && image.bitmap.width > imageMaxWidthHeight) {
image.resize(imageMaxWidthHeight, jimp.AUTO);
2018-02-11 13:18:59 +08:00
}
else if (image.bitmap.height > imageMaxWidthHeight) {
image.resize(jimp.AUTO, imageMaxWidthHeight);
2018-02-11 13:18:59 +08:00
}
// we do resizing with max quality which will be trimmed during optimization step next
image.quality(100);
// when converting PNG to JPG we lose alpha channel, this is replaced by white to match Trilium white background
image.background(0xFFFFFFFF);
2019-02-26 04:22:57 +08:00
return image.getBufferAsync(jimp.MIME_JPEG);
2018-02-11 13:18:59 +08:00
}
async function optimize(buffer) {
return await imagemin.buffer(buffer, {
plugins: [
imageminMozJpeg({
quality: await optionService.getOptionInt('imageJpegQuality')
2018-02-11 13:18:59 +08:00
}),
imageminPngQuant({
2019-07-11 02:38:27 +08:00
quality: [0, 0.7]
2018-02-11 13:18:59 +08:00
}),
imageminGifLossy({
lossy: 80,
optimize: '3' // needs to be string
})
]
});
}
module.exports = {
2019-11-09 05:34:30 +08:00
saveImage,
updateImage
2018-02-11 13:18:59 +08:00
};