2025-08-22 20:17:00 +03:00

203 lines
7.1 KiB
TypeScript

import utils from "./utils.js";
import server from "./server.js";
type ExecFunction = (command: string, cb: (err: string, stdout: string, stderror: string) => void) => void;
interface TmpResponse {
tmpFilePath: string;
}
function checkType(type: string) {
if (type !== "notes" && type !== "attachments") {
throw new Error(`Unrecognized type '${type}', should be 'notes' or 'attachments'`);
}
}
function getFileUrl(type: string, noteId?: string) {
checkType(type);
return getUrlForDownload(`api/${type}/${noteId}/download`);
}
function getOpenFileUrl(type: string, noteId: string) {
checkType(type);
return getUrlForDownload(`api/${type}/${noteId}/open`);
}
function download(url: string) {
if (utils.isElectron()) {
const remote = utils.dynamicRequire("@electron/remote");
remote.getCurrentWebContents().downloadURL(url);
} else {
window.location.href = url;
}
}
export function downloadFileNote(noteId: string) {
const url = `${getFileUrl("notes", noteId)}?${Date.now()}`; // don't use cache
download(url);
}
function downloadAttachment(attachmentId: string) {
const url = `${getFileUrl("attachments", attachmentId)}?${Date.now()}`; // don't use cache
download(url);
}
async function openCustom(type: string, entityId: string, mime: string) {
checkType(type);
if (!utils.isElectron() || utils.isMac()) {
return;
}
const resp = await server.post<TmpResponse>(`${type}/${entityId}/save-to-tmp-dir`);
let filePath = resp.tmpFilePath;
const exec = utils.dynamicRequire("child_process").exec as ExecFunction;
const platform = process.platform;
if (platform === "linux") {
// we don't know which terminal is available, try in succession
const terminals = ["x-terminal-emulator", "gnome-terminal", "konsole", "xterm", "xfce4-terminal", "mate-terminal", "rxvt", "terminator", "terminology"];
const openFileWithTerminal = (terminal: string) => {
const command = `${terminal} -e 'mimeopen -d "${filePath}"'`;
console.log(`Open Note custom: ${command} `);
exec(command, (error, stdout, stderr) => {
if (error) {
console.error(`Open Note custom: Failed to open file with ${terminal}: ${error}`);
searchTerminal(terminals.indexOf(terminal) + 1);
} else {
console.log(`Open Note custom: File opened with ${terminal}: ${stdout}`);
}
});
};
const searchTerminal = (index: number) => {
const terminal = terminals[index];
if (!terminal) {
console.error("Open Note custom: No terminal found!");
// TODO: Remove {url: true} if not needed.
(open as any)(getFileUrl(type, entityId), { url: true });
return;
}
exec(`which ${terminal}`, (error, stdout, stderr) => {
if (stdout.trim()) {
openFileWithTerminal(terminal);
} else {
searchTerminal(index + 1);
}
});
};
searchTerminal(0);
} else if (platform === "win32") {
if (filePath.indexOf("/") !== -1) {
// Note that the path separator must be \ instead of /
filePath = filePath.replace(/\//g, "\\");
}
const command = `rundll32.exe shell32.dll,OpenAs_RunDLL ` + filePath;
exec(command, (err, stdout, stderr) => {
if (err) {
console.error("Open Note custom: ", err);
// TODO: This appears to be broken, since getFileUrl expects two arguments, with the first one being the type.
// Also don't know why {url: true} is passed.
(open as any)(getFileUrl(entityId), { url: true });
return;
}
});
} else {
console.log('Currently "Open Note custom" only supports linux and windows systems');
// TODO: This appears to be broken, since getFileUrl expects two arguments, with the first one being the type.
// Also don't know why {url: true} is passed.
(open as any)(getFileUrl(entityId), { url: true });
}
}
const openNoteCustom = async (noteId: string, mime: string) => await openCustom("notes", noteId, mime);
const openAttachmentCustom = async (attachmentId: string, mime: string) => await openCustom("attachments", attachmentId, mime);
function downloadRevision(noteId: string, revisionId: string) {
const url = getUrlForDownload(`api/revisions/${revisionId}/download`);
download(url);
}
/**
* @param url - should be without initial slash!!!
*/
function getUrlForDownload(url: string) {
if (utils.isElectron()) {
// electron needs absolute URL, so we extract current host, port, protocol
return `${getHost()}/${url}`;
} else {
// web server can be deployed on subdomain, so we need to use a relative path
return url;
}
}
function canOpenInBrowser(mime: string) {
return mime === "application/pdf" || mime.startsWith("image") || mime.startsWith("audio") || mime.startsWith("video");
}
async function openExternally(type: string, entityId: string, mime: string) {
checkType(type);
if (utils.isElectron()) {
const resp = await server.post<TmpResponse>(`${type}/${entityId}/save-to-tmp-dir`);
const electron = utils.dynamicRequire("electron");
const res = await electron.shell.openPath(resp.tmpFilePath);
if (res) {
// fallback in case there's no default application for this file
window.open(getFileUrl(type, entityId));
}
} else {
// allow browser to handle opening common file
if (canOpenInBrowser(mime)) {
window.open(getOpenFileUrl(type, entityId));
} else {
window.location.href = getFileUrl(type, entityId);
}
}
}
export const openNoteExternally = async (noteId: string, mime: string) => await openExternally("notes", noteId, mime);
const openAttachmentExternally = async (attachmentId: string, mime: string) => await openExternally("attachments", attachmentId, mime);
function getHost() {
const url = new URL(window.location.href);
return `${url.protocol}//${url.hostname}:${url.port}`;
}
async function openDirectory(directory: string) {
try {
if (utils.isElectron()) {
const electron = utils.dynamicRequire("electron");
const res = await electron.shell.openPath(directory);
if (res) {
console.error("Failed to open directory:", res);
}
} else {
console.error("Not running in an Electron environment.");
}
} catch (err: any) {
// Handle file system errors (e.g. path does not exist or is inaccessible)
console.error("Error:", err.message);
}
}
export default {
download,
downloadFileNote,
downloadRevision,
downloadAttachment,
getUrlForDownload,
openNoteExternally,
openAttachmentExternally,
openNoteCustom,
openAttachmentCustom,
openDirectory
};