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 void)[]> = {}; const loadedPluginMap: Record< string, PluginDef > = {}; export const createContext = ( id: string, ): RendererContext => ({ 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 | undefined => { return loadedPluginMap[id]; }; export const getAllLoadedRendererPlugins = () => { return loadedPluginMap; };