mirror of
https://github.com/th-ch/youtube-music.git
synced 2026-01-11 10:31:47 +00:00
146 lines
3.8 KiB
TypeScript
146 lines
3.8 KiB
TypeScript
import { deepmerge } from 'deepmerge-ts';
|
|
|
|
import { rendererPlugins } from 'virtual:plugins';
|
|
|
|
import { LoggerPrefix, startPlugin, stopPlugin } from '@/utils';
|
|
|
|
import { t } from '@/i18n';
|
|
|
|
import type { RendererContext } from '@/types/contexts';
|
|
import type { PluginConfig, PluginDef } from '@/types/plugins';
|
|
|
|
const unregisterStyleMap: Record<string, (() => void)[]> = {};
|
|
const loadedPluginMap: Record<
|
|
string,
|
|
PluginDef<unknown, unknown, unknown>
|
|
> = {};
|
|
|
|
export const createContext = <Config extends PluginConfig>(
|
|
id: string,
|
|
): RendererContext<Config> => ({
|
|
getConfig: async () => window.ipcRenderer.invoke('ytmd:get-config', id),
|
|
setConfig: async (newConfig) => {
|
|
await window.ipcRenderer.invoke('ytmd:set-config', id, newConfig);
|
|
},
|
|
ipc: {
|
|
send: (event: string, ...args: unknown[]) => {
|
|
window.ipcRenderer.send(event, ...args);
|
|
},
|
|
invoke: (event: string, ...args: unknown[]) =>
|
|
window.ipcRenderer.invoke(event, ...args),
|
|
on: (event: string, listener: CallableFunction) => {
|
|
window.ipcRenderer.on(event, (_, ...args: unknown[]) => {
|
|
listener(...args);
|
|
});
|
|
},
|
|
removeAllListeners: (event: string) => {
|
|
window.ipcRenderer.removeAllListeners(event);
|
|
},
|
|
},
|
|
});
|
|
|
|
export const forceUnloadRendererPlugin = async (id: string) => {
|
|
unregisterStyleMap[id]?.forEach((unregister) => unregister());
|
|
|
|
delete unregisterStyleMap[id];
|
|
delete loadedPluginMap[id];
|
|
|
|
const plugin = rendererPlugins[id];
|
|
if (!plugin) return;
|
|
|
|
const hasStopped = await stopPlugin(id, plugin, {
|
|
ctx: 'renderer',
|
|
context: createContext(id),
|
|
});
|
|
if (plugin?.stylesheets) {
|
|
document.querySelector(`style#plugin-${id}`)?.remove();
|
|
}
|
|
if (hasStopped || (hasStopped === null && plugin?.renderer)) {
|
|
console.log(
|
|
LoggerPrefix,
|
|
t('common.console.plugins.unloaded', { pluginName: id }),
|
|
);
|
|
} else {
|
|
console.error(
|
|
LoggerPrefix,
|
|
t('common.console.plugins.unload-failed', { pluginName: id }),
|
|
);
|
|
}
|
|
};
|
|
|
|
export const forceLoadRendererPlugin = async (id: string) => {
|
|
const plugin = rendererPlugins[id];
|
|
if (!plugin) return;
|
|
|
|
const hasEvaled = await startPlugin(id, plugin, {
|
|
ctx: 'renderer',
|
|
context: createContext(id),
|
|
});
|
|
|
|
if (
|
|
hasEvaled ||
|
|
plugin?.stylesheets ||
|
|
(hasEvaled === null &&
|
|
typeof plugin?.renderer !== 'function' &&
|
|
plugin?.renderer)
|
|
) {
|
|
loadedPluginMap[id] = plugin;
|
|
|
|
if (plugin?.stylesheets) {
|
|
const styleSheetList = plugin.stylesheets.map((style) => {
|
|
const styleSheet = new CSSStyleSheet();
|
|
styleSheet.replaceSync(style);
|
|
|
|
return styleSheet;
|
|
});
|
|
|
|
document.adoptedStyleSheets = [
|
|
...document.adoptedStyleSheets,
|
|
...styleSheetList,
|
|
];
|
|
}
|
|
|
|
console.log(
|
|
LoggerPrefix,
|
|
t('common.console.plugins.loaded', { pluginName: id }),
|
|
);
|
|
} else {
|
|
console.log(
|
|
LoggerPrefix,
|
|
t('common.console.plugins.initialize-failed', { pluginName: id }),
|
|
);
|
|
}
|
|
};
|
|
|
|
export const loadAllRendererPlugins = async () => {
|
|
const pluginConfigs = window.mainConfig.plugins.getPlugins();
|
|
|
|
for (const [pluginId, pluginDef] of Object.entries(rendererPlugins)) {
|
|
const config = deepmerge(pluginDef.config, pluginConfigs[pluginId] ?? {});
|
|
|
|
if (config.enabled) {
|
|
await forceLoadRendererPlugin(pluginId);
|
|
} else {
|
|
if (loadedPluginMap[pluginId]) {
|
|
await forceUnloadRendererPlugin(pluginId);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
export const unloadAllRendererPlugins = async () => {
|
|
for (const id of Object.keys(loadedPluginMap)) {
|
|
await forceUnloadRendererPlugin(id);
|
|
}
|
|
};
|
|
|
|
export const getLoadedRendererPlugin = (
|
|
id: string,
|
|
): PluginDef<unknown, unknown, unknown> | undefined => {
|
|
return loadedPluginMap[id];
|
|
};
|
|
|
|
export const getAllLoadedRendererPlugins = () => {
|
|
return loadedPluginMap;
|
|
};
|