The live-status batch poller (60s cadence, every streamer in the watch list) was sending two things on every tick: - `changes` — the diff vs. the previous tick, used by the renderer - `snapshot` — the full Map<login, boolean> serialized as a record Renderer destructures only `changes` (renderer-streamers.ts line 20). The snapshot field was wire-noise. For a typical 30-50 streamer watch list, that snapshot is ~1.5KB of JSON every minute, never read on the other side. Dropped from the broadcast payload. Initial-state sync still works: the renderer's initLiveStatusSubscription calls window.api.getLiveStatusSnapshot() once at boot to pre-fill its map. The broadcast is only for diffs. Also added a short-circuit on the main side: if changes.length === 0 (every streamer's live status matched the cached value this tick), don't broadcast at all. The renderer would just iterate an empty array and trigger a no-op render; saves the wakeup entirely. Type signature updates ride through preload.ts + renderer-globals.d.ts so the API contract stays accurate. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
183 lines
9.0 KiB
TypeScript
183 lines
9.0 KiB
TypeScript
import { contextBridge, ipcRenderer } from 'electron';
|
|
import { CustomClip, MergeGroupItem, MergeGroup, QueueItem, DownloadProgress } from './types';
|
|
|
|
// Types
|
|
interface RuntimeMetricsSnapshot {
|
|
cacheHits: number;
|
|
cacheMisses: number;
|
|
duplicateSkips: number;
|
|
retriesScheduled: number;
|
|
retriesExhausted: number;
|
|
integrityFailures: number;
|
|
downloadsStarted: number;
|
|
downloadsCompleted: number;
|
|
downloadsFailed: number;
|
|
downloadedBytesTotal: number;
|
|
lastSpeedBytesPerSec: number;
|
|
avgSpeedBytesPerSec: number;
|
|
activeItemId: string | null;
|
|
activeItemTitle: string | null;
|
|
lastErrorClass: string | null;
|
|
lastRetryDelaySeconds: number;
|
|
timestamp: string;
|
|
queue: {
|
|
pending: number;
|
|
downloading: number;
|
|
paused: number;
|
|
completed: number;
|
|
error: number;
|
|
total: number;
|
|
};
|
|
caches: {
|
|
loginToUserId: number;
|
|
vodList: number;
|
|
clipInfo: number;
|
|
};
|
|
config: {
|
|
performanceMode: 'stability' | 'balanced' | 'speed';
|
|
smartScheduler: boolean;
|
|
metadataCacheMinutes: number;
|
|
duplicatePrevention: boolean;
|
|
};
|
|
}
|
|
|
|
interface VideoInfo {
|
|
duration: number;
|
|
width: number;
|
|
height: number;
|
|
fps: number;
|
|
}
|
|
|
|
// Expose protected methods to renderer
|
|
contextBridge.exposeInMainWorld('api', {
|
|
// Config
|
|
getConfig: () => ipcRenderer.invoke('get-config'),
|
|
saveConfig: (config: any) => ipcRenderer.invoke('save-config', config),
|
|
|
|
// Auth
|
|
login: () => ipcRenderer.invoke('login'),
|
|
|
|
// Twitch API
|
|
getUserId: (username: string) => ipcRenderer.invoke('get-user-id', username),
|
|
getVODs: (userId: string, forceRefresh: boolean = false) => ipcRenderer.invoke('get-vods', userId, forceRefresh),
|
|
|
|
// Queue
|
|
getQueue: () => ipcRenderer.invoke('get-queue'),
|
|
addToQueue: (item: Omit<QueueItem, 'id' | 'status' | 'progress'>) => ipcRenderer.invoke('add-to-queue', item),
|
|
startLiveRecording: (streamerName: string) => ipcRenderer.invoke('start-live-recording', streamerName),
|
|
removeFromQueue: (id: string) => ipcRenderer.invoke('remove-from-queue', id),
|
|
reorderQueue: (orderIds: string[]) => ipcRenderer.invoke('reorder-queue', orderIds),
|
|
clearCompleted: () => ipcRenderer.invoke('clear-completed'),
|
|
retryFailedDownloads: () => ipcRenderer.invoke('retry-failed-downloads'),
|
|
retryQueueItem: (id: string) => ipcRenderer.invoke('retry-queue-item', id),
|
|
createMergeGroup: (itemIds: string[]) => ipcRenderer.invoke('create-merge-group', itemIds),
|
|
|
|
// Download
|
|
startDownload: () => ipcRenderer.invoke('start-download'),
|
|
pauseDownload: () => ipcRenderer.invoke('pause-download'),
|
|
cancelDownload: () => ipcRenderer.invoke('cancel-download'),
|
|
isDownloading: () => ipcRenderer.invoke('is-downloading'),
|
|
downloadClip: (url: string) => ipcRenderer.invoke('download-clip', url),
|
|
|
|
// Files
|
|
selectFolder: () => ipcRenderer.invoke('select-folder'),
|
|
selectVideoFile: () => ipcRenderer.invoke('select-video-file'),
|
|
selectMultipleVideos: () => ipcRenderer.invoke('select-multiple-videos'),
|
|
saveVideoDialog: (defaultName: string) => ipcRenderer.invoke('save-video-dialog', defaultName),
|
|
openFolder: (path: string) => ipcRenderer.invoke('open-folder', path),
|
|
openFile: (path: string) => ipcRenderer.invoke('open-file', path),
|
|
showInFolder: (path: string) => ipcRenderer.invoke('show-in-folder', path),
|
|
openDebugLogFile: () => ipcRenderer.invoke('open-debug-log-file'),
|
|
checkFolderWritable: (path: string) => ipcRenderer.invoke('check-folder-writable', path),
|
|
getStorageStats: () => ipcRenderer.invoke('get-storage-stats'),
|
|
getArchiveStats: () => ipcRenderer.invoke('get-archive-stats'),
|
|
getStreamerProfile: (login: string, forceRefresh?: boolean) => ipcRenderer.invoke('get-streamer-profile', login, forceRefresh),
|
|
getVodStoryboard: (vodId: string) => ipcRenderer.invoke('get-vod-storyboard', vodId),
|
|
getLiveStatusSnapshot: () => ipcRenderer.invoke('get-live-status-snapshot'),
|
|
onLiveStatusBatchUpdate: (callback: (info: { changes: Array<{ login: string; isLive: boolean }> }) => void) => {
|
|
ipcRenderer.on('live-status-batch-update', (_, info) => callback(info));
|
|
},
|
|
searchArchive: (filter: Record<string, unknown>) => ipcRenderer.invoke('search-archive', filter),
|
|
runStorageCleanup: (options?: { dryRun?: boolean }) => ipcRenderer.invoke('run-storage-cleanup', options),
|
|
readChatFile: (filePath: string) => ipcRenderer.invoke('read-chat-file', filePath),
|
|
getAutomationStatus: () => ipcRenderer.invoke('get-automation-status'),
|
|
triggerAutoVodScan: () => ipcRenderer.invoke('trigger-auto-vod-scan'),
|
|
triggerAutoRecordScan: () => ipcRenderer.invoke('trigger-auto-record-scan'),
|
|
onAutoVodScanCompleted: (callback: (info: { queuedCount: number }) => void) => {
|
|
ipcRenderer.on('auto-vod-scan-completed', (_, info) => callback(info));
|
|
},
|
|
|
|
// Video Cutter
|
|
getVideoInfo: (filePath: string): Promise<VideoInfo | null> => ipcRenderer.invoke('get-video-info', filePath),
|
|
extractFrame: (filePath: string, timeSeconds: number): Promise<string | null> => ipcRenderer.invoke('extract-frame', filePath, timeSeconds),
|
|
cutVideo: (inputFile: string, startTime: number, endTime: number): Promise<{ success: boolean; outputFile: string | null }> =>
|
|
ipcRenderer.invoke('cut-video', inputFile, startTime, endTime),
|
|
|
|
// Merge Videos
|
|
mergeVideos: (inputFiles: string[], outputFile: string): Promise<{ success: boolean; outputFile: string | null }> =>
|
|
ipcRenderer.invoke('merge-videos', inputFiles, outputFile),
|
|
|
|
// App
|
|
getVersion: () => ipcRenderer.invoke('get-version'),
|
|
checkUpdate: () => ipcRenderer.invoke('check-update'),
|
|
downloadUpdate: () => ipcRenderer.invoke('download-update'),
|
|
installUpdate: () => ipcRenderer.invoke('install-update'),
|
|
openExternal: (url: string) => ipcRenderer.invoke('open-external', url),
|
|
runPreflight: (autoFix: boolean) => ipcRenderer.invoke('run-preflight', autoFix),
|
|
getDebugLog: (lines: number) => ipcRenderer.invoke('get-debug-log', lines),
|
|
getRuntimeMetrics: (): Promise<RuntimeMetricsSnapshot> => ipcRenderer.invoke('get-runtime-metrics'),
|
|
exportRuntimeMetrics: (): Promise<{ success: boolean; cancelled?: boolean; error?: string; filePath?: string }> =>
|
|
ipcRenderer.invoke('export-runtime-metrics'),
|
|
resetDownloadedVodIds: (): Promise<{ success: boolean; removedCount: number }> =>
|
|
ipcRenderer.invoke('reset-downloaded-vod-ids'),
|
|
markVodDownloaded: (vodId: string, mark: boolean): Promise<{ success: boolean }> =>
|
|
ipcRenderer.invoke('mark-vod-downloaded', vodId, mark),
|
|
exportConfig: (): Promise<{ success: boolean; cancelled?: boolean; error?: string; filePath?: string }> =>
|
|
ipcRenderer.invoke('export-config'),
|
|
importConfig: (): Promise<{ success: boolean; cancelled?: boolean; error?: string; filePath?: string }> =>
|
|
ipcRenderer.invoke('import-config'),
|
|
|
|
// Events
|
|
onDownloadProgress: (callback: (progress: DownloadProgress) => void) => {
|
|
ipcRenderer.on('download-progress', (_, progress) => callback(progress));
|
|
},
|
|
onQueueUpdated: (callback: (queue: QueueItem[]) => void) => {
|
|
ipcRenderer.on('queue-updated', (_, queue) => callback(queue));
|
|
},
|
|
onQueueDuplicateSkipped: (callback: (payload: { title: string; streamer: string; url: string }) => void) => {
|
|
ipcRenderer.on('queue-duplicate-skipped', (_, payload) => callback(payload));
|
|
},
|
|
onDownloadStarted: (callback: () => void) => {
|
|
ipcRenderer.on('download-started', () => callback());
|
|
},
|
|
onDownloadFinished: (callback: () => void) => {
|
|
ipcRenderer.on('download-finished', () => callback());
|
|
},
|
|
onCutProgress: (callback: (percent: number) => void) => {
|
|
ipcRenderer.on('cut-progress', (_, percent) => callback(percent));
|
|
},
|
|
onMergeProgress: (callback: (percent: number) => void) => {
|
|
ipcRenderer.on('merge-progress', (_, percent) => callback(percent));
|
|
},
|
|
|
|
// Auto-Update Events
|
|
onUpdateChecking: (callback: () => void) => {
|
|
ipcRenderer.on('update-checking', () => callback());
|
|
},
|
|
onUpdateAvailable: (callback: (info: { version: string; releaseDate?: string; releaseName?: string; releaseNotes?: string }) => void) => {
|
|
ipcRenderer.on('update-available', (_, info) => callback(info));
|
|
},
|
|
onUpdateNotAvailable: (callback: () => void) => {
|
|
ipcRenderer.on('update-not-available', () => callback());
|
|
},
|
|
onUpdateDownloadProgress: (callback: (progress: { percent: number; bytesPerSecond: number; transferred: number; total: number }) => void) => {
|
|
ipcRenderer.on('update-download-progress', (_, progress) => callback(progress));
|
|
},
|
|
onUpdateDownloaded: (callback: (info: { version: string; releaseDate?: string; releaseName?: string; releaseNotes?: string }) => void) => {
|
|
ipcRenderer.on('update-downloaded', (_, info) => callback(info));
|
|
},
|
|
onUpdateError: (callback: (payload: { message: string }) => void) => {
|
|
ipcRenderer.on('update-error', (_, payload) => callback(payload));
|
|
}
|
|
});
|