mirror of
https://github.com/mifi/lossless-cut.git
synced 2024-11-21 18:02:35 +01:00
parent
93fb2d2620
commit
6ddb3970f7
@ -82,7 +82,7 @@ import { askForOutDir, askForImportChapters, promptTimecode, askForFileOpenActio
|
||||
import { openSendReportDialog } from './reporting';
|
||||
import { fallbackLng } from './i18n';
|
||||
import { findSegmentsAtCursor, sortSegments, convertSegmentsToChapters, hasAnySegmentOverlap, isDurationValid, playOnlyCurrentSegment, getSegmentTags } from './segments';
|
||||
import { generateOutSegFileNames as generateOutSegFileNamesRaw, generateMergedFileNames as generateMergedFileNamesRaw, defaultOutSegTemplate, defaultMergedFileTemplate } from './util/outputNameTemplate';
|
||||
import { generateOutSegFileNames as generateOutSegFileNamesRaw, generateMergedFileNames as generateMergedFileNamesRaw, defaultOutSegTemplate, defaultCutMergedFileTemplate } from './util/outputNameTemplate';
|
||||
import { rightBarWidth, leftBarWidth, ffmpegExtractWindow, zoomMax } from './util/constants';
|
||||
import BigWaveform from './components/BigWaveform';
|
||||
|
||||
@ -193,7 +193,7 @@ function App() {
|
||||
}, [customFfPath]);
|
||||
|
||||
const outSegTemplateOrDefault = outSegTemplate || defaultOutSegTemplate;
|
||||
const mergedFileTemplateOrDefault = mergedFileTemplate || defaultMergedFileTemplate;
|
||||
const mergedFileTemplateOrDefault = mergedFileTemplate || defaultCutMergedFileTemplate;
|
||||
|
||||
useEffect(() => {
|
||||
const l = language || fallbackLng;
|
||||
|
@ -11,3 +11,8 @@
|
||||
border: .05em solid var(--gray7);
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.button:disabled {
|
||||
opacity: .5;
|
||||
cursor: initial;
|
||||
}
|
@ -13,11 +13,12 @@ import useFileFormatState from '../hooks/useFileFormatState';
|
||||
import OutputFormatSelect from './OutputFormatSelect';
|
||||
import useUserSettings from '../hooks/useUserSettings';
|
||||
import { isMov } from '../util/streams';
|
||||
import { getOutFileExtension, getSuffixedFileName } from '../util';
|
||||
import { getOutDir, getOutFileExtension } from '../util';
|
||||
import { FFprobeChapter, FFprobeFormat, FFprobeStream } from '../../../../ffprobe';
|
||||
import Sheet from './Sheet';
|
||||
import TextInput from './TextInput';
|
||||
import Button from './Button';
|
||||
import { defaultMergedFileTemplate, generateMergedFileNames, maxFileNameLength } from '../util/outputNameTemplate';
|
||||
|
||||
const { basename } = window.require('path');
|
||||
|
||||
@ -36,7 +37,7 @@ function ConcatDialog({ isShown, onHide, paths, onConcat, alwaysConcatMultipleFi
|
||||
isShown: boolean, onHide: () => void, paths: string[], onConcat: (a: { paths: string[], includeAllStreams: boolean, streams: FFprobeStream[], outFileName: string, fileFormat: string, clearBatchFilesAfterConcat: boolean }) => Promise<void>, alwaysConcatMultipleFiles: boolean, setAlwaysConcatMultipleFiles: (a: boolean) => void,
|
||||
}) {
|
||||
const { t } = useTranslation();
|
||||
const { preserveMovData, setPreserveMovData, segmentsToChapters, setSegmentsToChapters, preserveMetadataOnMerge, setPreserveMetadataOnMerge } = useUserSettings();
|
||||
const { preserveMovData, setPreserveMovData, segmentsToChapters, setSegmentsToChapters, preserveMetadataOnMerge, setPreserveMetadataOnMerge, safeOutputFileName, customOutDir } = useUserSettings();
|
||||
|
||||
const [includeAllStreams, setIncludeAllStreams] = useState(false);
|
||||
const [fileMeta, setFileMeta] = useState<{ format: FFprobeFormat, streams: FFprobeStream[], chapters: FFprobeChapter[] }>();
|
||||
@ -80,16 +81,32 @@ function ConcatDialog({ isShown, onHide, paths, onConcat, alwaysConcatMultipleFi
|
||||
}, [firstPath, isShown, setDetectedFileFormat, setFileFormat]);
|
||||
|
||||
useEffect(() => {
|
||||
if (fileFormat == null || firstPath == null) {
|
||||
if (fileFormat == null || firstPath == null || uniqueSuffix == null) {
|
||||
setOutFileName(undefined);
|
||||
return;
|
||||
}
|
||||
const ext = getOutFileExtension({ isCustomFormatSelected, outFormat: fileFormat, filePath: firstPath });
|
||||
const outputDir = getOutDir(customOutDir, firstPath);
|
||||
|
||||
setOutFileName((existingOutputName) => {
|
||||
if (existingOutputName == null) return getSuffixedFileName(firstPath, `merged-${uniqueSuffix}${ext}`);
|
||||
return existingOutputName.replace(/(\.[^.]*)?$/, ext); // make sure the last (optional) .* is replaced by .ext`
|
||||
// here we only generate the file name the first time. Then the user can edit it manually as they please in the text input field.
|
||||
// todo allow user to edit template instead of this "hack"
|
||||
if (existingOutputName == null) {
|
||||
(async () => {
|
||||
const generated = await generateMergedFileNames({ template: defaultMergedFileTemplate, isCustomFormatSelected, fileFormat, filePath: firstPath, outputDir, safeOutputFileName, epochMs: uniqueSuffix });
|
||||
// todo show to user more errors?
|
||||
const [fileName] = generated.fileNames;
|
||||
invariant(fileName != null);
|
||||
setOutFileName(fileName);
|
||||
})();
|
||||
return existingOutputName; // async later (above)
|
||||
}
|
||||
|
||||
// in case the user has chosen a different output format:
|
||||
// make sure the last (optional) .* is replaced by .ext`
|
||||
return existingOutputName.replace(/(\.[^.]*)?$/, ext);
|
||||
});
|
||||
}, [fileFormat, firstPath, isCustomFormatSelected, uniqueSuffix]);
|
||||
}, [customOutDir, fileFormat, firstPath, isCustomFormatSelected, safeOutputFileName, uniqueSuffix]);
|
||||
|
||||
const allFilesMeta = useMemo(() => {
|
||||
if (paths.length === 0) return undefined;
|
||||
@ -97,7 +114,8 @@ function ConcatDialog({ isShown, onHide, paths, onConcat, alwaysConcatMultipleFi
|
||||
return filtered.length === paths.length ? filtered : undefined;
|
||||
}, [allFilesMetaCache, paths]);
|
||||
|
||||
const isOutFileNameValid = outFileName != null && outFileName.length > 0;
|
||||
const isOutFileNameTooLong = outFileName != null && outFileName.length > maxFileNameLength;
|
||||
const isOutFileNameValid = outFileName != null && outFileName.length > 0 && !isOutFileNameTooLong;
|
||||
|
||||
const problemsByFile = useMemo(() => {
|
||||
if (!allFilesMeta) return {};
|
||||
@ -209,9 +227,15 @@ function ConcatDialog({ isShown, onHide, paths, onConcat, alwaysConcatMultipleFi
|
||||
<div style={{ display: 'flex', alignItems: 'center', flexWrap: 'wrap', justifyContent: 'flex-end', marginBottom: '1em' }}>
|
||||
<div style={{ marginRight: '.5em' }}>{t('Output file name')}:</div>
|
||||
<TextInput value={outFileName || ''} onChange={(e) => setOutFileName(e.target.value)} />
|
||||
<Button disabled={detectedFileFormat == null || !isOutFileNameValid} onClick={onConcatClick} style={{ fontSize: '1.3em', padding: '0 .3em', marginLeft: '1em' }}><AiOutlineMergeCells style={{ fontSize: '1.4em', verticalAlign: 'middle' }} /> {t('Merge!')}</Button>
|
||||
<Button disabled={detectedFileFormat == null || !isOutFileNameValid} onClick={onConcatClick} style={{ fontSize: '1.3em', padding: '0 .3em', marginLeft: '1em' }}>
|
||||
<AiOutlineMergeCells style={{ fontSize: '1.4em', verticalAlign: 'middle' }} /> {t('Merge!')}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
{isOutFileNameTooLong && (
|
||||
<Alert text={t('File name is too long and cannot be exported.')} />
|
||||
)}
|
||||
|
||||
{enableReadFileMeta && (!allFilesMeta || Object.values(problemsByFile).length > 0) && (
|
||||
<Alert text={t('A mismatch was detected in at least one file. You may proceed, but the resulting file might not be playable.')} />
|
||||
)}
|
||||
|
@ -15,6 +15,9 @@ export const segSuffixVariable = 'SEG_SUFFIX';
|
||||
export const extVariable = 'EXT';
|
||||
export const segTagsVariable = 'SEG_TAGS';
|
||||
|
||||
// I don't remember why I set it to 200, but on Windows max length seems to be 256, on MacOS it seems to be 255.
|
||||
export const maxFileNameLength = 200;
|
||||
|
||||
const { parse: parsePath, sep: pathSep, join: pathJoin, normalize: pathNormalize, basename }: PlatformPath = window.require('path');
|
||||
|
||||
|
||||
@ -104,7 +107,9 @@ function getTemplateProblems({ fileNames, filePath, outputDir, safeOutputFileNam
|
||||
// eslint-disable-next-line no-template-curly-in-string
|
||||
export const defaultOutSegTemplate = '${FILENAME}-${CUT_FROM}-${CUT_TO}${SEG_SUFFIX}${EXT}';
|
||||
// eslint-disable-next-line no-template-curly-in-string
|
||||
export const defaultMergedFileTemplate = '${FILENAME}-cut-merged-${EPOCH_MS}${EXT}';
|
||||
export const defaultCutMergedFileTemplate = '${FILENAME}-cut-merged-${EPOCH_MS}${EXT}';
|
||||
// eslint-disable-next-line no-template-curly-in-string
|
||||
export const defaultMergedFileTemplate = '${FILENAME}-merged-${EPOCH_MS}${EXT}';
|
||||
|
||||
async function interpolateOutFileName(template: string, { epochMs, inputFileNameWithoutExt, ext, segSuffix, segNum, segNumPadded, segLabel, cutFrom, cutTo, tags }: {
|
||||
epochMs: number,
|
||||
@ -151,7 +156,7 @@ function maybeTruncatePath(fileName: string, truncate: boolean) {
|
||||
return [
|
||||
...rest,
|
||||
// If sanitation is enabled, make sure filename (last seg of the path) is not too long
|
||||
truncate ? lastSeg!.slice(0, 200) : lastSeg,
|
||||
truncate ? lastSeg!.slice(0, maxFileNameLength) : lastSeg,
|
||||
].join(pathSep);
|
||||
}
|
||||
|
||||
@ -223,17 +228,16 @@ export type GenerateOutFileNames = (a: { template: string }) => Promise<{
|
||||
},
|
||||
}>;
|
||||
|
||||
export async function generateMergedFileNames({ template: desiredTemplate, isCustomFormatSelected, fileFormat, filePath, outputDir, safeOutputFileName }: {
|
||||
export async function generateMergedFileNames({ template: desiredTemplate, isCustomFormatSelected, fileFormat, filePath, outputDir, safeOutputFileName, epochMs = Date.now() }: {
|
||||
template: string,
|
||||
isCustomFormatSelected: boolean,
|
||||
fileFormat: string,
|
||||
filePath: string,
|
||||
outputDir: string,
|
||||
safeOutputFileName: boolean,
|
||||
epochMs?: number,
|
||||
}) {
|
||||
async function generate(template: string) {
|
||||
const epochMs = Date.now();
|
||||
|
||||
const { name: inputFileNameWithoutExt } = parsePath(filePath);
|
||||
|
||||
const fileName = await interpolateOutFileName(template, {
|
||||
@ -249,7 +253,7 @@ export async function generateMergedFileNames({ template: desiredTemplate, isCus
|
||||
|
||||
const problems = getTemplateProblems({ fileNames: [fileName], filePath, outputDir, safeOutputFileName });
|
||||
if (problems.error != null) {
|
||||
fileName = await generate(defaultMergedFileTemplate);
|
||||
fileName = await generate(defaultCutMergedFileTemplate);
|
||||
}
|
||||
|
||||
return { fileNames: [fileName], problems };
|
||||
|
Loading…
Reference in New Issue
Block a user