mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-03-01 06:22:44 -08:00
refactor: extract startup lifecycle runtime deps
This commit is contained in:
@@ -16,7 +16,7 @@
|
|||||||
"docs:build": "vitepress build docs",
|
"docs:build": "vitepress build docs",
|
||||||
"docs:preview": "vitepress preview docs --host 0.0.0.0 --port 4173 --strictPort",
|
"docs:preview": "vitepress preview docs --host 0.0.0.0 --port 4173 --strictPort",
|
||||||
"test:config": "pnpm run build && node --test dist/config/config.test.js",
|
"test:config": "pnpm run build && node --test dist/config/config.test.js",
|
||||||
"test:core": "pnpm run build && node --test dist/cli/args.test.js dist/cli/help.test.js dist/core/services/cli-command-service.test.js dist/core/services/cli-command-deps-runtime-service.test.js dist/core/services/ipc-deps-runtime-service.test.js dist/core/services/anki-jimaku-ipc-deps-runtime-service.test.js dist/core/services/field-grouping-overlay-runtime-service.test.js dist/core/services/subsync-deps-runtime-service.test.js dist/core/services/numeric-shortcut-runtime-service.test.js dist/core/services/numeric-shortcut-session-service.test.js dist/core/services/overlay-visibility-facade-deps-runtime-service.test.js dist/core/services/mpv-command-ipc-deps-runtime-service.test.js dist/core/services/runtime-options-ipc-deps-runtime-service.test.js dist/core/services/tokenizer-deps-runtime-service.test.js dist/core/services/overlay-runtime-deps-service.test.js dist/core/services/secondary-subtitle-service.test.js dist/core/services/mpv-render-metrics-service.test.js dist/core/services/mpv-runtime-service.test.js dist/core/services/runtime-options-runtime-service.test.js dist/core/services/overlay-modal-restore-service.test.js dist/core/services/runtime-config-service.test.js dist/core/services/overlay-bridge-runtime-service.test.js dist/core/services/overlay-visibility-facade-service.test.js dist/core/services/overlay-broadcast-runtime-service.test.js dist/core/services/app-ready-runtime-service.test.js dist/core/services/app-shutdown-runtime-service.test.js dist/core/services/mpv-client-deps-runtime-service.test.js dist/core/services/app-lifecycle-deps-runtime-service.test.js dist/core/services/runtime-options-manager-runtime-service.test.js dist/core/services/config-warning-runtime-service.test.js dist/core/services/app-logging-runtime-service.test.js dist/core/services/startup-resource-runtime-service.test.js dist/core/services/config-generation-runtime-service.test.js",
|
"test:core": "pnpm run build && node --test dist/cli/args.test.js dist/cli/help.test.js dist/core/services/cli-command-service.test.js dist/core/services/cli-command-deps-runtime-service.test.js dist/core/services/ipc-deps-runtime-service.test.js dist/core/services/anki-jimaku-ipc-deps-runtime-service.test.js dist/core/services/field-grouping-overlay-runtime-service.test.js dist/core/services/subsync-deps-runtime-service.test.js dist/core/services/numeric-shortcut-runtime-service.test.js dist/core/services/numeric-shortcut-session-service.test.js dist/core/services/overlay-visibility-facade-deps-runtime-service.test.js dist/core/services/mpv-command-ipc-deps-runtime-service.test.js dist/core/services/runtime-options-ipc-deps-runtime-service.test.js dist/core/services/tokenizer-deps-runtime-service.test.js dist/core/services/overlay-runtime-deps-service.test.js dist/core/services/startup-lifecycle-runtime-deps-service.test.js dist/core/services/secondary-subtitle-service.test.js dist/core/services/mpv-render-metrics-service.test.js dist/core/services/mpv-runtime-service.test.js dist/core/services/runtime-options-runtime-service.test.js dist/core/services/overlay-modal-restore-service.test.js dist/core/services/runtime-config-service.test.js dist/core/services/overlay-bridge-runtime-service.test.js dist/core/services/overlay-visibility-facade-service.test.js dist/core/services/overlay-broadcast-runtime-service.test.js dist/core/services/app-ready-runtime-service.test.js dist/core/services/app-shutdown-runtime-service.test.js dist/core/services/mpv-client-deps-runtime-service.test.js dist/core/services/app-lifecycle-deps-runtime-service.test.js dist/core/services/runtime-options-manager-runtime-service.test.js dist/core/services/config-warning-runtime-service.test.js dist/core/services/app-logging-runtime-service.test.js dist/core/services/startup-resource-runtime-service.test.js dist/core/services/config-generation-runtime-service.test.js",
|
||||||
"test:subtitle": "pnpm run build && node --test dist/subtitle/stages.test.js dist/subtitle/pipeline.test.js",
|
"test:subtitle": "pnpm run build && node --test dist/subtitle/stages.test.js dist/subtitle/pipeline.test.js",
|
||||||
"generate:config-example": "pnpm run build && node dist/generate-config-example.js",
|
"generate:config-example": "pnpm run build && node dist/generate-config-example.js",
|
||||||
"start": "pnpm run build && electron . --start",
|
"start": "pnpm run build && electron . --start",
|
||||||
|
|||||||
@@ -0,0 +1,74 @@
|
|||||||
|
import test from "node:test";
|
||||||
|
import assert from "node:assert/strict";
|
||||||
|
import {
|
||||||
|
createStartupAppReadyDepsRuntimeService,
|
||||||
|
createStartupAppShutdownDepsRuntimeService,
|
||||||
|
} from "./startup-lifecycle-runtime-deps-service";
|
||||||
|
|
||||||
|
test("createStartupAppReadyDepsRuntimeService preserves runtime deps behavior", async () => {
|
||||||
|
const calls: string[] = [];
|
||||||
|
const deps = createStartupAppReadyDepsRuntimeService({
|
||||||
|
loadSubtitlePosition: () => calls.push("loadSubtitlePosition"),
|
||||||
|
resolveKeybindings: () => calls.push("resolveKeybindings"),
|
||||||
|
createMpvClient: () => calls.push("createMpvClient"),
|
||||||
|
reloadConfig: () => calls.push("reloadConfig"),
|
||||||
|
getResolvedConfig: () => ({
|
||||||
|
secondarySub: { defaultMode: "hover" },
|
||||||
|
websocket: { enabled: "auto", port: 1234 },
|
||||||
|
}),
|
||||||
|
getConfigWarnings: () => [],
|
||||||
|
logConfigWarning: () => {},
|
||||||
|
initRuntimeOptionsManager: () => calls.push("initRuntimeOptionsManager"),
|
||||||
|
setSecondarySubMode: () => calls.push("setSecondarySubMode"),
|
||||||
|
defaultSecondarySubMode: "hover",
|
||||||
|
defaultWebsocketPort: 8765,
|
||||||
|
hasMpvWebsocketPlugin: () => true,
|
||||||
|
startSubtitleWebsocket: () => calls.push("startSubtitleWebsocket"),
|
||||||
|
log: () => calls.push("log"),
|
||||||
|
createMecabTokenizerAndCheck: async () => {
|
||||||
|
calls.push("createMecab");
|
||||||
|
},
|
||||||
|
createSubtitleTimingTracker: () => calls.push("createSubtitleTimingTracker"),
|
||||||
|
loadYomitanExtension: async () => {
|
||||||
|
calls.push("loadYomitan");
|
||||||
|
},
|
||||||
|
texthookerOnlyMode: false,
|
||||||
|
shouldAutoInitializeOverlayRuntimeFromConfig: () => false,
|
||||||
|
initializeOverlayRuntime: () => calls.push("initOverlayRuntime"),
|
||||||
|
handleInitialArgs: () => calls.push("handleInitialArgs"),
|
||||||
|
});
|
||||||
|
|
||||||
|
deps.loadSubtitlePosition();
|
||||||
|
await deps.createMecabTokenizerAndCheck();
|
||||||
|
deps.handleInitialArgs();
|
||||||
|
|
||||||
|
assert.equal(deps.defaultWebsocketPort, 8765);
|
||||||
|
assert.equal(deps.defaultSecondarySubMode, "hover");
|
||||||
|
assert.deepEqual(calls, ["loadSubtitlePosition", "createMecab", "handleInitialArgs"]);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("createStartupAppShutdownDepsRuntimeService preserves shutdown handlers", () => {
|
||||||
|
const calls: string[] = [];
|
||||||
|
const deps = createStartupAppShutdownDepsRuntimeService({
|
||||||
|
unregisterAllGlobalShortcuts: () => calls.push("unregisterAllGlobalShortcuts"),
|
||||||
|
stopSubtitleWebsocket: () => calls.push("stopSubtitleWebsocket"),
|
||||||
|
stopTexthookerService: () => calls.push("stopTexthookerService"),
|
||||||
|
destroyYomitanParserWindow: () => calls.push("destroyYomitanParserWindow"),
|
||||||
|
clearYomitanParserPromises: () => calls.push("clearYomitanParserPromises"),
|
||||||
|
stopWindowTracker: () => calls.push("stopWindowTracker"),
|
||||||
|
destroyMpvSocket: () => calls.push("destroyMpvSocket"),
|
||||||
|
clearReconnectTimer: () => calls.push("clearReconnectTimer"),
|
||||||
|
destroySubtitleTimingTracker: () => calls.push("destroySubtitleTimingTracker"),
|
||||||
|
destroyAnkiIntegration: () => calls.push("destroyAnkiIntegration"),
|
||||||
|
});
|
||||||
|
|
||||||
|
deps.stopSubtitleWebsocket();
|
||||||
|
deps.clearReconnectTimer();
|
||||||
|
deps.destroyAnkiIntegration();
|
||||||
|
|
||||||
|
assert.deepEqual(calls, [
|
||||||
|
"stopSubtitleWebsocket",
|
||||||
|
"clearReconnectTimer",
|
||||||
|
"destroyAnkiIntegration",
|
||||||
|
]);
|
||||||
|
});
|
||||||
55
src/core/services/startup-lifecycle-runtime-deps-service.ts
Normal file
55
src/core/services/startup-lifecycle-runtime-deps-service.ts
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import {
|
||||||
|
AppReadyRuntimeDeps,
|
||||||
|
} from "./app-ready-runtime-service";
|
||||||
|
import {
|
||||||
|
AppShutdownRuntimeDeps,
|
||||||
|
} from "./app-shutdown-runtime-service";
|
||||||
|
|
||||||
|
export type StartupAppReadyDepsRuntimeOptions = AppReadyRuntimeDeps;
|
||||||
|
export type StartupAppShutdownDepsRuntimeOptions = AppShutdownRuntimeDeps;
|
||||||
|
|
||||||
|
export function createStartupAppReadyDepsRuntimeService(
|
||||||
|
options: StartupAppReadyDepsRuntimeOptions,
|
||||||
|
): AppReadyRuntimeDeps {
|
||||||
|
return {
|
||||||
|
loadSubtitlePosition: options.loadSubtitlePosition,
|
||||||
|
resolveKeybindings: options.resolveKeybindings,
|
||||||
|
createMpvClient: options.createMpvClient,
|
||||||
|
reloadConfig: options.reloadConfig,
|
||||||
|
getResolvedConfig: options.getResolvedConfig,
|
||||||
|
getConfigWarnings: options.getConfigWarnings,
|
||||||
|
logConfigWarning: options.logConfigWarning,
|
||||||
|
initRuntimeOptionsManager: options.initRuntimeOptionsManager,
|
||||||
|
setSecondarySubMode: options.setSecondarySubMode,
|
||||||
|
defaultSecondarySubMode: options.defaultSecondarySubMode,
|
||||||
|
defaultWebsocketPort: options.defaultWebsocketPort,
|
||||||
|
hasMpvWebsocketPlugin: options.hasMpvWebsocketPlugin,
|
||||||
|
startSubtitleWebsocket: options.startSubtitleWebsocket,
|
||||||
|
log: options.log,
|
||||||
|
createMecabTokenizerAndCheck: options.createMecabTokenizerAndCheck,
|
||||||
|
createSubtitleTimingTracker: options.createSubtitleTimingTracker,
|
||||||
|
loadYomitanExtension: options.loadYomitanExtension,
|
||||||
|
texthookerOnlyMode: options.texthookerOnlyMode,
|
||||||
|
shouldAutoInitializeOverlayRuntimeFromConfig:
|
||||||
|
options.shouldAutoInitializeOverlayRuntimeFromConfig,
|
||||||
|
initializeOverlayRuntime: options.initializeOverlayRuntime,
|
||||||
|
handleInitialArgs: options.handleInitialArgs,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createStartupAppShutdownDepsRuntimeService(
|
||||||
|
options: StartupAppShutdownDepsRuntimeOptions,
|
||||||
|
): AppShutdownRuntimeDeps {
|
||||||
|
return {
|
||||||
|
unregisterAllGlobalShortcuts: options.unregisterAllGlobalShortcuts,
|
||||||
|
stopSubtitleWebsocket: options.stopSubtitleWebsocket,
|
||||||
|
stopTexthookerService: options.stopTexthookerService,
|
||||||
|
destroyYomitanParserWindow: options.destroyYomitanParserWindow,
|
||||||
|
clearYomitanParserPromises: options.clearYomitanParserPromises,
|
||||||
|
stopWindowTracker: options.stopWindowTracker,
|
||||||
|
destroyMpvSocket: options.destroyMpvSocket,
|
||||||
|
clearReconnectTimer: options.clearReconnectTimer,
|
||||||
|
destroySubtitleTimingTracker: options.destroySubtitleTimingTracker,
|
||||||
|
destroyAnkiIntegration: options.destroyAnkiIntegration,
|
||||||
|
};
|
||||||
|
}
|
||||||
16
src/main.ts
16
src/main.ts
@@ -207,6 +207,10 @@ import {
|
|||||||
createOverlayWindowRuntimeDepsService,
|
createOverlayWindowRuntimeDepsService,
|
||||||
createVisibleOverlayVisibilityDepsRuntimeService,
|
createVisibleOverlayVisibilityDepsRuntimeService,
|
||||||
} from "./core/services/overlay-runtime-deps-service";
|
} from "./core/services/overlay-runtime-deps-service";
|
||||||
|
import {
|
||||||
|
createStartupAppReadyDepsRuntimeService,
|
||||||
|
createStartupAppShutdownDepsRuntimeService,
|
||||||
|
} from "./core/services/startup-lifecycle-runtime-deps-service";
|
||||||
import { createRuntimeOptionsManagerRuntimeService } from "./core/services/runtime-options-manager-runtime-service";
|
import { createRuntimeOptionsManagerRuntimeService } from "./core/services/runtime-options-manager-runtime-service";
|
||||||
import { createAppLoggingRuntimeService } from "./core/services/app-logging-runtime-service";
|
import { createAppLoggingRuntimeService } from "./core/services/app-logging-runtime-service";
|
||||||
import {
|
import {
|
||||||
@@ -502,7 +506,8 @@ if (
|
|||||||
printHelp: () => printHelp(DEFAULT_TEXTHOOKER_PORT),
|
printHelp: () => printHelp(DEFAULT_TEXTHOOKER_PORT),
|
||||||
logNoRunningInstance: () => appLogger.logNoRunningInstance(),
|
logNoRunningInstance: () => appLogger.logNoRunningInstance(),
|
||||||
onReady: async () => {
|
onReady: async () => {
|
||||||
await runAppReadyRuntimeService({
|
await runAppReadyRuntimeService(
|
||||||
|
createStartupAppReadyDepsRuntimeService({
|
||||||
loadSubtitlePosition: () => loadSubtitlePosition(),
|
loadSubtitlePosition: () => loadSubtitlePosition(),
|
||||||
resolveKeybindings: () => {
|
resolveKeybindings: () => {
|
||||||
keybindings = resolveKeybindings(getResolvedConfig(), DEFAULT_KEYBINDINGS);
|
keybindings = resolveKeybindings(getResolvedConfig(), DEFAULT_KEYBINDINGS);
|
||||||
@@ -605,10 +610,12 @@ if (
|
|||||||
shouldAutoInitializeOverlayRuntimeFromConfig(),
|
shouldAutoInitializeOverlayRuntimeFromConfig(),
|
||||||
initializeOverlayRuntime: () => initializeOverlayRuntime(),
|
initializeOverlayRuntime: () => initializeOverlayRuntime(),
|
||||||
handleInitialArgs: () => handleInitialArgs(),
|
handleInitialArgs: () => handleInitialArgs(),
|
||||||
});
|
}),
|
||||||
|
);
|
||||||
},
|
},
|
||||||
onWillQuitCleanup: () => {
|
onWillQuitCleanup: () => {
|
||||||
runAppShutdownRuntimeService({
|
runAppShutdownRuntimeService(
|
||||||
|
createStartupAppShutdownDepsRuntimeService({
|
||||||
unregisterAllGlobalShortcuts: () => {
|
unregisterAllGlobalShortcuts: () => {
|
||||||
globalShortcut.unregisterAll();
|
globalShortcut.unregisterAll();
|
||||||
},
|
},
|
||||||
@@ -653,7 +660,8 @@ if (
|
|||||||
ankiIntegration.destroy();
|
ankiIntegration.destroy();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
});
|
}),
|
||||||
|
);
|
||||||
},
|
},
|
||||||
shouldRestoreWindowsOnActivate: () =>
|
shouldRestoreWindowsOnActivate: () =>
|
||||||
overlayRuntimeInitialized && BrowserWindow.getAllWindows().length === 0,
|
overlayRuntimeInitialized && BrowserWindow.getAllWindows().length === 0,
|
||||||
|
|||||||
Reference in New Issue
Block a user