diff --git a/invokeai/frontend/web/src/common/components/IAIDndImage.tsx b/invokeai/frontend/web/src/common/components/IAIDndImage.tsx
index 669a68c88a..e54b4a8872 100644
--- a/invokeai/frontend/web/src/common/components/IAIDndImage.tsx
+++ b/invokeai/frontend/web/src/common/components/IAIDndImage.tsx
@@ -9,7 +9,7 @@ import {
import { useDraggable, useDroppable } from '@dnd-kit/core';
import { useCombinedRefs } from '@dnd-kit/utilities';
import IAIIconButton from 'common/components/IAIIconButton';
-import { IAIImageFallback } from 'common/components/IAIImageFallback';
+import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
import ImageMetadataOverlay from 'common/components/ImageMetadataOverlay';
import { AnimatePresence } from 'framer-motion';
import { ReactElement, SyntheticEvent, useCallback } from 'react';
@@ -53,7 +53,7 @@ const IAIDndImage = (props: IAIDndImageProps) => {
isDropDisabled = false,
isDragDisabled = false,
isUploadDisabled = false,
- fallback = ,
+ fallback = ,
payloadImage,
minSize = 24,
postUploadAction,
diff --git a/invokeai/frontend/web/src/common/components/IAIImageFallback.tsx b/invokeai/frontend/web/src/common/components/IAIImageFallback.tsx
index 3d34fbca9e..03a00d5b1c 100644
--- a/invokeai/frontend/web/src/common/components/IAIImageFallback.tsx
+++ b/invokeai/frontend/web/src/common/components/IAIImageFallback.tsx
@@ -1,10 +1,20 @@
-import { Flex, FlexProps, Spinner, SpinnerProps } from '@chakra-ui/react';
+import {
+ As,
+ Flex,
+ FlexProps,
+ Icon,
+ IconProps,
+ Spinner,
+ SpinnerProps,
+} from '@chakra-ui/react';
+import { ReactElement } from 'react';
+import { FaImage } from 'react-icons/fa';
type Props = FlexProps & {
spinnerProps?: SpinnerProps;
};
-export const IAIImageFallback = (props: Props) => {
+export const IAIImageLoadingFallback = (props: Props) => {
const { spinnerProps, ...rest } = props;
const { sx, ...restFlexProps } = rest;
return (
@@ -25,3 +35,35 @@ export const IAIImageFallback = (props: Props) => {
);
};
+
+type IAINoImageFallbackProps = {
+ flexProps?: FlexProps;
+ iconProps?: IconProps;
+ as?: As;
+};
+
+export const IAINoImageFallback = (props: IAINoImageFallbackProps) => {
+ const { sx: flexSx, ...restFlexProps } = props.flexProps ?? { sx: {} };
+ const { sx: iconSx, ...restIconProps } = props.iconProps ?? { sx: {} };
+ return (
+
+
+
+ );
+};
diff --git a/invokeai/frontend/web/src/features/controlNet/components/ControlNetImagePreview.tsx b/invokeai/frontend/web/src/features/controlNet/components/ControlNetImagePreview.tsx
index a121875f59..217caf9461 100644
--- a/invokeai/frontend/web/src/features/controlNet/components/ControlNetImagePreview.tsx
+++ b/invokeai/frontend/web/src/features/controlNet/components/ControlNetImagePreview.tsx
@@ -11,7 +11,7 @@ import IAIDndImage from 'common/components/IAIDndImage';
import { createSelector } from '@reduxjs/toolkit';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { AnimatePresence, motion } from 'framer-motion';
-import { IAIImageFallback } from 'common/components/IAIImageFallback';
+import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
import IAIIconButton from 'common/components/IAIIconButton';
import { FaUndo } from 'react-icons/fa';
import { useGetImageDTOQuery } from 'services/apiSlice';
@@ -173,7 +173,7 @@ const ControlNetImagePreview = (props: Props) => {
h: 'full',
}}
>
-
+
)}
{controlImage && (
diff --git a/invokeai/frontend/web/src/features/gallery/components/Boards/AddBoardButton.tsx b/invokeai/frontend/web/src/features/gallery/components/Boards/AddBoardButton.tsx
index 284e6558ac..632cebcb33 100644
--- a/invokeai/frontend/web/src/features/gallery/components/Boards/AddBoardButton.tsx
+++ b/invokeai/frontend/web/src/features/gallery/components/Boards/AddBoardButton.tsx
@@ -1,6 +1,5 @@
-import { Flex, Icon, Spinner, Text } from '@chakra-ui/react';
+import IAIButton from 'common/components/IAIButton';
import { useCallback } from 'react';
-import { FaPlus } from 'react-icons/fa';
import { useCreateBoardMutation } from 'services/apiSlice';
const DEFAULT_BOARD_NAME = 'My Board';
@@ -13,38 +12,15 @@ const AddBoardButton = () => {
}, [createBoard]);
return (
-
-
- {isLoading ? (
-
- ) : (
-
- )}
-
- New Board
-
+ Add Board
+
);
};
diff --git a/invokeai/frontend/web/src/features/gallery/components/Boards/AllImagesBoard.tsx b/invokeai/frontend/web/src/features/gallery/components/Boards/AllImagesBoard.tsx
index 51a7609678..51e95b64c4 100644
--- a/invokeai/frontend/web/src/features/gallery/components/Boards/AllImagesBoard.tsx
+++ b/invokeai/frontend/web/src/features/gallery/components/Boards/AllImagesBoard.tsx
@@ -2,12 +2,15 @@ import { Flex, Icon, Text } from '@chakra-ui/react';
import { FaImages } from 'react-icons/fa';
import { boardIdSelected } from '../../store/boardSlice';
import { useDispatch } from 'react-redux';
+import { IAINoImageFallback } from 'common/components/IAIImageFallback';
+import { AnimatePresence } from 'framer-motion';
+import { SelectedItemOverlay } from '../SelectedItemOverlay';
const AllImagesBoard = ({ isSelected }: { isSelected: boolean }) => {
const dispatch = useDispatch();
const handleAllImagesBoardClick = () => {
- dispatch(boardIdSelected(null));
+ dispatch(boardIdSelected());
};
return (
@@ -19,25 +22,34 @@ const AllImagesBoard = ({ isSelected }: { isSelected: boolean }) => {
cursor: 'pointer',
w: 'full',
h: 'full',
- gap: 1,
+ borderRadius: 'base',
}}
onClick={handleAllImagesBoardClick}
>
-
+
+
+ {isSelected && }
+
- All Images
+
+ All Images
+
);
};
diff --git a/invokeai/frontend/web/src/features/gallery/components/Boards/BoardsList.tsx b/invokeai/frontend/web/src/features/gallery/components/Boards/BoardsList.tsx
index 5854c3fe7c..fb68021dee 100644
--- a/invokeai/frontend/web/src/features/gallery/components/Boards/BoardsList.tsx
+++ b/invokeai/frontend/web/src/features/gallery/components/Boards/BoardsList.tsx
@@ -1,12 +1,11 @@
import {
- Box,
- Divider,
+ Collapse,
+ Flex,
Grid,
+ IconButton,
Input,
InputGroup,
InputRightElement,
- Spacer,
- useDisclosure,
} from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
@@ -16,33 +15,36 @@ import {
selectBoardsAll,
setBoardSearchText,
} from 'features/gallery/store/boardSlice';
-import { memo, useEffect, useState } from 'react';
+import { memo, useState } from 'react';
import HoverableBoard from './HoverableBoard';
import { OverlayScrollbarsComponent } from 'overlayscrollbars-react';
import AddBoardButton from './AddBoardButton';
import AllImagesBoard from './AllImagesBoard';
-import { searchBoardsSelector } from '../../store/boardSelectors';
-import { useSelector } from 'react-redux';
-import IAICollapse from '../../../../common/components/IAICollapse';
import { CloseIcon } from '@chakra-ui/icons';
import { useListBoardsQuery } from 'services/apiSlice';
const selector = createSelector(
[selectBoardsAll, boardsSelector],
(boards, boardsState) => {
- const selectedBoard = boards.find(
- (board) => board.board_id === boardsState.selectedBoardId
- );
- return { selectedBoard, searchText: boardsState.searchText };
+ // const selectedBoard = boards.find(
+ // (board) => board.board_id === boardsState.selectedBoardId
+ // );
+ // return { selectedBoard, searchText: boardsState.searchText };
+ const { selectedBoardId, searchText } = boardsState;
+ return { selectedBoardId, searchText };
},
defaultSelectorOptions
);
-const BoardsList = () => {
+type Props = {
+ isOpen: boolean;
+};
+
+const BoardsList = (props: Props) => {
+ const { isOpen } = props;
const dispatch = useAppDispatch();
- const { selectedBoard, searchText } = useAppSelector(selector);
+ const { selectedBoardId, searchText } = useAppSelector(selector);
// const filteredBoards = useSelector(searchBoardsSelector);
- const { isOpen, onToggle } = useDisclosure();
const { data } = useListBoardsQuery({ offset: 0, limit: 8 });
@@ -64,9 +66,18 @@ const BoardsList = () => {
};
return (
-
- <>
-
+
+
+
{
/>
{searchText && searchText.length && (
-
+ }
+ />
)}
-
+
+
{
className="list-container"
sx={{
gap: 2,
- gridTemplateRows: '5rem 5rem',
+ gridTemplateRows: '5.5rem 5.5rem',
gridAutoFlow: 'column dense',
gridAutoColumns: '4rem',
}}
>
- {!searchMode && (
- <>
-
-
- >
- )}
+ {!searchMode && }
{filteredBoards &&
filteredBoards.map((board) => (
))}
- >
-
+
+
);
};
diff --git a/invokeai/frontend/web/src/features/gallery/components/Boards/HoverableBoard.tsx b/invokeai/frontend/web/src/features/gallery/components/Boards/HoverableBoard.tsx
index 71e080ff17..a2c07e4870 100644
--- a/invokeai/frontend/web/src/features/gallery/components/Boards/HoverableBoard.tsx
+++ b/invokeai/frontend/web/src/features/gallery/components/Boards/HoverableBoard.tsx
@@ -1,31 +1,22 @@
import {
+ Badge,
Box,
Editable,
EditableInput,
EditablePreview,
Flex,
+ Image,
MenuItem,
MenuList,
- Text,
} from '@chakra-ui/react';
-import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
+import { useAppDispatch } from 'app/store/storeHooks';
import { memo, useCallback } from 'react';
-import { FaTrash } from 'react-icons/fa';
+import { FaFolder, FaTrash } from 'react-icons/fa';
import { ContextMenu } from 'chakra-ui-contextmenu';
import { BoardDTO, ImageDTO } from 'services/api';
-import { IAIImageFallback } from 'common/components/IAIImageFallback';
+import { IAINoImageFallback } from 'common/components/IAIImageFallback';
import { boardIdSelected } from 'features/gallery/store/boardSlice';
-import {
- boardDeleted,
- boardUpdated,
- imageAddedToBoard,
-} from '../../../../services/thunks/board';
-import { selectImagesAll, selectImagesById } from '../../store/imagesSlice';
-import IAIDndImage from '../../../../common/components/IAIDndImage';
-import { defaultSelectorOptions } from '../../../../app/store/util/defaultMemoizeOptions';
-import { createSelector } from '@reduxjs/toolkit';
-import { RootState } from '../../../../app/store/store';
import {
useAddImageToBoardMutation,
useDeleteBoardMutation,
@@ -33,21 +24,10 @@ import {
useUpdateBoardMutation,
} from 'services/apiSlice';
import { skipToken } from '@reduxjs/toolkit/dist/query';
-
-const coverImageSelector = (imageName: string | undefined) =>
- createSelector(
- [(state: RootState) => state],
- (state) => {
- const coverImage = imageName
- ? selectImagesById(state, imageName)
- : undefined;
-
- return {
- coverImage,
- };
- },
- defaultSelectorOptions
- );
+import { useDroppable } from '@dnd-kit/core';
+import { AnimatePresence } from 'framer-motion';
+import IAIDropOverlay from 'common/components/IAIDropOverlay';
+import { SelectedItemOverlay } from '../SelectedItemOverlay';
interface HoverableBoardProps {
board: BoardDTO;
@@ -94,6 +74,17 @@ const HoverableBoard = memo(({ board, isSelected }: HoverableBoardProps) => {
[addImageToBoard, board_id]
);
+ const {
+ isOver,
+ setNodeRef,
+ active: isDropActive,
+ } = useDroppable({
+ id: `board_droppable_${board_id}`,
+ data: {
+ handleDrop,
+ },
+ });
+
return (
@@ -112,7 +103,6 @@ const HoverableBoard = memo(({ board, isSelected }: HoverableBoardProps) => {
>
{(ref) => (
{
cursor: 'pointer',
w: 'full',
h: 'full',
- gap: 1,
}}
>
- }
- isUploadDisabled={true}
- />
+ {board.cover_image_name && coverImage?.image_url && (
+
+ )}
+ {!(board.cover_image_name && coverImage?.image_url) && (
+
+ )}
+
+ {board.image_count}
+
+
+ {isSelected && }
+
+
+ {isDropActive && }
+
- {
- handleUpdateBoardName(nextValue);
- }}
- >
-
-
+ {
+ handleUpdateBoardName(nextValue);
}}
- />
-
-
- {board.image_count}
-
+ >
+
+
+
+
)}
diff --git a/invokeai/frontend/web/src/features/gallery/components/CurrentImagePreview.tsx b/invokeai/frontend/web/src/features/gallery/components/CurrentImagePreview.tsx
index bff32f1d78..49376b4807 100644
--- a/invokeai/frontend/web/src/features/gallery/components/CurrentImagePreview.tsx
+++ b/invokeai/frontend/web/src/features/gallery/components/CurrentImagePreview.tsx
@@ -14,7 +14,7 @@ import { useAppToaster } from 'app/components/Toaster';
import { imageSelected } from '../store/gallerySlice';
import IAIDndImage from 'common/components/IAIDndImage';
import { ImageDTO } from 'services/api';
-import { IAIImageFallback } from 'common/components/IAIImageFallback';
+import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
import { RootState } from 'app/store/store';
import { selectImagesById } from '../store/imagesSlice';
import { useGetImageDTOQuery } from 'services/apiSlice';
@@ -116,7 +116,7 @@ const CurrentImagePreview = () => {
}
+ fallback={}
isUploadDisabled={true}
/>
diff --git a/invokeai/frontend/web/src/features/gallery/components/ImageGalleryContent.tsx b/invokeai/frontend/web/src/features/gallery/components/ImageGalleryContent.tsx
index 48bd2bde74..8648962c8c 100644
--- a/invokeai/frontend/web/src/features/gallery/components/ImageGalleryContent.tsx
+++ b/invokeai/frontend/web/src/features/gallery/components/ImageGalleryContent.tsx
@@ -1,12 +1,15 @@
import {
Box,
+ Button,
ButtonGroup,
Flex,
FlexProps,
Grid,
Icon,
Text,
+ VStack,
forwardRef,
+ useDisclosure,
} from '@chakra-ui/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import IAIButton from 'common/components/IAIButton';
@@ -54,10 +57,10 @@ import {
selectImagesAll,
} from '../store/imagesSlice';
import { receivedPageOfImages } from 'services/thunks/image';
-import { boardSelector } from '../store/boardSelectors';
-import { boardCreated } from '../../../services/thunks/board';
import BoardsList from './Boards/BoardsList';
-import { selectBoardsById } from '../store/boardSlice';
+import { boardsSelector, selectBoardsById } from '../store/boardSlice';
+import { ChevronUpIcon } from '@chakra-ui/icons';
+import { useListAllBoardsQuery } from 'services/apiSlice';
const itemSelector = createSelector(
[(state: RootState) => state],
@@ -89,7 +92,7 @@ const itemSelector = createSelector(
);
const mainSelector = createSelector(
- [gallerySelector, uiSelector, boardSelector],
+ [gallerySelector, uiSelector, boardsSelector],
(gallery, ui, boards) => {
const {
galleryImageMinimumWidth,
@@ -109,7 +112,7 @@ const mainSelector = createSelector(
shouldUseSingleGalleryColumn,
selectedImage,
galleryView,
- boards,
+ selectedBoardId: boards.selectedBoardId,
};
},
defaultSelectorOptions
@@ -142,12 +145,18 @@ const ImageGalleryContent = () => {
shouldUseSingleGalleryColumn,
selectedImage,
galleryView,
- boards,
+ selectedBoardId,
} = useAppSelector(mainSelector);
- const { items, areMoreAvailable, isLoading, categories, selectedBoard } =
+ const { items, areMoreAvailable, isLoading, categories } =
useAppSelector(itemSelector);
+ const { selectedBoard } = useListAllBoardsQuery(undefined, {
+ selectFromResult: ({ data }) => ({
+ selectedBoard: data?.find((b) => b.board_id === selectedBoardId),
+ }),
+ });
+
const handleLoadMoreImages = useCallback(() => {
dispatch(receivedPageOfImages());
}, [dispatch]);
@@ -159,6 +168,8 @@ const ImageGalleryContent = () => {
return undefined;
}, [areMoreAvailable, handleLoadMoreImages, isLoading]);
+ const { isOpen: isBoardListOpen, onToggle } = useDisclosure();
+
const handleChangeGalleryImageMinimumWidth = (v: number) => {
dispatch(setGalleryImageMinimumWidth(v));
};
@@ -197,50 +208,71 @@ const ImageGalleryContent = () => {
dispatch(setGalleryView('assets'));
}, [dispatch]);
- const handleClickBoardsView = useCallback(() => {
- dispatch(setGalleryView('boards'));
- }, [dispatch]);
-
return (
-
-
-
-
+
+
+ }
+ />
+ }
+ />
+
+ }
- />
- }
- />
-
-
-
- {selectedBoard ? selectedBoard.board_name : 'All Images'}
-
-
-
+ variant="ghost"
+ sx={{
+ w: 'full',
+ justifyContent: 'center',
+ alignItems: 'center',
+ px: 2,
+ _hover: {
+ bg: 'base.800',
+ },
+ }}
+ >
+
+ {selectedBoard ? selectedBoard.board_name : 'All Images'}
+
+
+
{
icon={shouldPinGallery ? : }
/>
-
-
-
+
+
+
-
+
{items.length || areMoreAvailable ? (
<>
@@ -378,7 +410,7 @@ const ImageGalleryContent = () => {
)}
-
+
);
};
diff --git a/invokeai/frontend/web/src/features/gallery/components/SelectedItemOverlay.tsx b/invokeai/frontend/web/src/features/gallery/components/SelectedItemOverlay.tsx
new file mode 100644
index 0000000000..7038b4b64f
--- /dev/null
+++ b/invokeai/frontend/web/src/features/gallery/components/SelectedItemOverlay.tsx
@@ -0,0 +1,26 @@
+import { motion } from 'framer-motion';
+
+export const SelectedItemOverlay = () => (
+
+);
diff --git a/invokeai/frontend/web/src/features/parameters/components/Parameters/ImageToImage/InitialImagePreview.tsx b/invokeai/frontend/web/src/features/parameters/components/Parameters/ImageToImage/InitialImagePreview.tsx
index e4b3a9191e..fbfa00e2a1 100644
--- a/invokeai/frontend/web/src/features/parameters/components/Parameters/ImageToImage/InitialImagePreview.tsx
+++ b/invokeai/frontend/web/src/features/parameters/components/Parameters/ImageToImage/InitialImagePreview.tsx
@@ -10,7 +10,7 @@ import { generationSelector } from 'features/parameters/store/generationSelector
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIDndImage from 'common/components/IAIDndImage';
import { ImageDTO } from 'services/api';
-import { IAIImageFallback } from 'common/components/IAIImageFallback';
+import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
import { useGetImageDTOQuery } from 'services/apiSlice';
import { skipToken } from '@reduxjs/toolkit/dist/query';
@@ -65,7 +65,7 @@ const InitialImagePreview = () => {
image={image}
onDrop={handleDrop}
onReset={handleReset}
- fallback={}
+ fallback={}
postUploadAction={{ type: 'SET_INITIAL_IMAGE' }}
withResetIcon
/>