CutScript is a local-first, Descript-like video editor where you edit video by editing text. Delete a word from the transcript and it's cut from the video. Features: - Word-level transcription with WhisperX - Text-based video editing with undo/redo - AI filler word removal (Ollama/OpenAI/Claude) - AI clip creation for shorts - Waveform timeline with virtualized transcript - FFmpeg stream-copy (fast) and re-encode (4K) export - Caption burn-in and sidecar SRT generation - Studio Sound audio enhancement (DeepFilterNet) - Keyboard shortcuts (J/K/L, Space, Delete, Ctrl+Z/S/E) - Encrypted API key storage - Project save/load (.aive files) Architecture: - Electron + React + Tailwind (frontend) - FastAPI + Python (backend) - WhisperX for transcription - FFmpeg for video processing - Multi-provider AI support Performance optimizations: - RAF-throttled time updates - Zustand selectors for granular subscriptions - Dual-canvas waveform rendering - Virtualized transcript with react-virtuoso Built on top of DataAnts-AI/VideoTranscriber, completely rewritten as a desktop application. License: MIT
132 lines
3.3 KiB
JavaScript
132 lines
3.3 KiB
JavaScript
const { app, BrowserWindow, ipcMain, dialog, safeStorage } = require('electron');
|
|
const path = require('path');
|
|
const { PythonBackend } = require('./python-bridge');
|
|
|
|
let mainWindow = null;
|
|
let pythonBackend = null;
|
|
|
|
const isDev = !app.isPackaged;
|
|
const BACKEND_PORT = 8642;
|
|
|
|
function createWindow() {
|
|
mainWindow = new BrowserWindow({
|
|
width: 1400,
|
|
height: 900,
|
|
minWidth: 1024,
|
|
minHeight: 700,
|
|
title: 'CutScript',
|
|
webPreferences: {
|
|
preload: path.join(__dirname, 'preload.js'),
|
|
contextIsolation: true,
|
|
nodeIntegration: false,
|
|
webSecurity: isDev ? false : true,
|
|
},
|
|
show: false,
|
|
});
|
|
|
|
if (isDev) {
|
|
mainWindow.loadURL('http://localhost:5173');
|
|
mainWindow.webContents.openDevTools();
|
|
} else {
|
|
mainWindow.loadFile(path.join(__dirname, '..', 'frontend', 'dist', 'index.html'));
|
|
}
|
|
|
|
mainWindow.once('ready-to-show', () => {
|
|
mainWindow.show();
|
|
});
|
|
|
|
mainWindow.on('closed', () => {
|
|
mainWindow = null;
|
|
});
|
|
}
|
|
|
|
app.whenReady().then(async () => {
|
|
pythonBackend = new PythonBackend(BACKEND_PORT, isDev);
|
|
await pythonBackend.start();
|
|
|
|
createWindow();
|
|
|
|
app.on('activate', () => {
|
|
if (BrowserWindow.getAllWindows().length === 0) {
|
|
createWindow();
|
|
}
|
|
});
|
|
});
|
|
|
|
app.on('window-all-closed', () => {
|
|
if (process.platform !== 'darwin') {
|
|
app.quit();
|
|
}
|
|
});
|
|
|
|
app.on('before-quit', () => {
|
|
if (pythonBackend) {
|
|
pythonBackend.stop();
|
|
}
|
|
});
|
|
|
|
// IPC Handlers
|
|
|
|
ipcMain.handle('dialog:openFile', async (_event, options) => {
|
|
const result = await dialog.showOpenDialog(mainWindow, {
|
|
properties: ['openFile'],
|
|
filters: [
|
|
{ name: 'Video Files', extensions: ['mp4', 'avi', 'mov', 'mkv', 'webm'] },
|
|
{ name: 'Audio Files', extensions: ['m4a', 'wav', 'mp3', 'flac'] },
|
|
{ name: 'All Files', extensions: ['*'] },
|
|
],
|
|
...options,
|
|
});
|
|
return result.canceled ? null : result.filePaths[0];
|
|
});
|
|
|
|
ipcMain.handle('dialog:saveFile', async (_event, options) => {
|
|
const result = await dialog.showSaveDialog(mainWindow, {
|
|
filters: [
|
|
{ name: 'Video Files', extensions: ['mp4', 'mov', 'webm'] },
|
|
{ name: 'Project Files', extensions: ['aive'] },
|
|
],
|
|
...options,
|
|
});
|
|
return result.canceled ? null : result.filePath;
|
|
});
|
|
|
|
ipcMain.handle('dialog:openProject', async () => {
|
|
const result = await dialog.showOpenDialog(mainWindow, {
|
|
properties: ['openFile'],
|
|
filters: [
|
|
{ name: 'AI Video Editor Project', extensions: ['aive'] },
|
|
],
|
|
});
|
|
return result.canceled ? null : result.filePaths[0];
|
|
});
|
|
|
|
ipcMain.handle('safe-storage:encrypt', (_event, data) => {
|
|
if (safeStorage.isEncryptionAvailable()) {
|
|
return safeStorage.encryptString(data).toString('base64');
|
|
}
|
|
return data;
|
|
});
|
|
|
|
ipcMain.handle('safe-storage:decrypt', (_event, encrypted) => {
|
|
if (safeStorage.isEncryptionAvailable()) {
|
|
return safeStorage.decryptString(Buffer.from(encrypted, 'base64'));
|
|
}
|
|
return encrypted;
|
|
});
|
|
|
|
ipcMain.handle('get-backend-url', () => {
|
|
return `http://localhost:${BACKEND_PORT}`;
|
|
});
|
|
|
|
ipcMain.handle('fs:readFile', async (_event, filePath) => {
|
|
const fs = require('fs');
|
|
return fs.readFileSync(filePath, 'utf-8');
|
|
});
|
|
|
|
ipcMain.handle('fs:writeFile', async (_event, filePath, content) => {
|
|
const fs = require('fs');
|
|
fs.writeFileSync(filePath, content, 'utf-8');
|
|
return true;
|
|
});
|