2023-09-17 11:40:50 +01:00
|
|
|
/* eslint-disable multiline-comment-style */
|
|
|
|
|
2023-09-17 14:08:55 +01:00
|
|
|
import Resource from '../../../models/Resource';
|
|
|
|
import Setting from '../../../models/Setting';
|
|
|
|
import shim from '../../../shim';
|
|
|
|
import { Rectangle } from './types';
|
|
|
|
|
2023-09-17 11:40:50 +01:00
|
|
|
export interface CreateFromBufferOptions {
|
|
|
|
width?: number;
|
|
|
|
height?: number;
|
|
|
|
scaleFactor?: number;
|
|
|
|
}
|
|
|
|
|
2024-03-27 11:53:24 -07:00
|
|
|
export interface CreateFromPdfOptions {
|
|
|
|
/**
|
|
|
|
* The first page to export. Defaults to `1`, the first page in
|
|
|
|
* the document.
|
|
|
|
*/
|
|
|
|
minPage?: number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of the last page to convert. Defaults to the last page
|
|
|
|
* if not given.
|
|
|
|
*
|
|
|
|
* If `maxPage` is greater than the number of pages in the PDF, all pages
|
|
|
|
* in the PDF will be converted to images.
|
|
|
|
*/
|
|
|
|
maxPage?: number;
|
|
|
|
|
|
|
|
scaleFactor?: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface PdfInfo {
|
|
|
|
pageCount: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface Implementation {
|
2024-04-27 11:35:49 +01:00
|
|
|
createFromPath: (path: string)=> Promise<unknown>;
|
|
|
|
createFromPdf: (path: string, options: CreateFromPdfOptions)=> Promise<unknown[]>;
|
2024-03-27 11:53:24 -07:00
|
|
|
getPdfInfo: (path: string)=> Promise<PdfInfo>;
|
|
|
|
}
|
|
|
|
|
2023-09-17 11:40:50 +01:00
|
|
|
export interface ResizeOptions {
|
|
|
|
width?: number;
|
|
|
|
height?: number;
|
|
|
|
quality?: 'good' | 'better' | 'best';
|
|
|
|
}
|
|
|
|
|
|
|
|
export type Handle = string;
|
|
|
|
|
|
|
|
interface Image {
|
|
|
|
handle: Handle;
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-09-17 11:40:50 +01:00
|
|
|
data: any;
|
|
|
|
}
|
|
|
|
|
2024-03-27 11:53:24 -07:00
|
|
|
const getResourcePath = async (resourceId: string): Promise<string> => {
|
|
|
|
const resource = await Resource.load(resourceId);
|
|
|
|
if (!resource) throw new Error(`No such resource: ${resourceId}`);
|
|
|
|
const resourcePath = await Resource.fullPath(resource);
|
|
|
|
if (!(await shim.fsDriver().exists(resourcePath))) throw new Error(`Could not load resource path: ${resourcePath}`);
|
|
|
|
return resourcePath;
|
|
|
|
};
|
|
|
|
|
2023-09-17 11:40:50 +01:00
|
|
|
/**
|
|
|
|
* Provides imaging functions to resize or process images. You create an image
|
|
|
|
* using one of the `createFrom` functions, then use the other functions to
|
|
|
|
* process the image.
|
|
|
|
*
|
|
|
|
* Images are associated with a handle which is what will be available to the
|
|
|
|
* plugin. Once you are done with an image, free it using the `free()` function.
|
|
|
|
*
|
|
|
|
* [View the
|
|
|
|
* example](https://github.com/laurent22/joplin/blob/dev/packages/app-cli/tests/support/plugins/imaging/src/index.ts)
|
|
|
|
*
|
2024-03-26 04:36:15 -07:00
|
|
|
* <span class="platform-desktop">desktop</span>
|
2023-09-17 11:40:50 +01:00
|
|
|
*/
|
|
|
|
export default class JoplinImaging {
|
|
|
|
|
|
|
|
private implementation_: Implementation;
|
|
|
|
private images_: Image[] = [];
|
|
|
|
|
|
|
|
public constructor(implementation: Implementation) {
|
|
|
|
this.implementation_ = implementation;
|
|
|
|
}
|
|
|
|
|
|
|
|
private createImageHandle(): Handle {
|
|
|
|
return [Date.now(), Math.random()].join(':');
|
|
|
|
}
|
|
|
|
|
|
|
|
private imageByHandle(handle: Handle) {
|
|
|
|
const image = this.images_.find(i => i.handle === handle);
|
|
|
|
if (!image) throw new Error(`No image with handle ${handle}`);
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-09-17 11:40:50 +01:00
|
|
|
private cacheImage(data: any) {
|
|
|
|
const handle = this.createImageHandle();
|
|
|
|
this.images_.push({
|
|
|
|
handle,
|
|
|
|
data,
|
|
|
|
});
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
2023-09-17 18:07:24 +01:00
|
|
|
// Create an image from a buffer - however only use this for very small
|
|
|
|
// images. It requires transferring the full image data from the plugin to
|
|
|
|
// the app, which is extremely slow and will freeze the app. Instead, use
|
|
|
|
// `createFromPath` or `createFromResource`, which will manipulate the image
|
|
|
|
// data directly from the main process.
|
|
|
|
//
|
|
|
|
// public async createFromBuffer(buffer: any, options: CreateFromBufferOptions = null): Promise<Handle> {
|
|
|
|
// return this.cacheImage(this.implementation_.nativeImage.createFromBuffer(buffer, options));
|
|
|
|
// }
|
2023-09-17 11:40:50 +01:00
|
|
|
|
2024-04-27 11:35:49 +01:00
|
|
|
/**
|
|
|
|
* Creates an image from the provided path. Note that images and PDFs are supported. If you
|
|
|
|
* provide a URL instead of a local path, the file will be downloaded first then converted to an
|
|
|
|
* image.
|
|
|
|
*/
|
2023-09-17 14:08:55 +01:00
|
|
|
public async createFromPath(filePath: string): Promise<Handle> {
|
2024-04-27 11:35:49 +01:00
|
|
|
return this.cacheImage(await this.implementation_.createFromPath(filePath));
|
2023-09-17 14:08:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public async createFromResource(resourceId: string): Promise<Handle> {
|
2024-03-27 11:53:24 -07:00
|
|
|
return this.createFromPath(await getResourcePath(resourceId));
|
|
|
|
}
|
|
|
|
|
|
|
|
public async createFromPdfPath(path: string, options?: CreateFromPdfOptions): Promise<Handle[]> {
|
2024-04-27 11:35:49 +01:00
|
|
|
const images = await this.implementation_.createFromPdf(path, options);
|
2024-03-27 11:53:24 -07:00
|
|
|
return images.map(image => this.cacheImage(image));
|
|
|
|
}
|
|
|
|
|
|
|
|
public async createFromPdfResource(resourceId: string, options?: CreateFromPdfOptions): Promise<Handle[]> {
|
|
|
|
return this.createFromPdfPath(await getResourcePath(resourceId), options);
|
|
|
|
}
|
|
|
|
|
|
|
|
public async getPdfInfoFromPath(path: string): Promise<PdfInfo> {
|
|
|
|
return await this.implementation_.getPdfInfo(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
public async getPdfInfoFromResource(resourceId: string): Promise<PdfInfo> {
|
|
|
|
return this.getPdfInfoFromPath(await getResourcePath(resourceId));
|
2023-09-17 14:08:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public async getSize(handle: Handle) {
|
|
|
|
const image = this.imageByHandle(handle);
|
|
|
|
return image.data.getSize();
|
|
|
|
}
|
|
|
|
|
2023-09-17 11:40:50 +01:00
|
|
|
public async resize(handle: Handle, options: ResizeOptions = null) {
|
|
|
|
const image = this.imageByHandle(handle);
|
|
|
|
const resizedImage = image.data.resize(options);
|
|
|
|
return this.cacheImage(resizedImage);
|
|
|
|
}
|
|
|
|
|
2024-02-26 10:16:23 +00:00
|
|
|
public async crop(handle: Handle, rectangle: Rectangle) {
|
2023-09-17 14:08:55 +01:00
|
|
|
const image = this.imageByHandle(handle);
|
2024-02-26 10:16:23 +00:00
|
|
|
const croppedImage = image.data.crop(rectangle);
|
2023-09-17 14:08:55 +01:00
|
|
|
return this.cacheImage(croppedImage);
|
|
|
|
}
|
|
|
|
|
2023-09-17 18:07:24 +01:00
|
|
|
// Warning: requires transferring the complete image from the app to the
|
|
|
|
// plugin which may freeze the app. Consider using one of the `toXxxFile()`
|
|
|
|
// or `toXxxResource()` methods instead.
|
|
|
|
//
|
|
|
|
// public async toDataUrl(handle: Handle): Promise<string> {
|
|
|
|
// const image = this.imageByHandle(handle);
|
|
|
|
// return image.data.toDataURL();
|
|
|
|
// }
|
|
|
|
|
|
|
|
// Warnings: requires transferring the complete image from the app to the
|
|
|
|
// plugin which may freeze the app. Consider using one of the `toXxxFile()`
|
|
|
|
// or `toXxxResource()` methods instead.
|
|
|
|
//
|
|
|
|
// public async toBase64(handle: Handle) {
|
|
|
|
// const dataUrl = await this.toDataUrl(handle);
|
|
|
|
// const s = dataUrl.split('base64,');
|
|
|
|
// if (s.length !== 2) throw new Error('Could not convert to base64');
|
|
|
|
// return s[1];
|
|
|
|
// }
|
2023-09-17 11:40:50 +01:00
|
|
|
|
2023-09-17 14:08:55 +01:00
|
|
|
public async toPngFile(handle: Handle, filePath: string) {
|
|
|
|
const image = this.imageByHandle(handle);
|
|
|
|
const data = image.data.toPNG();
|
|
|
|
await shim.fsDriver().writeFile(filePath, data, 'buffer');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Quality is between 0 and 100
|
|
|
|
*/
|
|
|
|
public async toJpgFile(handle: Handle, filePath: string, quality = 80) {
|
|
|
|
const image = this.imageByHandle(handle);
|
|
|
|
const data = image.data.toJPEG(quality);
|
|
|
|
await shim.fsDriver().writeFile(filePath, data, 'buffer');
|
|
|
|
}
|
|
|
|
|
|
|
|
private tempFilePath(ext: string) {
|
|
|
|
return `${Setting.value('tempDir')}/${Date.now()}_${Math.random()}.${ext}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new Joplin resource from the image data. The image will be
|
|
|
|
* first converted to a JPEG.
|
|
|
|
*/
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-09-17 14:08:55 +01:00
|
|
|
public async toJpgResource(handle: Handle, resourceProps: any, quality = 80) {
|
|
|
|
const tempFilePath = this.tempFilePath('jpg');
|
|
|
|
await this.toJpgFile(handle, tempFilePath, quality);
|
|
|
|
const newResource = await shim.createResourceFromPath(tempFilePath, resourceProps, { resizeLargeImages: 'never' });
|
|
|
|
await shim.fsDriver().remove(tempFilePath);
|
|
|
|
return newResource;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new Joplin resource from the image data. The image will be
|
|
|
|
* first converted to a PNG.
|
|
|
|
*/
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-09-17 14:08:55 +01:00
|
|
|
public async toPngResource(handle: Handle, resourceProps: any) {
|
|
|
|
const tempFilePath = this.tempFilePath('png');
|
|
|
|
await this.toPngFile(handle, tempFilePath);
|
|
|
|
const newResource = await shim.createResourceFromPath(tempFilePath, resourceProps, { resizeLargeImages: 'never' });
|
|
|
|
await shim.fsDriver().remove(tempFilePath);
|
|
|
|
return newResource;
|
|
|
|
}
|
|
|
|
|
2023-09-17 11:40:50 +01:00
|
|
|
/**
|
|
|
|
* Image data is not automatically deleted by Joplin so make sure you call
|
|
|
|
* this method on the handle once you are done.
|
|
|
|
*/
|
2024-03-27 11:53:24 -07:00
|
|
|
public async free(handles: Handle[]|Handle) {
|
|
|
|
if (!Array.isArray(handles)) {
|
|
|
|
handles = [handles];
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const handle of handles) {
|
|
|
|
const index = this.images_.findIndex(i => i.handle === handle);
|
|
|
|
if (index >= 0) this.images_.splice(index, 1);
|
|
|
|
}
|
2023-09-17 11:40:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|