2020-10-09 19:35:46 +02:00
|
|
|
import { Draft } from 'immer';
|
2024-03-09 13:03:57 +02:00
|
|
|
import { ContainerType } from './WebviewController';
|
|
|
|
import { ButtonSpec } from './api/types';
|
2020-10-09 19:35:46 +02:00
|
|
|
|
2024-03-09 13:03:57 +02:00
|
|
|
export interface PluginViewState {
|
2020-11-12 21:29:22 +02:00
|
|
|
id: string;
|
|
|
|
type: string;
|
2024-03-09 13:03:57 +02:00
|
|
|
opened: boolean;
|
|
|
|
buttons: ButtonSpec[];
|
|
|
|
fitToContent?: boolean;
|
|
|
|
scripts?: string[];
|
|
|
|
html?: string;
|
|
|
|
commandName?: string;
|
|
|
|
location?: string;
|
|
|
|
containerType: ContainerType;
|
2020-10-09 19:35:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
interface PluginViewStates {
|
2020-11-12 21:29:22 +02:00
|
|
|
[key: string]: PluginViewState;
|
2020-10-09 19:35:46 +02:00
|
|
|
}
|
|
|
|
|
2020-10-21 01:23:55 +02:00
|
|
|
interface PluginContentScriptState {
|
2020-11-12 21:29:22 +02:00
|
|
|
id: string;
|
|
|
|
path: string;
|
2020-10-21 01:23:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
interface PluginContentScriptStates {
|
2020-11-12 21:13:28 +02:00
|
|
|
[type: string]: PluginContentScriptState[];
|
2020-10-21 01:23:55 +02:00
|
|
|
}
|
|
|
|
|
2020-10-09 19:35:46 +02:00
|
|
|
interface PluginState {
|
2020-11-12 21:29:22 +02:00
|
|
|
id: string;
|
|
|
|
contentScripts: PluginContentScriptStates;
|
|
|
|
views: PluginViewStates;
|
2020-10-09 19:35:46 +02:00
|
|
|
}
|
|
|
|
|
2024-03-09 13:03:57 +02:00
|
|
|
export interface ViewInfo {
|
|
|
|
view: PluginViewState;
|
|
|
|
plugin: PluginState;
|
|
|
|
}
|
|
|
|
|
2020-10-09 19:35:46 +02:00
|
|
|
export interface PluginStates {
|
2020-11-12 21:13:28 +02:00
|
|
|
[key: string]: PluginState;
|
2020-10-09 19:35:46 +02:00
|
|
|
}
|
|
|
|
|
2022-04-17 13:41:27 +02:00
|
|
|
export interface PluginHtmlContent {
|
|
|
|
[viewId: string]: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface PluginHtmlContents {
|
|
|
|
[pluginId: string]: PluginHtmlContent;
|
|
|
|
}
|
|
|
|
|
2020-10-09 19:35:46 +02:00
|
|
|
export interface State {
|
2020-11-12 21:29:22 +02:00
|
|
|
plugins: PluginStates;
|
2022-04-17 13:41:27 +02:00
|
|
|
pluginHtmlContents: PluginHtmlContents;
|
2023-09-18 18:40:36 +02:00
|
|
|
allPluginsStarted: boolean;
|
2020-10-09 19:35:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export const stateRootKey = 'pluginService';
|
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
export const defaultState: State = {
|
2020-10-09 19:35:46 +02:00
|
|
|
plugins: {},
|
2022-04-17 13:41:27 +02:00
|
|
|
pluginHtmlContents: {},
|
2023-09-18 18:40:36 +02:00
|
|
|
allPluginsStarted: false,
|
2020-10-09 19:35:46 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
export const utils = {
|
2020-10-13 13:57:03 +02:00
|
|
|
|
|
|
|
// It is best to use viewsByType instead as this method creates new objects
|
2024-02-26 12:16:23 +02:00
|
|
|
// which might trigger unnecessary renders even when plugin and views haven't changed.
|
2020-11-12 21:13:28 +02:00
|
|
|
viewInfosByType: function(plugins: PluginStates, type: string): ViewInfo[] {
|
|
|
|
const output: ViewInfo[] = [];
|
2020-10-09 19:35:46 +02:00
|
|
|
|
|
|
|
for (const pluginId in plugins) {
|
|
|
|
const plugin = plugins[pluginId];
|
|
|
|
for (const viewId in plugin.views) {
|
|
|
|
const view = plugin.views[viewId];
|
|
|
|
if (view.type !== type) continue;
|
|
|
|
|
|
|
|
output.push({
|
|
|
|
plugin: plugin,
|
|
|
|
view: view,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
},
|
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
viewsByType: function(plugins: PluginStates, type: string): any[] {
|
|
|
|
const output: any[] = [];
|
2020-10-13 13:57:03 +02:00
|
|
|
|
|
|
|
for (const pluginId in plugins) {
|
|
|
|
const plugin = plugins[pluginId];
|
|
|
|
for (const viewId in plugin.views) {
|
|
|
|
const view = plugin.views[viewId];
|
|
|
|
if (view.type !== type) continue;
|
|
|
|
|
|
|
|
output.push(view);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
},
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
viewInfoByViewId: function(plugins: PluginStates, viewId: string): ViewInfo {
|
|
|
|
for (const pluginId in plugins) {
|
|
|
|
const plugin = plugins[pluginId];
|
|
|
|
if (plugin.views[viewId]) {
|
|
|
|
return {
|
|
|
|
plugin: plugin,
|
|
|
|
view: plugin.views[viewId],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
allViewIds: function(plugins: PluginStates): string[] {
|
|
|
|
const output = [];
|
|
|
|
for (const pluginId in plugins) {
|
|
|
|
const plugin = plugins[pluginId];
|
|
|
|
for (const viewId in plugin.views) {
|
|
|
|
output.push(viewId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
},
|
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
commandNamesFromViews: function(plugins: PluginStates, toolbarType: string): string[] {
|
2020-10-09 19:35:46 +02:00
|
|
|
const infos = utils.viewInfosByType(plugins, 'toolbarButton');
|
|
|
|
|
|
|
|
return infos
|
2020-11-12 21:13:28 +02:00
|
|
|
.filter((info: ViewInfo) => info.view.location === toolbarType)
|
|
|
|
.map((info: ViewInfo) => info.view.commandName);
|
2020-10-09 19:35:46 +02:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
const reducer = (draftRoot: Draft<any>, action: any) => {
|
2020-10-09 19:35:46 +02:00
|
|
|
if (action.type.indexOf('PLUGIN_') !== 0) return;
|
|
|
|
|
|
|
|
// All actions should be scoped to a plugin, except when adding a new plugin
|
|
|
|
if (!action.pluginId && action.type !== 'PLUGIN_ADD') throw new Error(`action.pluginId is required. Action was: ${JSON.stringify(action)}`);
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
const draft = draftRoot.pluginService as State;
|
|
|
|
|
2020-10-09 19:35:46 +02:00
|
|
|
try {
|
|
|
|
switch (action.type) {
|
|
|
|
|
|
|
|
case 'PLUGIN_ADD':
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
if (draft.plugins[action.plugin.id]) throw new Error(`Plugin is already loaded: ${JSON.stringify(action)}`);
|
|
|
|
draft.plugins[action.plugin.id] = action.plugin;
|
2020-10-09 19:35:46 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'PLUGIN_VIEW_ADD':
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
draft.plugins[action.pluginId].views[action.view.id] = { ...action.view };
|
2020-10-09 19:35:46 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'PLUGIN_VIEW_PROP_SET':
|
|
|
|
|
2022-04-17 13:41:27 +02:00
|
|
|
if (action.name !== 'html') {
|
|
|
|
(draft.plugins[action.pluginId].views[action.id] as any)[action.name] = action.value;
|
|
|
|
} else {
|
|
|
|
draft.pluginHtmlContents[action.pluginId] ??= {};
|
|
|
|
draft.pluginHtmlContents[action.pluginId][action.id] = action.value;
|
|
|
|
}
|
2020-10-09 19:35:46 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'PLUGIN_VIEW_PROP_PUSH':
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
(draft.plugins[action.pluginId].views[action.id] as any)[action.name].push(action.value);
|
2020-10-09 19:35:46 +02:00
|
|
|
break;
|
|
|
|
|
2023-09-18 18:40:36 +02:00
|
|
|
case 'PLUGIN_All_STARTED_SET':
|
|
|
|
|
|
|
|
draft.allPluginsStarted = action.value;
|
|
|
|
break;
|
|
|
|
|
2020-10-21 01:23:55 +02:00
|
|
|
case 'PLUGIN_CONTENT_SCRIPTS_ADD': {
|
|
|
|
|
|
|
|
const type = action.contentScript.type;
|
2020-11-13 19:09:28 +02:00
|
|
|
if (!draft.plugins[action.pluginId].contentScripts[type]) draft.plugins[action.pluginId].contentScripts[type] = [];
|
2020-10-21 01:23:55 +02:00
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
draft.plugins[action.pluginId].contentScripts[type].push({
|
2020-10-21 01:23:55 +02:00
|
|
|
id: action.contentScript.id,
|
|
|
|
path: action.contentScript.path,
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2024-03-09 13:03:57 +02:00
|
|
|
case 'PLUGIN_UNLOAD':
|
|
|
|
delete draft.plugins[action.pluginId];
|
|
|
|
break;
|
|
|
|
|
2020-10-09 19:35:46 +02:00
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
error.message = `In plugin reducer: ${error.message} Action: ${JSON.stringify(action)}`;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export default reducer;
|