mirror of https://github.com/mifi/lossless-cut
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
324 lines
9.9 KiB
JavaScript
324 lines
9.9 KiB
JavaScript
import Swal from 'sweetalert2';
|
|
import i18n from 'i18next';
|
|
import lodashTemplate from 'lodash/template';
|
|
import pMap from 'p-map';
|
|
|
|
const { dirname, parse: parsePath, join, basename, extname, isAbsolute, resolve } = window.require('path');
|
|
const fs = window.require('fs-extra');
|
|
const open = window.require('open');
|
|
const os = window.require('os');
|
|
const trash = window.require('trash');
|
|
|
|
const { readdir, unlink } = fs;
|
|
|
|
export function getFileDir(filePath) {
|
|
return filePath ? dirname(filePath) : undefined;
|
|
}
|
|
|
|
export function getOutDir(customOutDir, filePath) {
|
|
if (customOutDir) return customOutDir;
|
|
if (filePath) return getFileDir(filePath);
|
|
return undefined;
|
|
}
|
|
|
|
export function getFileBaseName(filePath) {
|
|
if (!filePath) return undefined;
|
|
const parsed = parsePath(filePath);
|
|
return parsed.name;
|
|
}
|
|
|
|
export function getOutPath({ customOutDir, filePath, nameSuffix }) {
|
|
if (!filePath) return undefined;
|
|
return join(getOutDir(customOutDir, filePath), `${getFileBaseName(filePath)}-${nameSuffix}`);
|
|
}
|
|
|
|
export async function havePermissionToReadFile(filePath) {
|
|
try {
|
|
const fd = await fs.open(filePath, 'r');
|
|
try {
|
|
await fs.close(fd);
|
|
} catch (err) {
|
|
console.error('Failed to close fd', err);
|
|
}
|
|
} catch (err) {
|
|
if (['EPERM', 'EACCES'].includes(err.code)) return false;
|
|
console.error(err);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
export async function checkDirWriteAccess(dirPath) {
|
|
try {
|
|
await fs.access(dirPath, fs.constants.W_OK);
|
|
} catch (err) {
|
|
if (err.code === 'EPERM') return false; // Thrown on Mac (MAS build) when user has not yet allowed access
|
|
if (err.code === 'EACCES') return false; // Thrown on Linux when user doesn't have access to output dir
|
|
console.error(err);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
export async function pathExists(pathIn) {
|
|
return fs.pathExists(pathIn);
|
|
}
|
|
|
|
export async function getPathReadAccessError(pathIn) {
|
|
try {
|
|
await fs.access(pathIn, fs.constants.R_OK);
|
|
return undefined;
|
|
} catch (err) {
|
|
return err.code;
|
|
}
|
|
}
|
|
|
|
export async function dirExists(dirPath) {
|
|
return (await pathExists(dirPath)) && (await fs.lstat(dirPath)).isDirectory();
|
|
}
|
|
|
|
export async function transferTimestamps(inPath, outPath, offset = 0) {
|
|
try {
|
|
const { atime, mtime } = await fs.stat(inPath);
|
|
await fs.utimes(outPath, (atime.getTime() / 1000) + offset, (mtime.getTime() / 1000) + offset);
|
|
} catch (err) {
|
|
console.error('Failed to set output file modified time', err);
|
|
}
|
|
}
|
|
|
|
export const toast = Swal.mixin({
|
|
toast: true,
|
|
position: 'top',
|
|
showConfirmButton: false,
|
|
showCloseButton: true,
|
|
timer: 5000,
|
|
timerProgressBar: true,
|
|
didOpen: (self) => {
|
|
self.addEventListener('mouseenter', Swal.stopTimer);
|
|
self.addEventListener('mouseleave', Swal.resumeTimer);
|
|
},
|
|
});
|
|
|
|
export const errorToast = (text) => toast.fire({
|
|
icon: 'error',
|
|
text,
|
|
});
|
|
|
|
export function handleError(arg1, arg2) {
|
|
console.error('handleError', arg1, arg2);
|
|
|
|
let msg;
|
|
let errorMsg;
|
|
if (typeof arg1 === 'string') msg = arg1;
|
|
else if (typeof arg2 === 'string') msg = arg2;
|
|
|
|
if (arg1 instanceof Error) errorMsg = arg1.message;
|
|
if (arg2 instanceof Error) errorMsg = arg2.message;
|
|
|
|
toast.fire({
|
|
icon: 'error',
|
|
title: msg || i18n.t('An error has occurred.'),
|
|
text: errorMsg ? errorMsg.substr(0, 300) : undefined,
|
|
});
|
|
}
|
|
|
|
|
|
export const openDirToast = async ({ dirPath, ...props }) => {
|
|
const { value } = await toast.fire({ icon: 'success', timer: 5000, showConfirmButton: true, confirmButtonText: i18n.t('Show'), showCancelButton: true, cancelButtonText: i18n.t('Close'), ...props });
|
|
if (value) open(dirPath);
|
|
};
|
|
|
|
export function setFileNameTitle(filePath) {
|
|
const appName = 'LosslessCut';
|
|
document.title = filePath ? `${appName} - ${basename(filePath)}` : appName;
|
|
}
|
|
|
|
export function filenamify(name) {
|
|
return name.replace(/[^0-9a-zA-Z_\-.]/g, '_');
|
|
}
|
|
|
|
export function withBlur(cb) {
|
|
return (e) => {
|
|
cb(e);
|
|
e.target.blur();
|
|
};
|
|
}
|
|
|
|
export function dragPreventer(ev) {
|
|
ev.preventDefault();
|
|
}
|
|
|
|
export const isMasBuild = window.process.mas;
|
|
export const isWindowsStoreBuild = window.process.windowsStore;
|
|
export const isStoreBuild = isMasBuild || isWindowsStoreBuild;
|
|
|
|
export const isDurationValid = (duration) => Number.isFinite(duration) && duration > 0;
|
|
|
|
export const platform = os.platform();
|
|
export const arch = os.arch();
|
|
|
|
export const isWindows = platform === 'win32';
|
|
export const isMac = platform === 'darwin';
|
|
|
|
export function getExtensionForFormat(format) {
|
|
const ext = {
|
|
matroska: 'mkv',
|
|
ipod: 'm4a',
|
|
}[format];
|
|
|
|
return ext || format;
|
|
}
|
|
|
|
export function getOutFileExtension({ isCustomFormatSelected, outFormat, filePath }) {
|
|
if (!isCustomFormatSelected) {
|
|
const ext = extname(filePath);
|
|
// QuickTime is quirky about the file extension of mov files (has to be .mov)
|
|
// https://github.com/mifi/lossless-cut/issues/1075#issuecomment-1072084286
|
|
const hasMovIncorrectExtension = outFormat === 'mov' && ext.toLowerCase() !== '.mov';
|
|
|
|
// OK, just keep the current extension. Because most players will not care about the extension
|
|
if (!hasMovIncorrectExtension) return extname(filePath);
|
|
}
|
|
return `.${getExtensionForFormat(outFormat)}`;
|
|
}
|
|
|
|
// This is used as a fallback and so it has to always generate unique file names
|
|
// eslint-disable-next-line no-template-curly-in-string
|
|
export const defaultOutSegTemplate = '${FILENAME}-${CUT_FROM}-${CUT_TO}${SEG_SUFFIX}${EXT}';
|
|
|
|
export function generateSegFileName({ template, inputFileNameWithoutExt, segSuffix, ext, segNum, segLabel, cutFrom, cutTo, tags }) {
|
|
const compiled = lodashTemplate(template);
|
|
const data = {
|
|
FILENAME: inputFileNameWithoutExt,
|
|
SEG_SUFFIX: segSuffix,
|
|
EXT: ext,
|
|
SEG_NUM: segNum,
|
|
SEG_LABEL: segLabel,
|
|
CUT_FROM: cutFrom,
|
|
CUT_TO: cutTo,
|
|
SEG_TAGS: {
|
|
// allow both original case and uppercase
|
|
...tags,
|
|
...Object.fromEntries(Object.entries(tags).map(([key, value]) => [`${key.toLocaleUpperCase('en-US')}`, value])),
|
|
},
|
|
};
|
|
return compiled(data);
|
|
}
|
|
|
|
export const hasDuplicates = (arr) => new Set(arr).size !== arr.length;
|
|
|
|
// Need to resolve relative paths from the command line https://github.com/mifi/lossless-cut/issues/639
|
|
export const resolvePathIfNeeded = (inPath) => (isAbsolute(inPath) ? inPath : resolve(inPath));
|
|
|
|
export const html5ifiedPrefix = 'html5ified-';
|
|
export const html5dummySuffix = 'dummy';
|
|
|
|
export async function findExistingHtml5FriendlyFile(fp, cod) {
|
|
// The order is the priority we will search:
|
|
const suffixes = ['slowest', 'slow-audio', 'slow', 'fast-audio-remux', 'fast-audio', 'fast', 'fastest-audio', 'fastest-audio-remux', html5dummySuffix];
|
|
const prefix = `${getFileBaseName(fp)}-${html5ifiedPrefix}`;
|
|
|
|
const outDir = getOutDir(cod, fp);
|
|
const dirEntries = await readdir(outDir);
|
|
|
|
const html5ifiedDirEntries = dirEntries.filter((entry) => entry.startsWith(prefix));
|
|
|
|
let matches = [];
|
|
suffixes.forEach((suffix) => {
|
|
const entryWithSuffix = html5ifiedDirEntries.find((entry) => new RegExp(`${suffix}\\..*$`).test(entry.replace(prefix, '')));
|
|
if (entryWithSuffix) matches = [...matches, { entry: entryWithSuffix, suffix }];
|
|
});
|
|
|
|
const nonMatches = html5ifiedDirEntries.filter((entry) => !matches.some((m) => m.entry === entry)).map((entry) => ({ entry }));
|
|
|
|
// Allow for non-suffix matches too, e.g. user has a custom html5ified- file but with none of the suffixes above (but last priority)
|
|
matches = [...matches, ...nonMatches];
|
|
|
|
// console.log(matches);
|
|
if (matches.length < 1) return undefined;
|
|
|
|
const { suffix, entry } = matches[0];
|
|
|
|
return {
|
|
path: join(outDir, entry),
|
|
usingDummyVideo: ['fastest-audio', 'fastest-audio-remux', html5dummySuffix].includes(suffix),
|
|
};
|
|
}
|
|
|
|
export function getHtml5ifiedPath(cod, fp, type) {
|
|
// See also inside ffmpegHtml5ify
|
|
const ext = (isMac && ['slowest', 'slow', 'slow-audio'].includes(type)) ? 'mp4' : 'mkv';
|
|
return getOutPath({ customOutDir: cod, filePath: fp, nameSuffix: `${html5ifiedPrefix}${type}.${ext}` });
|
|
}
|
|
|
|
export async function deleteFiles({ toDelete, paths: { previewFilePath, sourceFilePath, projectFilePath } }) {
|
|
const failedToTrashFiles = [];
|
|
|
|
if (toDelete.tmpFiles && previewFilePath) {
|
|
try {
|
|
await trash(previewFilePath);
|
|
} catch (err) {
|
|
console.error(err);
|
|
failedToTrashFiles.push(previewFilePath);
|
|
}
|
|
}
|
|
if (toDelete.projectFile && projectFilePath) {
|
|
try {
|
|
// throw new Error('test');
|
|
await trash(projectFilePath);
|
|
} catch (err) {
|
|
console.error(err);
|
|
failedToTrashFiles.push(projectFilePath);
|
|
}
|
|
}
|
|
if (toDelete.sourceFile) {
|
|
try {
|
|
await trash(sourceFilePath);
|
|
} catch (err) {
|
|
console.error(err);
|
|
failedToTrashFiles.push(sourceFilePath);
|
|
}
|
|
}
|
|
|
|
if (failedToTrashFiles.length === 0) return; // All good!
|
|
|
|
const { value } = await Swal.fire({
|
|
icon: 'warning',
|
|
text: i18n.t('Unable to move file to trash. Do you want to permanently delete it?'),
|
|
confirmButtonText: i18n.t('Permanently delete'),
|
|
showCancelButton: true,
|
|
});
|
|
|
|
if (value) {
|
|
await pMap(failedToTrashFiles, async (path) => unlink(path), { concurrency: 1 });
|
|
}
|
|
}
|
|
|
|
export const deleteDispositionValue = 'llc_disposition_remove';
|
|
|
|
export const mirrorTransform = 'matrix(-1, 0, 0, 1, 0, 0)';
|
|
|
|
// A bit hacky but it works, unless someone has a file called "No space left on device" ( ͡° ͜ʖ ͡°)
|
|
export const isOutOfSpaceError = (err) => (
|
|
err && (err.exitCode === 1 || err.code === 'ENOENT')
|
|
&& typeof err.stderr === 'string' && err.stderr.includes('No space left on device')
|
|
);
|
|
|
|
// https://stackoverflow.com/a/2450976/6519037
|
|
export function shuffleArray(arrayIn) {
|
|
const array = [...arrayIn];
|
|
let currentIndex = array.length;
|
|
let randomIndex;
|
|
|
|
// While there remain elements to shuffle...
|
|
while (currentIndex !== 0) {
|
|
// Pick a remaining element...
|
|
randomIndex = Math.floor(Math.random() * currentIndex);
|
|
currentIndex -= 1;
|
|
|
|
// And swap it with the current element.
|
|
[array[currentIndex], array[randomIndex]] = [
|
|
array[randomIndex], array[currentIndex]];
|
|
}
|
|
|
|
return array;
|
|
}
|