mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-02-27 18:22:41 -08:00
refactor: extract field grouping overlay runtime service
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/numeric-shortcut-session-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/numeric-shortcut-session-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,82 @@
|
|||||||
|
import test from "node:test";
|
||||||
|
import assert from "node:assert/strict";
|
||||||
|
import { KikuFieldGroupingChoice } from "../../types";
|
||||||
|
import { createFieldGroupingOverlayRuntimeService } from "./field-grouping-overlay-runtime-service";
|
||||||
|
|
||||||
|
test("createFieldGroupingOverlayRuntimeService sends overlay messages and sets restore flag", () => {
|
||||||
|
const sent: unknown[][] = [];
|
||||||
|
let visible = false;
|
||||||
|
const restore = new Set<"runtime-options" | "subsync">();
|
||||||
|
|
||||||
|
const runtime =
|
||||||
|
createFieldGroupingOverlayRuntimeService<"runtime-options" | "subsync">({
|
||||||
|
getMainWindow: () => ({
|
||||||
|
isDestroyed: () => false,
|
||||||
|
webContents: {
|
||||||
|
send: (...args: unknown[]) => {
|
||||||
|
sent.push(args);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
getVisibleOverlayVisible: () => visible,
|
||||||
|
getInvisibleOverlayVisible: () => false,
|
||||||
|
setVisibleOverlayVisible: (next) => {
|
||||||
|
visible = next;
|
||||||
|
},
|
||||||
|
setInvisibleOverlayVisible: () => {},
|
||||||
|
getResolver: () => null,
|
||||||
|
setResolver: () => {},
|
||||||
|
getRestoreVisibleOverlayOnModalClose: () => restore,
|
||||||
|
});
|
||||||
|
|
||||||
|
const ok = runtime.sendToVisibleOverlay("runtime-options:open", undefined, {
|
||||||
|
restoreOnModalClose: "runtime-options",
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(ok, true);
|
||||||
|
assert.equal(visible, true);
|
||||||
|
assert.equal(restore.has("runtime-options"), true);
|
||||||
|
assert.deepEqual(sent, [["runtime-options:open"]]);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("createFieldGroupingOverlayRuntimeService callback cancels when send fails", async () => {
|
||||||
|
let resolver: ((choice: KikuFieldGroupingChoice) => void) | null = null;
|
||||||
|
const runtime =
|
||||||
|
createFieldGroupingOverlayRuntimeService<"runtime-options" | "subsync">({
|
||||||
|
getMainWindow: () => null,
|
||||||
|
getVisibleOverlayVisible: () => false,
|
||||||
|
getInvisibleOverlayVisible: () => false,
|
||||||
|
setVisibleOverlayVisible: () => {},
|
||||||
|
setInvisibleOverlayVisible: () => {},
|
||||||
|
getResolver: () => resolver,
|
||||||
|
setResolver: (next) => {
|
||||||
|
resolver = next;
|
||||||
|
},
|
||||||
|
getRestoreVisibleOverlayOnModalClose: () =>
|
||||||
|
new Set<"runtime-options" | "subsync">(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const callback = runtime.createFieldGroupingCallback();
|
||||||
|
const result = await callback({
|
||||||
|
original: {
|
||||||
|
noteId: 1,
|
||||||
|
expression: "a",
|
||||||
|
sentencePreview: "a",
|
||||||
|
hasAudio: false,
|
||||||
|
hasImage: false,
|
||||||
|
isOriginal: true,
|
||||||
|
},
|
||||||
|
duplicate: {
|
||||||
|
noteId: 2,
|
||||||
|
expression: "b",
|
||||||
|
sentencePreview: "b",
|
||||||
|
hasAudio: false,
|
||||||
|
hasImage: false,
|
||||||
|
isOriginal: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(result.cancelled, true);
|
||||||
|
assert.equal(result.keepNoteId, 0);
|
||||||
|
assert.equal(result.deleteNoteId, 0);
|
||||||
|
});
|
||||||
77
src/core/services/field-grouping-overlay-runtime-service.ts
Normal file
77
src/core/services/field-grouping-overlay-runtime-service.ts
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
import {
|
||||||
|
KikuFieldGroupingChoice,
|
||||||
|
KikuFieldGroupingRequestData,
|
||||||
|
} from "../../types";
|
||||||
|
import {
|
||||||
|
createFieldGroupingCallbackRuntimeService,
|
||||||
|
sendToVisibleOverlayRuntimeService,
|
||||||
|
} from "./overlay-bridge-runtime-service";
|
||||||
|
|
||||||
|
interface WindowLike {
|
||||||
|
isDestroyed: () => boolean;
|
||||||
|
webContents: {
|
||||||
|
send: (channel: string, payload?: unknown) => void;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FieldGroupingOverlayRuntimeOptions<T extends string> {
|
||||||
|
getMainWindow: () => WindowLike | null;
|
||||||
|
getVisibleOverlayVisible: () => boolean;
|
||||||
|
getInvisibleOverlayVisible: () => boolean;
|
||||||
|
setVisibleOverlayVisible: (visible: boolean) => void;
|
||||||
|
setInvisibleOverlayVisible: (visible: boolean) => void;
|
||||||
|
getResolver: () => ((choice: KikuFieldGroupingChoice) => void) | null;
|
||||||
|
setResolver: (
|
||||||
|
resolver: ((choice: KikuFieldGroupingChoice) => void) | null,
|
||||||
|
) => void;
|
||||||
|
getRestoreVisibleOverlayOnModalClose: () => Set<T>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createFieldGroupingOverlayRuntimeService<T extends string>(
|
||||||
|
options: FieldGroupingOverlayRuntimeOptions<T>,
|
||||||
|
): {
|
||||||
|
sendToVisibleOverlay: (
|
||||||
|
channel: string,
|
||||||
|
payload?: unknown,
|
||||||
|
runtimeOptions?: { restoreOnModalClose?: T },
|
||||||
|
) => boolean;
|
||||||
|
createFieldGroupingCallback: () => (
|
||||||
|
data: KikuFieldGroupingRequestData,
|
||||||
|
) => Promise<KikuFieldGroupingChoice>;
|
||||||
|
} {
|
||||||
|
const sendToVisibleOverlay = (
|
||||||
|
channel: string,
|
||||||
|
payload?: unknown,
|
||||||
|
runtimeOptions?: { restoreOnModalClose?: T },
|
||||||
|
): boolean => {
|
||||||
|
return sendToVisibleOverlayRuntimeService({
|
||||||
|
mainWindow: options.getMainWindow() as never,
|
||||||
|
visibleOverlayVisible: options.getVisibleOverlayVisible(),
|
||||||
|
setVisibleOverlayVisible: options.setVisibleOverlayVisible,
|
||||||
|
channel,
|
||||||
|
payload,
|
||||||
|
restoreOnModalClose: runtimeOptions?.restoreOnModalClose,
|
||||||
|
restoreVisibleOverlayOnModalClose:
|
||||||
|
options.getRestoreVisibleOverlayOnModalClose(),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const createFieldGroupingCallback = (): ((
|
||||||
|
data: KikuFieldGroupingRequestData,
|
||||||
|
) => Promise<KikuFieldGroupingChoice>) => {
|
||||||
|
return createFieldGroupingCallbackRuntimeService({
|
||||||
|
getVisibleOverlayVisible: options.getVisibleOverlayVisible,
|
||||||
|
getInvisibleOverlayVisible: options.getInvisibleOverlayVisible,
|
||||||
|
setVisibleOverlayVisible: options.setVisibleOverlayVisible,
|
||||||
|
setInvisibleOverlayVisible: options.setInvisibleOverlayVisible,
|
||||||
|
getResolver: options.getResolver,
|
||||||
|
setResolver: options.setResolver,
|
||||||
|
sendToVisibleOverlay,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
sendToVisibleOverlay,
|
||||||
|
createFieldGroupingCallback,
|
||||||
|
};
|
||||||
|
}
|
||||||
55
src/main.ts
55
src/main.ts
@@ -191,10 +191,6 @@ import {
|
|||||||
import {
|
import {
|
||||||
handleOverlayModalClosedService,
|
handleOverlayModalClosedService,
|
||||||
} from "./core/services/overlay-modal-restore-service";
|
} from "./core/services/overlay-modal-restore-service";
|
||||||
import {
|
|
||||||
createFieldGroupingCallbackRuntimeService,
|
|
||||||
sendToVisibleOverlayRuntimeService,
|
|
||||||
} from "./core/services/overlay-bridge-runtime-service";
|
|
||||||
import {
|
import {
|
||||||
broadcastRuntimeOptionsChangedRuntimeService,
|
broadcastRuntimeOptionsChangedRuntimeService,
|
||||||
broadcastToOverlayWindowsRuntimeService,
|
broadcastToOverlayWindowsRuntimeService,
|
||||||
@@ -208,6 +204,7 @@ import { createAppLifecycleDepsRuntimeService } from "./core/services/app-lifecy
|
|||||||
import { createCliCommandDepsRuntimeService } from "./core/services/cli-command-deps-runtime-service";
|
import { createCliCommandDepsRuntimeService } from "./core/services/cli-command-deps-runtime-service";
|
||||||
import { createIpcDepsRuntimeService } from "./core/services/ipc-deps-runtime-service";
|
import { createIpcDepsRuntimeService } from "./core/services/ipc-deps-runtime-service";
|
||||||
import { createAnkiJimakuIpcDepsRuntimeService } from "./core/services/anki-jimaku-ipc-deps-runtime-service";
|
import { createAnkiJimakuIpcDepsRuntimeService } from "./core/services/anki-jimaku-ipc-deps-runtime-service";
|
||||||
|
import { createFieldGroupingOverlayRuntimeService } from "./core/services/field-grouping-overlay-runtime-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 {
|
||||||
@@ -320,6 +317,21 @@ let trackerNotReadyWarningShown = false;
|
|||||||
let overlayDebugVisualizationEnabled = false;
|
let overlayDebugVisualizationEnabled = false;
|
||||||
type OverlayHostedModal = "runtime-options" | "subsync";
|
type OverlayHostedModal = "runtime-options" | "subsync";
|
||||||
const restoreVisibleOverlayOnModalClose = new Set<OverlayHostedModal>();
|
const restoreVisibleOverlayOnModalClose = new Set<OverlayHostedModal>();
|
||||||
|
const fieldGroupingOverlayRuntime = createFieldGroupingOverlayRuntimeService<OverlayHostedModal>({
|
||||||
|
getMainWindow: () => mainWindow,
|
||||||
|
getVisibleOverlayVisible: () => visibleOverlayVisible,
|
||||||
|
getInvisibleOverlayVisible: () => invisibleOverlayVisible,
|
||||||
|
setVisibleOverlayVisible: (visible) => setVisibleOverlayVisible(visible),
|
||||||
|
setInvisibleOverlayVisible: (visible) => setInvisibleOverlayVisible(visible),
|
||||||
|
getResolver: () => fieldGroupingResolver,
|
||||||
|
setResolver: (resolver) => {
|
||||||
|
fieldGroupingResolver = resolver;
|
||||||
|
},
|
||||||
|
getRestoreVisibleOverlayOnModalClose: () => restoreVisibleOverlayOnModalClose,
|
||||||
|
});
|
||||||
|
const sendToVisibleOverlay = fieldGroupingOverlayRuntime.sendToVisibleOverlay;
|
||||||
|
const createFieldGroupingCallback =
|
||||||
|
fieldGroupingOverlayRuntime.createFieldGroupingCallback;
|
||||||
|
|
||||||
const SUBTITLE_POSITIONS_DIR = path.join(CONFIG_DIR, "subtitle-positions");
|
const SUBTITLE_POSITIONS_DIR = path.join(CONFIG_DIR, "subtitle-positions");
|
||||||
|
|
||||||
@@ -1289,41 +1301,6 @@ registerIpcHandlersService(
|
|||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
|
||||||
* Create and show a desktop notification with robust icon handling.
|
|
||||||
* Supports both file paths (preferred on Linux/Wayland) and data URLs (fallback).
|
|
||||||
*/
|
|
||||||
function createFieldGroupingCallback() {
|
|
||||||
return createFieldGroupingCallbackRuntimeService({
|
|
||||||
getVisibleOverlayVisible: () => visibleOverlayVisible,
|
|
||||||
getInvisibleOverlayVisible: () => invisibleOverlayVisible,
|
|
||||||
setVisibleOverlayVisible: (visible) => setVisibleOverlayVisible(visible),
|
|
||||||
setInvisibleOverlayVisible: (visible) => setInvisibleOverlayVisible(visible),
|
|
||||||
getResolver: () => fieldGroupingResolver,
|
|
||||||
setResolver: (resolver) => {
|
|
||||||
fieldGroupingResolver = resolver;
|
|
||||||
},
|
|
||||||
sendToVisibleOverlay: (
|
|
||||||
channel,
|
|
||||||
payload,
|
|
||||||
runtimeOptions?: { restoreOnModalClose?: OverlayHostedModal },
|
|
||||||
) =>
|
|
||||||
sendToVisibleOverlay(channel, payload, runtimeOptions),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function sendToVisibleOverlay(channel: string, payload?: unknown, options?: { restoreOnModalClose?: OverlayHostedModal }): boolean {
|
|
||||||
return sendToVisibleOverlayRuntimeService({
|
|
||||||
mainWindow,
|
|
||||||
visibleOverlayVisible,
|
|
||||||
setVisibleOverlayVisible: (visible) => setVisibleOverlayVisible(visible),
|
|
||||||
channel,
|
|
||||||
payload,
|
|
||||||
restoreOnModalClose: options?.restoreOnModalClose,
|
|
||||||
restoreVisibleOverlayOnModalClose,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
registerAnkiJimakuIpcRuntimeService(
|
registerAnkiJimakuIpcRuntimeService(
|
||||||
createAnkiJimakuIpcDepsRuntimeService({
|
createAnkiJimakuIpcDepsRuntimeService({
|
||||||
patchAnkiConnectEnabled: (enabled) => {
|
patchAnkiConnectEnabled: (enabled) => {
|
||||||
|
|||||||
Reference in New Issue
Block a user