diff --git a/frontend/src/components/App/PluginSettings/PluginSettings.stories.tsx b/frontend/src/components/App/PluginSettings/PluginSettings.stories.tsx index 3ed2ae0f04..700ed48a9e 100644 --- a/frontend/src/components/App/PluginSettings/PluginSettings.stories.tsx +++ b/frontend/src/components/App/PluginSettings/PluginSettings.stories.tsx @@ -1,4 +1,5 @@ import { Meta, StoryFn } from '@storybook/react'; +import { PluginInfo } from '../../../plugin/pluginsSlice'; import { TestContext } from '../../../test'; import { PluginSettingsPure, PluginSettingsPureProps } from './PluginSettings'; @@ -40,6 +41,17 @@ function createDemoData(arrSize: number, useHomepage?: boolean) { return pluginArr; } +/** + * create demo data for pluginsEnabledList + */ +function createDemoEnabledList(arr: PluginInfo[]): Record { + const enabledList = arr.reduce((acc, p) => { + acc[p.name] = !!p.isEnabled; + return acc; + }, {} as Record); + return enabledList; +} + /** * Creation of data arrays ranging from 0 to 50 to demo state of empty, few, many, and large numbers of data objects. * NOTE: The numbers used are up to the users preference. @@ -55,6 +67,7 @@ const demoEmpty = createDemoData(0); export const FewItems = Template.bind({}); FewItems.args = { plugins: demoFew, + pluginsEnabledList: createDemoEnabledList(demoFew), onSave: plugins => { console.log('demo few', plugins); }, @@ -63,12 +76,14 @@ FewItems.args = { export const Empty = Template.bind({}); Empty.args = { plugins: demoEmpty, + pluginsEnabledList: createDemoEnabledList(demoEmpty), }; /** NOTE: The save button will load by default on plugin page regardless of data */ export const DefaultSaveEnable = Template.bind({}); DefaultSaveEnable.args = { plugins: demoFewSaveEnable, + pluginsEnabledList: createDemoEnabledList(demoFewSaveEnable), onSave: plugins => { console.log('demo few', plugins); }, @@ -78,6 +93,7 @@ DefaultSaveEnable.args = { export const ManyItems = Template.bind({}); ManyItems.args = { plugins: demoMany, + pluginsEnabledList: createDemoEnabledList(demoMany), onSave: plugins => { console.log('demo many', plugins); }, @@ -86,6 +102,7 @@ ManyItems.args = { export const MoreItems = Template.bind({}); MoreItems.args = { plugins: demoMore, + pluginsEnabledList: createDemoEnabledList(demoMore), onSave: plugins => { console.log('demo more', plugins); }, @@ -94,6 +111,7 @@ MoreItems.args = { export const EmptyHomepageItems = Template.bind({}); EmptyHomepageItems.args = { plugins: demoHomepageEmpty, + pluginsEnabledList: createDemoEnabledList(demoHomepageEmpty), onSave: (plugins: any) => { console.log('Empty Homepage', plugins); }, diff --git a/frontend/src/components/App/PluginSettings/PluginSettings.tsx b/frontend/src/components/App/PluginSettings/PluginSettings.tsx index 3135cb314c..870bf7343e 100644 --- a/frontend/src/components/App/PluginSettings/PluginSettings.tsx +++ b/frontend/src/components/App/PluginSettings/PluginSettings.tsx @@ -3,12 +3,12 @@ import Box from '@mui/material/Box'; import Button from '@mui/material/Button'; import Link from '@mui/material/Link'; import { MRT_Row } from 'material-react-table'; -import { useEffect, useState } from 'react'; +import { useMemo, useState } from 'react'; import { useTranslation } from 'react-i18next'; import { useDispatch } from 'react-redux'; import helpers from '../../../helpers'; import { useFilterFunc } from '../../../lib/util'; -import { PluginInfo, reloadPage, setPluginSettings } from '../../../plugin/pluginsSlice'; +import { PluginInfo, reloadPage, setEnablePlugin } from '../../../plugin/pluginsSlice'; import { useTypedSelector } from '../../../redux/reducers/reducers'; import { Link as HeadlampLink, SectionBox, Table } from '../../common'; import SectionFilterHeader from '../../common/SectionFilterHeader'; @@ -24,6 +24,7 @@ import SectionFilterHeader from '../../common/SectionFilterHeader'; */ export interface PluginSettingsPureProps { plugins: PluginInfo[]; + pluginsEnabledList: Record; onSave: (plugins: PluginInfo[]) => void; saveAlwaysEnable?: boolean; } @@ -92,72 +93,66 @@ const EnableSwitch = (props: SwitchProps) => { /** PluginSettingsPure is the main component to where we render the plugin data. */ export function PluginSettingsPure(props: PluginSettingsPureProps) { + const dispatch = useDispatch(); + const { t } = useTranslation(['translation']); /** Plugin arr to be rendered to the page from prop data */ - const pluginArr: any = props.plugins ? props.plugins : []; - - /** enableSave state enables the save button when changes are made to the plugin list */ - const [enableSave, setEnableSave] = useState(false); - - /** pluginChanges state is the array of plugin data and any current changes made by the user to a plugin's "Enable" field via toggler */ - const [pluginChanges, setPluginChanges] = useState(() => pluginArr.map((p: any) => p)); + const [pluginArr, setPluginArr] = useState(props.plugins); /** - * useEffect to control the rendering of the save button. - * By default, the enableSave is set to false. - * If props.plugins matches pluginChanges enableSave is set to false, disabling the save button. + * pendingPluginsEnabled is either from the local storage or the prop data */ - useEffect(() => { - /** This matcher function compares the fields of name and isEnabled of each object in props.plugins to each object in pluginChanges */ - function matcher(objA: PluginInfo, objB: PluginInfo) { - return objA.name === objB.name && objA.isEnabled === objB.isEnabled; - } - - /** - * arrayComp returns true if each object in both arrays are identical by name and isEnabled. - * If both arrays are identical in this scope, then no changes need to be saved. - * If they do not match, there are changes in the pluginChanges array that can be saved and thus enableSave should be enabled. - */ - const arrayComp = props.plugins.every((val, key) => matcher(val, pluginChanges[key])); + const [pendingPluginsEnabled, setPendingPluginsEnabled] = useState>( + props.pluginsEnabledList + ); - /** For storybook usage, determines if the save button should be enabled by default */ - if (props.saveAlwaysEnable) { - setEnableSave(true); - } else { - if (arrayComp) { - setEnableSave(false); - } - if (!arrayComp) { - setEnableSave(true); - } - } - }, [pluginChanges]); + /** + * useEffect / useMemo to update the pluginArr state with the prop data + */ + const enableSave = useMemo(() => { + return !pluginArr.every((plugin: { name: string }) => { + // if the user selected state is the same as saved + return ( + Boolean(props.pluginsEnabledList[plugin.name]) === + Boolean(pendingPluginsEnabled[plugin.name]) + ); + }); + }, [pendingPluginsEnabled, pluginArr, props.pluginsEnabledList]); /** * onSaveButton function to be called once the user clicks the Save button. * This function then takes the current state of the pluginChanges array and inputs it to the onSave prop function. */ function onSaveButtonHandler() { - props.onSave(pluginChanges); + dispatch(setEnablePlugin(pendingPluginsEnabled)); + dispatch(reloadPage()); } /** - * On change function handler to control the enableSave state and update the pluginChanges state. - * This function is called on every plugin toggle action and recreates the state for pluginChanges. + * On change function handler to control the enableSave state and update the pluginArr state. + * This function is called on every plugin toggle action and recreates the state for pluginArr. * Once the user clicks a toggle, the Save button is also rendered via setEnableSave. */ - function switchChangeHanlder(plug: { name: any }) { + function switchChangeHandler(plug: { name: any }) { const plugName = plug.name; - setPluginChanges((currentInfo: any[]) => - currentInfo.map((p: { name: any; isEnabled: any }) => { + setPluginArr([ + ...pluginArr.map(p => { if (p.name === plugName) { - return { ...p, isEnabled: !p.isEnabled }; + return { + ...p, + isEnabled: !p.isEnabled, + }; } return p; - }) - ); + }), + ]); + + setPendingPluginsEnabled({ + ...pendingPluginsEnabled, + [plugName]: !pendingPluginsEnabled[plugName], + }); } return ( @@ -220,7 +215,9 @@ export function PluginSettingsPure(props: PluginSettingsPureProps) { if (plugin.isCompatible === false) { return t('translation|Incompatible'); } - return plugin.isEnabled ? t('translation|Enabled') : t('translation|Disabled'); + return pendingPluginsEnabled[plugin.name] + ? t('translation|Enabled') + : t('translation|Disabled'); }, }, { @@ -232,8 +229,8 @@ export function PluginSettingsPure(props: PluginSettingsPureProps) { return ( switchChangeHanlder(plugin)} + checked={pendingPluginsEnabled[plugin.name]} + onChange={() => switchChangeHandler(plugin)} color="primary" name={plugin.name} /> @@ -245,7 +242,7 @@ export function PluginSettingsPure(props: PluginSettingsPureProps) { ] // remove the enable column if we're not in app mode .filter(el => !(el.header === t('translation|Enable') && !helpers.isElectron()))} - data={pluginChanges} + data={pluginArr} filterFunction={useFilterFunc(['.name'])} /> @@ -269,13 +266,40 @@ export function PluginSettingsPure(props: PluginSettingsPureProps) { export default function PluginSettings() { const dispatch = useDispatch(); - const pluginSettings = useTypedSelector(state => state.plugins.pluginSettings); + const pluginData = useTypedSelector(state => state.plugins.pluginData); + + /** + * We need to search for the local storage before using it in the slice later + */ + const localEnabledList = localStorage.getItem('headlampPluginSettings'); + + let pluginsEnabledList; + + /** + * If `localEnabledList` exists, parse it and assign it to `pluginsEnabledList`. + * This indicates that previous plugin settings have been saved and can be used. + * + * If `localEnabledList` does not exist, it means the settings are not initialized + * and no previous plugin settings have been saved. In this case, default the plugins + * to being disabled to allow users to turn on their desired plugins. + */ + if (localEnabledList) { + pluginsEnabledList = JSON.parse(localEnabledList) as Record; + } else { + pluginsEnabledList = pluginData.reduce((acc, p) => { + acc[p.name] = !p.isEnabled; + return acc; + }, {} as Record); + + dispatch(setEnablePlugin(pluginsEnabledList)); + dispatch(reloadPage()); + } return ( { - dispatch(setPluginSettings(plugins)); + plugins={pluginData} + pluginsEnabledList={pluginsEnabledList} + onSave={() => { dispatch(reloadPage()); }} /> diff --git a/frontend/src/components/App/PluginSettings/PluginSettingsDetails.tsx b/frontend/src/components/App/PluginSettings/PluginSettingsDetails.tsx index 3b78df9fc8..218063828c 100644 --- a/frontend/src/components/App/PluginSettings/PluginSettingsDetails.tsx +++ b/frontend/src/components/App/PluginSettings/PluginSettingsDetails.tsx @@ -54,7 +54,7 @@ const PluginSettingsDetailsInitializer = (props: { plugin: PluginInfo }) => { }; export default function PluginSettingsDetails() { - const pluginSettings = useTypedSelector(state => state.plugins.pluginSettings); + const pluginSettings = useTypedSelector(state => state.plugins.pluginData); const { name } = useParams<{ name: string }>(); const plugin = useMemo(() => { diff --git a/frontend/src/components/App/PluginSettings/__snapshots__/PluginSettings.DefaultSaveEnable.stories.storyshot b/frontend/src/components/App/PluginSettings/__snapshots__/PluginSettings.DefaultSaveEnable.stories.storyshot index b128024ebb..9684f1383c 100644 --- a/frontend/src/components/App/PluginSettings/__snapshots__/PluginSettings.DefaultSaveEnable.stories.storyshot +++ b/frontend/src/components/App/PluginSettings/__snapshots__/PluginSettings.DefaultSaveEnable.stories.storyshot @@ -788,19 +788,5 @@ -
- -
\ No newline at end of file diff --git a/frontend/src/plugin/Plugins.tsx b/frontend/src/plugin/Plugins.tsx index 46521ec7b5..8a28888f72 100644 --- a/frontend/src/plugin/Plugins.tsx +++ b/frontend/src/plugin/Plugins.tsx @@ -8,7 +8,7 @@ import helpers from '../helpers'; import { UI_INITIALIZE_PLUGIN_VIEWS } from '../redux/actions/actions'; import { useTypedSelector } from '../redux/reducers/reducers'; import { fetchAndExecutePlugins } from './index'; -import { pluginsLoaded, setPluginSettings } from './pluginsSlice'; +import { pluginsLoaded, setPluginData } from './pluginsSlice'; /** * For discovering and executing plugins. @@ -26,7 +26,8 @@ export default function Plugins() { const history = useHistory(); const { t } = useTranslation(); - const settingsPlugins = useTypedSelector(state => state.plugins.pluginSettings); + const settingsPlugins = useTypedSelector(state => state.plugins.pluginData); + const enabledPlugins = useTypedSelector(state => state.plugins.enabledPlugins); // only run on first load useEffect(() => { @@ -34,8 +35,9 @@ export default function Plugins() { fetchAndExecutePlugins( settingsPlugins, + enabledPlugins, updatedSettingsPackages => { - dispatch(setPluginSettings(updatedSettingsPackages)); + dispatch(setPluginData(updatedSettingsPackages)); }, incompatiblePlugins => { const pluginList = Object.values(incompatiblePlugins) diff --git a/frontend/src/plugin/index.ts b/frontend/src/plugin/index.ts index d42a63f672..98678aa7a1 100644 --- a/frontend/src/plugin/index.ts +++ b/frontend/src/plugin/index.ts @@ -137,6 +137,7 @@ export async function initializePlugins() { export function filterSources( sources: string[], packageInfos: PluginInfo[], + enabledPlugins: Record, appMode: boolean, compatibleVersion: string, settingsPackages?: PluginInfo[] @@ -161,9 +162,8 @@ export function filterSources( // settingsPackages might have a different order or length than packageInfos // If it's not in the settings don't enable the plugin. - const enabledInSettings = - settingsPackages[settingsPackages.findIndex(x => x.name === packageInfo.name)]?.isEnabled === - true; + + const enabledInSettings = enabledPlugins[packageInfo.name]; return enabledInSettings; }); @@ -242,6 +242,7 @@ export function updateSettingsPackages( */ export async function fetchAndExecutePlugins( settingsPackages: PluginInfo[], + enabledPlugins: Record, onSettingsChange: (plugins: PluginInfo[]) => void, onIncompatible: (plugins: Record) => void ) { @@ -298,6 +299,7 @@ export async function fetchAndExecutePlugins( const { sourcesToExecute, incompatiblePlugins } = filterSources( sources, packageInfos, + enabledPlugins, helpers.isElectron(), compatibleHeadlampPluginVersion, updatedSettingsPackages diff --git a/frontend/src/plugin/pluginSlice.test.tsx b/frontend/src/plugin/pluginSlice.test.tsx index 805f6f31e3..3ba719ece1 100644 --- a/frontend/src/plugin/pluginSlice.test.tsx +++ b/frontend/src/plugin/pluginSlice.test.tsx @@ -11,7 +11,8 @@ const initialState: PluginsState = { /** Once the plugins have been fetched and executed. */ loaded: false, /** If plugin settings are saved use those. */ - pluginSettings: JSON.parse(localStorage.getItem('headlampPluginSettings') || '[]'), + enabledPlugins: JSON.parse(localStorage.getItem('headlampPluginSettings') || '{}'), + pluginData: [], }; // Mock React component for testing @@ -24,7 +25,7 @@ describe('pluginsSlice reducers', () => { const existingPluginName = 'test-plugin'; const initialStateWithPlugin: PluginsState = { ...initialState, - pluginSettings: [ + pluginData: [ { name: existingPluginName, settingsComponent: undefined, @@ -41,15 +42,15 @@ describe('pluginsSlice reducers', () => { const newState = pluginsSlice.reducer(initialStateWithPlugin, action); - expect(newState.pluginSettings[0].settingsComponent).toBeDefined(); - expect(newState.pluginSettings[0].displaySettingsComponentWithSaveButton).toBe(true); + expect(newState.pluginData[0].settingsComponent).toBeDefined(); + expect(newState.pluginData[0].displaySettingsComponentWithSaveButton).toBe(true); }); test('should not modify state when plugin name does not match any existing plugin', () => { const nonExistingPluginName = 'non-existing-plugin'; const initialStateWithPlugin: PluginsState = { ...initialState, - pluginSettings: [ + pluginData: [ { name: 'existing-plugin', settingsComponent: undefined, diff --git a/frontend/src/plugin/pluginsSlice.ts b/frontend/src/plugin/pluginsSlice.ts index 500e5b31ce..02770cf564 100644 --- a/frontend/src/plugin/pluginsSlice.ts +++ b/frontend/src/plugin/pluginsSlice.ts @@ -86,14 +86,17 @@ export type PluginInfo = { export interface PluginsState { /** Have plugins finished executing? */ loaded: boolean; + /** Information for the plugin's enable */ + enabledPlugins: Record; /** Information stored by settings about plugins. */ - pluginSettings: PluginInfo[]; + pluginData: PluginInfo[]; } const initialState: PluginsState = { /** Once the plugins have been fetched and executed. */ loaded: false, /** If plugin settings are saved use those. */ - pluginSettings: JSON.parse(localStorage.getItem('headlampPluginSettings') || '[]'), + enabledPlugins: JSON.parse(localStorage.getItem('headlampPluginSettings') || '[]'), + pluginData: [], }; export const pluginsSlice = createSlice({ @@ -103,13 +106,16 @@ export const pluginsSlice = createSlice({ pluginsLoaded(state) { state.loaded = true; }, - /** - * Save the plugin settings. To both the store, and localStorage. - */ - setPluginSettings(state, action: PayloadAction) { - state.pluginSettings = action.payload; + + /** Updates the local storage for plugin enable settings */ + setEnablePlugin(state, action: PayloadAction>) { + state.enabledPlugins = action.payload; localStorage.setItem('headlampPluginSettings', JSON.stringify(action.payload)); }, + + setPluginData(state, action: PayloadAction) { + state.pluginData = action.payload; + }, /** Reloads the browser page */ reloadPage() { window.location.reload(); @@ -126,7 +132,7 @@ export const pluginsSlice = createSlice({ }> ) { const { name, component, displaySaveButton } = action.payload; - state.pluginSettings = state.pluginSettings.map(plugin => { + state.pluginData = state.pluginData.map(plugin => { if (plugin.name === name) { return { ...plugin, @@ -140,7 +146,12 @@ export const pluginsSlice = createSlice({ }, }); -export const { pluginsLoaded, setPluginSettings, setPluginSettingsComponent, reloadPage } = - pluginsSlice.actions; +export const { + pluginsLoaded, + setEnablePlugin, + setPluginData, + setPluginSettingsComponent, + reloadPage, +} = pluginsSlice.actions; export default pluginsSlice.reducer;