Maryhipp/disable panels (#3116)

This commit is contained in:
Mary Hipp Rogers 2023-04-05 07:22:22 -07:00 committed by GitHub
commit 8f12ec659c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 106 additions and 58 deletions

View File

@ -1,6 +1,7 @@
import React, { PropsWithChildren } from 'react'; import React, { PropsWithChildren } from 'react';
import { IAIPopoverProps } from '../web/src/common/components/IAIPopover'; import { IAIPopoverProps } from '../web/src/common/components/IAIPopover';
import { IAIIconButtonProps } from '../web/src/common/components/IAIIconButton'; import { IAIIconButtonProps } from '../web/src/common/components/IAIIconButton';
import { InvokeTabName } from 'features/ui/store/tabMap';
export {}; export {};
@ -68,6 +69,8 @@ declare module '@invoke-ai/invoke-ai-ui' {
interface InvokeProps extends PropsWithChildren { interface InvokeProps extends PropsWithChildren {
apiUrl?: string; apiUrl?: string;
disabledPanels?: string[];
disabledTabs?: InvokeTabName[];
} }
declare function Invoke(props: InvokeProps): JSX.Element; declare function Invoke(props: InvokeProps): JSX.Element;

View File

@ -13,16 +13,34 @@ import { Box, Flex, Grid, Portal, useColorMode } from '@chakra-ui/react';
import { APP_HEIGHT, APP_WIDTH } from 'theme/util/constants'; import { APP_HEIGHT, APP_WIDTH } from 'theme/util/constants';
import ImageGalleryPanel from 'features/gallery/components/ImageGalleryPanel'; import ImageGalleryPanel from 'features/gallery/components/ImageGalleryPanel';
import Lightbox from 'features/lightbox/components/Lightbox'; import Lightbox from 'features/lightbox/components/Lightbox';
import { useAppSelector } from './storeHooks'; import { useAppDispatch, useAppSelector } from './storeHooks';
import { PropsWithChildren, useEffect } from 'react'; import { PropsWithChildren, useEffect } from 'react';
import { setDisabledPanels, setDisabledTabs } from 'features/ui/store/uiSlice';
import { InvokeTabName } from 'features/ui/store/tabMap';
keepGUIAlive(); keepGUIAlive();
const App = (props: PropsWithChildren) => { interface Props extends PropsWithChildren {
options: {
disabledPanels: string[];
disabledTabs: InvokeTabName[];
};
}
const App = (props: Props) => {
useToastWatcher(); useToastWatcher();
const currentTheme = useAppSelector((state) => state.ui.currentTheme); const currentTheme = useAppSelector((state) => state.ui.currentTheme);
const { setColorMode } = useColorMode(); const { setColorMode } = useColorMode();
const dispatch = useAppDispatch();
useEffect(() => {
dispatch(setDisabledPanels(props.options.disabledPanels));
}, [dispatch, props.options.disabledPanels]);
useEffect(() => {
dispatch(setDisabledTabs(props.options.disabledTabs));
}, [dispatch, props.options.disabledTabs]);
useEffect(() => { useEffect(() => {
setColorMode(['light'].includes(currentTheme) ? 'light' : 'dark'); setColorMode(['light'].includes(currentTheme) ? 'light' : 'dark');

View File

@ -4,6 +4,7 @@ import { PersistGate } from 'redux-persist/integration/react';
import { store } from './app/store'; import { store } from './app/store';
import { persistor } from './persistor'; import { persistor } from './persistor';
import { OpenAPI } from 'services/api'; import { OpenAPI } from 'services/api';
import { InvokeTabName } from 'features/ui/store/tabMap';
import '@fontsource/inter/100.css'; import '@fontsource/inter/100.css';
import '@fontsource/inter/200.css'; import '@fontsource/inter/200.css';
import '@fontsource/inter/300.css'; import '@fontsource/inter/300.css';
@ -24,9 +25,16 @@ const ThemeLocaleProvider = lazy(() => import('./app/ThemeLocaleProvider'));
interface Props extends PropsWithChildren { interface Props extends PropsWithChildren {
apiUrl?: string; apiUrl?: string;
disabledPanels?: string[];
disabledTabs?: InvokeTabName[];
} }
export default function Component({ apiUrl, children }: Props) { export default function Component({
apiUrl,
disabledPanels = [],
disabledTabs = [],
children,
}: Props) {
useEffect(() => { useEffect(() => {
if (apiUrl) OpenAPI.BASE = apiUrl; if (apiUrl) OpenAPI.BASE = apiUrl;
}, [apiUrl]); }, [apiUrl]);
@ -37,7 +45,7 @@ export default function Component({ apiUrl, children }: Props) {
<PersistGate loading={<Loading />} persistor={persistor}> <PersistGate loading={<Loading />} persistor={persistor}>
<React.Suspense fallback={<Loading showText />}> <React.Suspense fallback={<Loading showText />}>
<ThemeLocaleProvider> <ThemeLocaleProvider>
<App>{children}</App> <App options={{ disabledPanels, disabledTabs }}>{children}</App>
</ThemeLocaleProvider> </ThemeLocaleProvider>
</React.Suspense> </React.Suspense>
</PersistGate> </PersistGate>

View File

@ -21,9 +21,10 @@ type ParametersAccordionsType = {
const ParametersAccordion = (props: ParametersAccordionsType) => { const ParametersAccordion = (props: ParametersAccordionsType) => {
const { accordionInfo } = props; const { accordionInfo } = props;
const openAccordions = useAppSelector( const { system, ui } = useAppSelector((state: RootState) => state);
(state: RootState) => state.system.openAccordions
); const { openAccordions } = system;
const { disabledParameterPanels } = ui;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
@ -39,15 +40,19 @@ const ParametersAccordion = (props: ParametersAccordionsType) => {
Object.keys(accordionInfo).forEach((key) => { Object.keys(accordionInfo).forEach((key) => {
const { header, feature, content, additionalHeaderComponents } = const { header, feature, content, additionalHeaderComponents } =
accordionInfo[key]; accordionInfo[key];
accordionsToRender.push(
<InvokeAccordionItem // do not render if panel is disabled in global state
key={key} if (disabledParameterPanels.indexOf(key) === -1) {
header={header} accordionsToRender.push(
feature={feature} <InvokeAccordionItem
content={content} key={key}
additionalHeaderComponents={additionalHeaderComponents} header={header}
/> feature={feature}
); content={content}
additionalHeaderComponents={additionalHeaderComponents}
/>
);
}
}); });
} }
return accordionsToRender; return accordionsToRender;

View File

@ -45,38 +45,41 @@ const tabIconStyles: ChakraProps['sx'] = {
boxSize: 6, boxSize: 6,
}; };
const tabInfo: InvokeTabInfo[] = [ const buildTabs = (disabledTabs: InvokeTabName[]): InvokeTabInfo[] => {
{ const tabs: InvokeTabInfo[] = [
id: 'txt2img', {
icon: <Icon as={MdTextFields} sx={tabIconStyles} />, id: 'txt2img',
workarea: <TextToImageWorkarea />, icon: <Icon as={MdTextFields} sx={tabIconStyles} />,
}, workarea: <TextToImageWorkarea />,
{ },
id: 'img2img', {
icon: <Icon as={MdPhotoLibrary} sx={tabIconStyles} />, id: 'img2img',
workarea: <ImageToImageWorkarea />, icon: <Icon as={MdPhotoLibrary} sx={tabIconStyles} />,
}, workarea: <ImageToImageWorkarea />,
{ },
id: 'unifiedCanvas', {
icon: <Icon as={MdGridOn} sx={tabIconStyles} />, id: 'unifiedCanvas',
workarea: <UnifiedCanvasWorkarea />, icon: <Icon as={MdGridOn} sx={tabIconStyles} />,
}, workarea: <UnifiedCanvasWorkarea />,
{ },
id: 'nodes', {
icon: <Icon as={MdDeviceHub} sx={tabIconStyles} />, id: 'nodes',
workarea: <NodesWIP />, icon: <Icon as={MdDeviceHub} sx={tabIconStyles} />,
}, workarea: <NodesWIP />,
{ },
id: 'postprocessing', {
icon: <Icon as={MdPhotoFilter} sx={tabIconStyles} />, id: 'postprocessing',
workarea: <PostProcessingWIP />, icon: <Icon as={MdPhotoFilter} sx={tabIconStyles} />,
}, workarea: <PostProcessingWIP />,
{ },
id: 'training', {
icon: <Icon as={MdFlashOn} sx={tabIconStyles} />, id: 'training',
workarea: <TrainingWIP />, icon: <Icon as={MdFlashOn} sx={tabIconStyles} />,
}, workarea: <TrainingWIP />,
]; },
];
return tabs.filter((tab) => !disabledTabs.includes(tab.id));
};
export default function InvokeTabs() { export default function InvokeTabs() {
const activeTab = useAppSelector(activeTabIndexSelector); const activeTab = useAppSelector(activeTabIndexSelector);
@ -85,13 +88,10 @@ export default function InvokeTabs() {
(state: RootState) => state.lightbox.isLightboxOpen (state: RootState) => state.lightbox.isLightboxOpen
); );
const shouldPinGallery = useAppSelector( const { shouldPinGallery, disabledTabs, shouldPinParametersPanel } =
(state: RootState) => state.ui.shouldPinGallery useAppSelector((state: RootState) => state.ui);
);
const shouldPinParametersPanel = useAppSelector( const activeTabs = buildTabs(disabledTabs);
(state: RootState) => state.ui.shouldPinParametersPanel
);
const { t } = useTranslation(); const { t } = useTranslation();
@ -142,7 +142,7 @@ export default function InvokeTabs() {
const tabs = useMemo( const tabs = useMemo(
() => () =>
tabInfo.map((tab) => ( activeTabs.map((tab) => (
<Tooltip <Tooltip
key={tab.id} key={tab.id}
hasArrow hasArrow
@ -157,13 +157,13 @@ export default function InvokeTabs() {
</Tab> </Tab>
</Tooltip> </Tooltip>
)), )),
[t] [t, activeTabs]
); );
const tabPanels = useMemo( const tabPanels = useMemo(
() => () =>
tabInfo.map((tab) => <TabPanel key={tab.id}>{tab.workarea}</TabPanel>), activeTabs.map((tab) => <TabPanel key={tab.id}>{tab.workarea}</TabPanel>),
[] [activeTabs]
); );
return ( return (

View File

@ -18,6 +18,8 @@ const initialtabsState: UIState = {
addNewModelUIOption: null, addNewModelUIOption: null,
shouldPinGallery: true, shouldPinGallery: true,
shouldShowGallery: true, shouldShowGallery: true,
disabledParameterPanels: [],
disabledTabs: [],
}; };
const initialState: UIState = initialtabsState; const initialState: UIState = initialtabsState;
@ -90,6 +92,12 @@ export const uiSlice = createSlice({
state.shouldShowParametersPanel = true; state.shouldShowParametersPanel = true;
} }
}, },
setDisabledPanels: (state, action: PayloadAction<string[]>) => {
state.disabledParameterPanels = action.payload;
},
setDisabledTabs: (state, action: PayloadAction<InvokeTabName[]>) => {
state.disabledTabs = action.payload;
},
}, },
extraReducers(builder) { extraReducers(builder) {
builder.addCase(initialImageSelected, (state) => { builder.addCase(initialImageSelected, (state) => {
@ -118,6 +126,8 @@ export const {
togglePinParametersPanel, togglePinParametersPanel,
toggleParametersPanel, toggleParametersPanel,
toggleGalleryPanel, toggleGalleryPanel,
setDisabledPanels,
setDisabledTabs,
} = uiSlice.actions; } = uiSlice.actions;
export default uiSlice.reducer; export default uiSlice.reducer;

View File

@ -1,3 +1,5 @@
import { InvokeTabName } from './tabMap';
export type AddNewModelType = 'ckpt' | 'diffusers' | null; export type AddNewModelType = 'ckpt' | 'diffusers' | null;
export interface UIState { export interface UIState {
@ -13,4 +15,6 @@ export interface UIState {
addNewModelUIOption: AddNewModelType; addNewModelUIOption: AddNewModelType;
shouldPinGallery: boolean; shouldPinGallery: boolean;
shouldShowGallery: boolean; shouldShowGallery: boolean;
disabledParameterPanels: string[];
disabledTabs: InvokeTabName[];
} }