mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-02-27 18:22:41 -08:00
refactor: extract tray runtime handler wiring
This commit is contained in:
@@ -6,7 +6,7 @@ Read first. Keep concise.
|
|||||||
| ------------ | -------------- | ---------------------------------------------------- | --------- | ------------------------------------- | ---------------------- |
|
| ------------ | -------------- | ---------------------------------------------------- | --------- | ------------------------------------- | ---------------------- |
|
||||||
| `codex-generate-minecard-image-20260220T112900Z-vsxr` | `codex-generate-minecard-image` | `Generate media fallbacks (GIF) from assets/minecard.webm and wire README/docs fallback markup` | `done` | `docs/subagents/agents/codex-generate-minecard-image-20260220T112900Z-vsxr.md` | `2026-02-20T11:35:30Z` |
|
| `codex-generate-minecard-image-20260220T112900Z-vsxr` | `codex-generate-minecard-image` | `Generate media fallbacks (GIF) from assets/minecard.webm and wire README/docs fallback markup` | `done` | `docs/subagents/agents/codex-generate-minecard-image-20260220T112900Z-vsxr.md` | `2026-02-20T11:35:30Z` |
|
||||||
| `codex-main` | `planner-exec` | `Fix frequency/N+1 regression in plugin --start flow` | `in_progress` | `docs/subagents/agents/codex-main.md` | `2026-02-19T19:36:46Z` |
|
| `codex-main` | `planner-exec` | `Fix frequency/N+1 regression in plugin --start flow` | `in_progress` | `docs/subagents/agents/codex-main.md` | `2026-02-19T19:36:46Z` |
|
||||||
| `codex-task85-20260219T233711Z-46hc` | `codex-task85` | `Resume TASK-85 maintainability refactor from latest handoff point` | `in_progress` | `docs/subagents/agents/codex-task85-20260219T233711Z-46hc.md` | `2026-02-20T10:06:15Z` |
|
| `codex-task85-20260219T233711Z-46hc` | `codex-task85` | `Resume TASK-85 maintainability refactor from latest handoff point` | `in_progress` | `docs/subagents/agents/codex-task85-20260219T233711Z-46hc.md` | `2026-02-20T10:14:17Z` |
|
||||||
| `codex-config-validation-20260219T172015Z-iiyf` | `codex-config-validation` | `Find root cause of config validation error for ~/.config/SubMiner/config.jsonc` | `completed` | `docs/subagents/agents/codex-config-validation-20260219T172015Z-iiyf.md` | `2026-02-19T17:26:17Z` |
|
| `codex-config-validation-20260219T172015Z-iiyf` | `codex-config-validation` | `Find root cause of config validation error for ~/.config/SubMiner/config.jsonc` | `completed` | `docs/subagents/agents/codex-config-validation-20260219T172015Z-iiyf.md` | `2026-02-19T17:26:17Z` |
|
||||||
| `codex-task85-20260219T233711Z-46hc` | `codex-task85` | `Resume TASK-85 maintainability refactor from latest handoff point` | `in_progress` | `docs/subagents/agents/codex-task85-20260219T233711Z-46hc.md` | `2026-02-20T02:56:34Z` |
|
| `codex-task85-20260219T233711Z-46hc` | `codex-task85` | `Resume TASK-85 maintainability refactor from latest handoff point` | `in_progress` | `docs/subagents/agents/codex-task85-20260219T233711Z-46hc.md` | `2026-02-20T02:56:34Z` |
|
||||||
| `codex-anilist-deeplink-20260219T233926Z` | `anilist-deeplink` | `Fix external subminer:// AniList callback handling from browser` | `done` | `docs/subagents/agents/codex-anilist-deeplink-20260219T233926Z.md` | `2026-02-19T23:59:21Z` |
|
| `codex-anilist-deeplink-20260219T233926Z` | `anilist-deeplink` | `Fix external subminer:// AniList callback handling from browser` | `done` | `docs/subagents/agents/codex-anilist-deeplink-20260219T233926Z.md` | `2026-02-19T23:59:21Z` |
|
||||||
|
|||||||
@@ -9,6 +9,10 @@
|
|||||||
|
|
||||||
## Current Work (newest first)
|
## Current Work (newest first)
|
||||||
|
|
||||||
|
- [2026-02-20T10:14:17Z] progress: extracted tray composition wiring from `src/main.ts` into `src/main/runtime/tray-runtime-handlers.ts`; `main.ts` now composes `resolveTrayIconPath`/`buildTrayMenu`/`ensureTray`/`destroyTray` through one factory.
|
||||||
|
- [2026-02-20T10:14:17Z] progress: added `src/main/runtime/tray-runtime-handlers.test.ts` covering composed tray handler behavior.
|
||||||
|
- [2026-02-20T10:14:17Z] test: `bun run build` pass (expected macOS helper Swift cache fallback) + focused suites pass for `tray-runtime-handlers*`, `tray-main-actions*`, `tray-main-deps*`, `tray-runtime*`, `tray-lifecycle*`, `cli-command-context-factory*`, and `overlay-visibility-runtime*` (12/12).
|
||||||
|
- [2026-02-20T10:14:17Z] scope: staging `src/main.ts`, new tray runtime handlers module/tests, and subagent bookkeeping only.
|
||||||
- [2026-02-20T10:06:15Z] progress: extracted CLI command-context composition from `src/main.ts` into `src/main/runtime/cli-command-context-factory.ts`; `main.ts` now creates context via `createCliCommandContextFactory(...)`.
|
- [2026-02-20T10:06:15Z] progress: extracted CLI command-context composition from `src/main.ts` into `src/main/runtime/cli-command-context-factory.ts`; `main.ts` now creates context via `createCliCommandContextFactory(...)`.
|
||||||
- [2026-02-20T10:06:15Z] progress: added `src/main/runtime/cli-command-context-factory.test.ts` to validate composed factory behavior.
|
- [2026-02-20T10:06:15Z] progress: added `src/main/runtime/cli-command-context-factory.test.ts` to validate composed factory behavior.
|
||||||
- [2026-02-20T10:06:15Z] test: `bun run build` pass (expected macOS helper Swift cache fallback) + focused suites pass for `cli-command-context-factory*`, `cli-command-context*`, `ipc-runtime-handlers*`, `ipc-bridge-actions*`, `overlay-visibility-runtime*`, and `yomitan-extension-runtime*` (12/12).
|
- [2026-02-20T10:06:15Z] test: `bun run build` pass (expected macOS helper Swift cache fallback) + focused suites pass for `cli-command-context-factory*`, `cli-command-context*`, `ipc-runtime-handlers*`, `ipc-bridge-actions*`, `overlay-visibility-runtime*`, and `yomitan-extension-runtime*` (12/12).
|
||||||
|
|||||||
48
src/main.ts
48
src/main.ts
@@ -289,7 +289,6 @@ import {
|
|||||||
createBuildUpdateVisibleOverlayBoundsMainDepsHandler,
|
createBuildUpdateVisibleOverlayBoundsMainDepsHandler,
|
||||||
} from './main/runtime/overlay-window-layout-main-deps';
|
} from './main/runtime/overlay-window-layout-main-deps';
|
||||||
import { buildTrayMenuTemplateRuntime, resolveTrayIconPathRuntime } from './main/runtime/tray-runtime';
|
import { buildTrayMenuTemplateRuntime, resolveTrayIconPathRuntime } from './main/runtime/tray-runtime';
|
||||||
import { createDestroyTrayHandler, createEnsureTrayHandler } from './main/runtime/tray-lifecycle';
|
|
||||||
import { createInitializeOverlayRuntimeHandler } from './main/runtime/overlay-runtime-bootstrap';
|
import { createInitializeOverlayRuntimeHandler } from './main/runtime/overlay-runtime-bootstrap';
|
||||||
import { createOpenYomitanSettingsHandler } from './main/runtime/yomitan-settings-opener';
|
import { createOpenYomitanSettingsHandler } from './main/runtime/yomitan-settings-opener';
|
||||||
import {
|
import {
|
||||||
@@ -393,19 +392,10 @@ import {
|
|||||||
createBuildCreateOverlayWindowMainDepsHandler,
|
createBuildCreateOverlayWindowMainDepsHandler,
|
||||||
} from './main/runtime/overlay-window-factory-main-deps';
|
} from './main/runtime/overlay-window-factory-main-deps';
|
||||||
import {
|
import {
|
||||||
createBuildTrayMenuTemplateHandler,
|
|
||||||
createResolveTrayIconPathHandler,
|
|
||||||
} from './main/runtime/tray-main-actions';
|
|
||||||
import {
|
|
||||||
createBuildResolveTrayIconPathMainDepsHandler,
|
|
||||||
createBuildTrayMenuTemplateMainDepsHandler,
|
|
||||||
} from './main/runtime/tray-main-deps';
|
|
||||||
import {
|
|
||||||
createBuildDestroyTrayMainDepsHandler,
|
|
||||||
createBuildEnsureTrayMainDepsHandler,
|
|
||||||
createBuildInitializeOverlayRuntimeBootstrapMainDepsHandler,
|
createBuildInitializeOverlayRuntimeBootstrapMainDepsHandler,
|
||||||
createBuildOpenYomitanSettingsMainDepsHandler,
|
createBuildOpenYomitanSettingsMainDepsHandler,
|
||||||
} from './main/runtime/app-runtime-main-deps';
|
} from './main/runtime/app-runtime-main-deps';
|
||||||
|
import { createTrayRuntimeHandlers } from './main/runtime/tray-runtime-handlers';
|
||||||
import { createYomitanExtensionRuntime } from './main/runtime/yomitan-extension-runtime';
|
import { createYomitanExtensionRuntime } from './main/runtime/yomitan-extension-runtime';
|
||||||
import { createBuildInitializeOverlayRuntimeOptionsHandler } from './main/runtime/overlay-runtime-options';
|
import { createBuildInitializeOverlayRuntimeOptionsHandler } from './main/runtime/overlay-runtime-options';
|
||||||
import { createBuildInitializeOverlayRuntimeMainDepsHandler } from './main/runtime/overlay-runtime-options-main-deps';
|
import { createBuildInitializeOverlayRuntimeMainDepsHandler } from './main/runtime/overlay-runtime-options-main-deps';
|
||||||
@@ -2525,7 +2515,7 @@ function resolveTrayIconPath(): string | null {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function buildTrayMenu(): Menu {
|
function buildTrayMenu(): Menu {
|
||||||
return Menu.buildFromTemplate(buildTrayMenuTemplateHandler());
|
return buildTrayMenuHandler();
|
||||||
}
|
}
|
||||||
|
|
||||||
function ensureTray(): void {
|
function ensureTray(): void {
|
||||||
@@ -3036,8 +3026,13 @@ const createInvisibleWindowHandler = createCreateInvisibleWindowHandler<BrowserW
|
|||||||
setInvisibleWindow: (window) => overlayManager.setInvisibleWindow(window),
|
setInvisibleWindow: (window) => overlayManager.setInvisibleWindow(window),
|
||||||
})(),
|
})(),
|
||||||
);
|
);
|
||||||
const resolveTrayIconPathHandler = createResolveTrayIconPathHandler(
|
const {
|
||||||
createBuildResolveTrayIconPathMainDepsHandler({
|
resolveTrayIconPath: resolveTrayIconPathHandler,
|
||||||
|
buildTrayMenu: buildTrayMenuHandler,
|
||||||
|
ensureTray: ensureTrayHandler,
|
||||||
|
destroyTray: destroyTrayHandler,
|
||||||
|
} = createTrayRuntimeHandlers({
|
||||||
|
resolveTrayIconPathDeps: {
|
||||||
resolveTrayIconPathRuntime,
|
resolveTrayIconPathRuntime,
|
||||||
platform: process.platform,
|
platform: process.platform,
|
||||||
resourcesPath: process.resourcesPath,
|
resourcesPath: process.resourcesPath,
|
||||||
@@ -3045,10 +3040,8 @@ const resolveTrayIconPathHandler = createResolveTrayIconPathHandler(
|
|||||||
dirname: __dirname,
|
dirname: __dirname,
|
||||||
joinPath: (...parts) => path.join(...parts),
|
joinPath: (...parts) => path.join(...parts),
|
||||||
fileExists: (candidate) => fs.existsSync(candidate),
|
fileExists: (candidate) => fs.existsSync(candidate),
|
||||||
})(),
|
},
|
||||||
);
|
buildTrayMenuTemplateDeps: {
|
||||||
const buildTrayMenuTemplateHandler = createBuildTrayMenuTemplateHandler(
|
|
||||||
createBuildTrayMenuTemplateMainDepsHandler({
|
|
||||||
buildTrayMenuTemplateRuntime,
|
buildTrayMenuTemplateRuntime,
|
||||||
initializeOverlayRuntime: () => initializeOverlayRuntime(),
|
initializeOverlayRuntime: () => initializeOverlayRuntime(),
|
||||||
isOverlayRuntimeInitialized: () => appState.overlayRuntimeInitialized,
|
isOverlayRuntimeInitialized: () => appState.overlayRuntimeInitialized,
|
||||||
@@ -3058,16 +3051,12 @@ const buildTrayMenuTemplateHandler = createBuildTrayMenuTemplateHandler(
|
|||||||
openJellyfinSetupWindow: () => openJellyfinSetupWindow(),
|
openJellyfinSetupWindow: () => openJellyfinSetupWindow(),
|
||||||
openAnilistSetupWindow: () => openAnilistSetupWindow(),
|
openAnilistSetupWindow: () => openAnilistSetupWindow(),
|
||||||
quitApp: () => app.quit(),
|
quitApp: () => app.quit(),
|
||||||
})(),
|
},
|
||||||
);
|
ensureTrayDeps: {
|
||||||
const ensureTrayHandler = createEnsureTrayHandler(
|
|
||||||
createBuildEnsureTrayMainDepsHandler({
|
|
||||||
getTray: () => appTray,
|
getTray: () => appTray,
|
||||||
setTray: (tray) => {
|
setTray: (tray) => {
|
||||||
appTray = tray as Tray | null;
|
appTray = tray as Tray | null;
|
||||||
},
|
},
|
||||||
buildTrayMenu: () => buildTrayMenu(),
|
|
||||||
resolveTrayIconPath: () => resolveTrayIconPath(),
|
|
||||||
createImageFromPath: (iconPath) => nativeImage.createFromPath(iconPath),
|
createImageFromPath: (iconPath) => nativeImage.createFromPath(iconPath),
|
||||||
createEmptyImage: () => nativeImage.createEmpty(),
|
createEmptyImage: () => nativeImage.createEmpty(),
|
||||||
createTray: (icon) => new Tray(icon as never),
|
createTray: (icon) => new Tray(icon as never),
|
||||||
@@ -3077,16 +3066,15 @@ const ensureTrayHandler = createEnsureTrayHandler(
|
|||||||
initializeOverlayRuntime: () => initializeOverlayRuntime(),
|
initializeOverlayRuntime: () => initializeOverlayRuntime(),
|
||||||
isOverlayRuntimeInitialized: () => appState.overlayRuntimeInitialized,
|
isOverlayRuntimeInitialized: () => appState.overlayRuntimeInitialized,
|
||||||
setVisibleOverlayVisible: (visible) => setVisibleOverlayVisible(visible),
|
setVisibleOverlayVisible: (visible) => setVisibleOverlayVisible(visible),
|
||||||
})(),
|
},
|
||||||
);
|
destroyTrayDeps: {
|
||||||
const destroyTrayHandler = createDestroyTrayHandler(
|
|
||||||
createBuildDestroyTrayMainDepsHandler({
|
|
||||||
getTray: () => appTray,
|
getTray: () => appTray,
|
||||||
setTray: (tray) => {
|
setTray: (tray) => {
|
||||||
appTray = tray as Tray | null;
|
appTray = tray as Tray | null;
|
||||||
},
|
},
|
||||||
})(),
|
},
|
||||||
);
|
buildMenuFromTemplate: (template) => Menu.buildFromTemplate(template),
|
||||||
|
});
|
||||||
const yomitanExtensionRuntime = createYomitanExtensionRuntime({
|
const yomitanExtensionRuntime = createYomitanExtensionRuntime({
|
||||||
loadYomitanExtensionCore,
|
loadYomitanExtensionCore,
|
||||||
userDataPath: USER_DATA_PATH,
|
userDataPath: USER_DATA_PATH,
|
||||||
|
|||||||
96
src/main/runtime/tray-runtime-handlers.test.ts
Normal file
96
src/main/runtime/tray-runtime-handlers.test.ts
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import assert from 'node:assert/strict';
|
||||||
|
import test from 'node:test';
|
||||||
|
import { createTrayRuntimeHandlers } from './tray-runtime-handlers';
|
||||||
|
|
||||||
|
test('tray runtime handlers compose resolve/menu/ensure/destroy handlers', () => {
|
||||||
|
let tray: { destroyed: boolean } | null = null;
|
||||||
|
let overlayInitialized = false;
|
||||||
|
let visibleOverlay = false;
|
||||||
|
const calls: string[] = [];
|
||||||
|
|
||||||
|
const runtime = createTrayRuntimeHandlers({
|
||||||
|
resolveTrayIconPathDeps: {
|
||||||
|
resolveTrayIconPathRuntime: () => '/tmp/SubMiner.png',
|
||||||
|
platform: 'darwin',
|
||||||
|
resourcesPath: '/resources',
|
||||||
|
appPath: '/app',
|
||||||
|
dirname: '/dirname',
|
||||||
|
joinPath: (...parts) => parts.join('/'),
|
||||||
|
fileExists: () => true,
|
||||||
|
},
|
||||||
|
buildTrayMenuTemplateDeps: {
|
||||||
|
buildTrayMenuTemplateRuntime: () => [{ label: 'Open Overlay' }],
|
||||||
|
initializeOverlayRuntime: () => {
|
||||||
|
overlayInitialized = true;
|
||||||
|
},
|
||||||
|
isOverlayRuntimeInitialized: () => overlayInitialized,
|
||||||
|
setVisibleOverlayVisible: (visible) => {
|
||||||
|
visibleOverlay = visible;
|
||||||
|
},
|
||||||
|
openYomitanSettings: () => {},
|
||||||
|
openRuntimeOptionsPalette: () => {},
|
||||||
|
openJellyfinSetupWindow: () => {},
|
||||||
|
openAnilistSetupWindow: () => {},
|
||||||
|
quitApp: () => {},
|
||||||
|
},
|
||||||
|
ensureTrayDeps: {
|
||||||
|
getTray: () => tray as never,
|
||||||
|
setTray: (next) => {
|
||||||
|
tray = next as { destroyed: boolean } | null;
|
||||||
|
},
|
||||||
|
createImageFromPath: () => ({
|
||||||
|
isEmpty: () => false,
|
||||||
|
resize: () => ({
|
||||||
|
isEmpty: () => false,
|
||||||
|
resize: () => ({} as never),
|
||||||
|
setTemplateImage: () => {},
|
||||||
|
}),
|
||||||
|
setTemplateImage: () => {},
|
||||||
|
}),
|
||||||
|
createEmptyImage: () => ({
|
||||||
|
isEmpty: () => true,
|
||||||
|
resize: () => ({} as never),
|
||||||
|
setTemplateImage: () => {},
|
||||||
|
}),
|
||||||
|
createTray: () => ({
|
||||||
|
setContextMenu: () => calls.push('setContextMenu'),
|
||||||
|
setToolTip: () => calls.push('setToolTip'),
|
||||||
|
on: (_event: 'click', handler: () => void) => {
|
||||||
|
calls.push('on-click');
|
||||||
|
handler();
|
||||||
|
},
|
||||||
|
destroy: () => {
|
||||||
|
calls.push('destroy');
|
||||||
|
if (tray) tray.destroyed = true;
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
trayTooltip: 'SubMiner',
|
||||||
|
platform: 'darwin',
|
||||||
|
logWarn: (message) => calls.push(`warn:${message}`),
|
||||||
|
initializeOverlayRuntime: () => {
|
||||||
|
overlayInitialized = true;
|
||||||
|
},
|
||||||
|
isOverlayRuntimeInitialized: () => overlayInitialized,
|
||||||
|
setVisibleOverlayVisible: (visible) => {
|
||||||
|
visibleOverlay = visible;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
destroyTrayDeps: {
|
||||||
|
getTray: () => tray as never,
|
||||||
|
setTray: (next) => {
|
||||||
|
tray = next as { destroyed: boolean } | null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
buildMenuFromTemplate: (template) => ({ template }),
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(runtime.resolveTrayIconPath(), '/tmp/SubMiner.png');
|
||||||
|
assert.deepEqual(runtime.buildTrayMenu(), { template: [{ label: 'Open Overlay' }] });
|
||||||
|
runtime.ensureTray();
|
||||||
|
assert.equal(overlayInitialized, true);
|
||||||
|
assert.equal(visibleOverlay, true);
|
||||||
|
assert.ok(tray);
|
||||||
|
runtime.destroyTray();
|
||||||
|
assert.equal(tray, null);
|
||||||
|
assert.deepEqual(calls, ['setToolTip', 'setContextMenu', 'on-click', 'destroy']);
|
||||||
|
});
|
||||||
48
src/main/runtime/tray-runtime-handlers.ts
Normal file
48
src/main/runtime/tray-runtime-handlers.ts
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
import { createDestroyTrayHandler, createEnsureTrayHandler } from './tray-lifecycle';
|
||||||
|
import { createBuildDestroyTrayMainDepsHandler, createBuildEnsureTrayMainDepsHandler } from './app-runtime-main-deps';
|
||||||
|
import { createBuildTrayMenuTemplateHandler, createResolveTrayIconPathHandler } from './tray-main-actions';
|
||||||
|
import {
|
||||||
|
createBuildResolveTrayIconPathMainDepsHandler,
|
||||||
|
createBuildTrayMenuTemplateMainDepsHandler,
|
||||||
|
} from './tray-main-deps';
|
||||||
|
|
||||||
|
type ResolveTrayIconPathMainDeps = Parameters<typeof createBuildResolveTrayIconPathMainDepsHandler>[0];
|
||||||
|
type BuildTrayMenuTemplateMainDeps<TMenuItem> = Parameters<
|
||||||
|
typeof createBuildTrayMenuTemplateMainDepsHandler<TMenuItem>
|
||||||
|
>[0];
|
||||||
|
type EnsureTrayMainDeps = Parameters<typeof createBuildEnsureTrayMainDepsHandler>[0];
|
||||||
|
type DestroyTrayMainDeps = Parameters<typeof createBuildDestroyTrayMainDepsHandler>[0];
|
||||||
|
|
||||||
|
export function createTrayRuntimeHandlers<TMenuItem, TMenu>(deps: {
|
||||||
|
resolveTrayIconPathDeps: ResolveTrayIconPathMainDeps;
|
||||||
|
buildTrayMenuTemplateDeps: BuildTrayMenuTemplateMainDeps<TMenuItem>;
|
||||||
|
ensureTrayDeps: Omit<EnsureTrayMainDeps, 'buildTrayMenu' | 'resolveTrayIconPath'>;
|
||||||
|
destroyTrayDeps: DestroyTrayMainDeps;
|
||||||
|
buildMenuFromTemplate: (template: TMenuItem[]) => TMenu;
|
||||||
|
}) {
|
||||||
|
const resolveTrayIconPath = createResolveTrayIconPathHandler(
|
||||||
|
createBuildResolveTrayIconPathMainDepsHandler(deps.resolveTrayIconPathDeps)(),
|
||||||
|
);
|
||||||
|
const buildTrayMenuTemplate = createBuildTrayMenuTemplateHandler(
|
||||||
|
createBuildTrayMenuTemplateMainDepsHandler(deps.buildTrayMenuTemplateDeps)(),
|
||||||
|
);
|
||||||
|
const buildTrayMenu = () => deps.buildMenuFromTemplate(buildTrayMenuTemplate());
|
||||||
|
|
||||||
|
const ensureTray = createEnsureTrayHandler(
|
||||||
|
createBuildEnsureTrayMainDepsHandler({
|
||||||
|
...deps.ensureTrayDeps,
|
||||||
|
buildTrayMenu: () => buildTrayMenu(),
|
||||||
|
resolveTrayIconPath: () => resolveTrayIconPath(),
|
||||||
|
})(),
|
||||||
|
);
|
||||||
|
const destroyTray = createDestroyTrayHandler(
|
||||||
|
createBuildDestroyTrayMainDepsHandler(deps.destroyTrayDeps)(),
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
resolveTrayIconPath,
|
||||||
|
buildTrayMenu,
|
||||||
|
ensureTray,
|
||||||
|
destroyTray,
|
||||||
|
};
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user