mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-26 18:58:21 +02:00
129 lines
4.1 KiB
TypeScript
129 lines
4.1 KiB
TypeScript
import Resource from '@joplin/lib/models/Resource';
|
|
import Logger from '@joplin/utils/Logger';
|
|
import { HtmlToMarkdownHandler, MarkupToHtmlHandler } from './types';
|
|
|
|
const logger = Logger.create('contextMenuUtils');
|
|
|
|
export enum ContextMenuItemType {
|
|
None = '',
|
|
Image = 'image',
|
|
Resource = 'resource',
|
|
Text = 'text',
|
|
Link = 'link',
|
|
}
|
|
|
|
export interface ContextMenuOptions {
|
|
itemType: ContextMenuItemType;
|
|
resourceId: string;
|
|
mime: string;
|
|
filename: string;
|
|
linkToCopy: string;
|
|
textToCopy: string;
|
|
htmlToCopy: string;
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
|
insertContent: Function;
|
|
isReadOnly?: boolean;
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
|
fireEditorEvent: Function;
|
|
htmlToMd: HtmlToMarkdownHandler;
|
|
mdToHtml: MarkupToHtmlHandler;
|
|
}
|
|
|
|
export interface ContextMenuItem {
|
|
label: string;
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
|
onAction: Function;
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
|
isActive: Function;
|
|
}
|
|
|
|
export interface ContextMenuItems {
|
|
[key: string]: ContextMenuItem;
|
|
}
|
|
|
|
export async function resourceInfo(options: ContextMenuOptions) {
|
|
const resource = options.resourceId ? await Resource.load(options.resourceId) : null;
|
|
const resourcePath = resource ? Resource.fullPath(resource) : null;
|
|
const filename = resource ? (resource.filename ? resource.filename : resource.title) : options.filename ? options.filename : '';
|
|
return { resource, resourcePath, filename };
|
|
}
|
|
|
|
export function textToDataUri(text: string, mime: string): string {
|
|
return `data:${mime};base64,${Buffer.from(text).toString('base64')}`;
|
|
}
|
|
export const svgDimensions = (document: Document, svg: string) => {
|
|
let width: number;
|
|
let height: number;
|
|
try {
|
|
const parser = new DOMParser();
|
|
const id = parser.parseFromString(svg, 'text/html').querySelector('svg').id;
|
|
({ width, height } = document.querySelector<HTMLIFrameElement>('.noteTextViewer').contentWindow.document.querySelector(`#${id}`).getBoundingClientRect());
|
|
} catch (error) {
|
|
logger.warn('Could not get SVG dimensions.');
|
|
logger.warn('Error was: ', error);
|
|
}
|
|
if (!width || !height) {
|
|
return [undefined, undefined];
|
|
}
|
|
return [width, height];
|
|
};
|
|
export const svgUriToPng = (document: Document, svg: string, width: number, height: number) => {
|
|
return new Promise<Uint8Array>((resolve, reject) => {
|
|
let canvas: HTMLCanvasElement;
|
|
let img: HTMLImageElement;
|
|
|
|
const cleanUpAndReject = (e: Error) => {
|
|
if (canvas) canvas.remove();
|
|
if (img) img.remove();
|
|
return reject(e);
|
|
};
|
|
|
|
try {
|
|
img = document.createElement('img');
|
|
if (!img) throw new Error('Failed to create img element');
|
|
} catch (e) {
|
|
return cleanUpAndReject(e);
|
|
}
|
|
|
|
img.onload = function() {
|
|
try {
|
|
canvas = document.createElement('canvas');
|
|
if (!canvas) throw new Error('Failed to create canvas element');
|
|
if (!width || !height) {
|
|
const maxDimension = 1024;
|
|
if (img.width > img.height) {
|
|
width = maxDimension;
|
|
height = width * (img.height / img.width);
|
|
} else {
|
|
height = maxDimension;
|
|
width = height * (img.width / img.height);
|
|
}
|
|
}
|
|
canvas.width = width;
|
|
canvas.height = height;
|
|
const ctx = canvas.getContext('2d');
|
|
if (!ctx) throw new Error('Failed to get context');
|
|
ctx.drawImage(img, 0, 0, canvas.width, canvas.height, 0, 0, canvas.width, canvas.height);
|
|
const pngUri = canvas.toDataURL('image/png');
|
|
if (!pngUri) throw new Error('Failed to generate png uri');
|
|
const pngBase64 = pngUri.split(',')[1];
|
|
const byteString = atob(pngBase64);
|
|
// write the bytes of the string to a typed array
|
|
const buff = new Uint8Array(byteString.length);
|
|
for (let i = 0; i < byteString.length; i++) {
|
|
buff[i] = byteString.charCodeAt(i);
|
|
}
|
|
canvas.remove();
|
|
img.remove();
|
|
resolve(buff);
|
|
} catch (error) {
|
|
cleanUpAndReject(error);
|
|
}
|
|
};
|
|
img.onerror = function(e) {
|
|
cleanUpAndReject(new Error(e.toString()));
|
|
};
|
|
img.src = svg;
|
|
});
|
|
};
|