feat: add multi-select to gallery

multi-select actions include:
- drag to board to move all to that board
- right click to add all to board or delete all

backend changes:
- add routes for changing board for list of image names, deleting list of images
- change image-specific routes to `images/i/{image_name}` to not clobber other routes (like `images/upload`, `images/delete`)
- subclass pydantic `BaseModel` as `BaseModelExcludeNull`, which excludes null values when calling `dict()` on the model. this fixes inconsistent types related to JSON parsing null values into `null` instead of `undefined`
- remove `board_id` from `remove_image_from_board`

frontend changes:
- multi-selection stuff uses `ImageDTO[]` as payloads, for dnd and other mutations. this gives us access to image `board_id`s when hitting routes, and enables efficient cache updates.
- consolidate change board and delete image modals to handle single and multiples
- board totals are now re-fetched on mutation and not kept in sync manually - was way too tedious to do this
- fixed warning about nested `<p>` elements
- closes #4088 , need to handle case when `autoAddBoardId` is `"none"`
- add option to show gallery image delete button on every gallery image

frontend refactors/organisation:
- make typegen script js instead of ts
- enable `noUncheckedIndexedAccess` to help avoid bugs when indexing into arrays, many small changes needed to satisfy TS after this
- move all image-related endpoints into `endpoints/images.ts`, its a big file now, but this fixes a number of circular dependency issues that were otherwise felt impossible to resolve
This commit is contained in:
psychedelicious 2023-07-31 18:16:52 +10:00
parent e080fd1e08
commit bf94412d14
116 changed files with 2470 additions and 2181 deletions

View File

@ -1,24 +1,30 @@
from fastapi import Body, HTTPException, Path, Query from fastapi import Body, HTTPException
from fastapi.routing import APIRouter from fastapi.routing import APIRouter
from invokeai.app.services.board_record_storage import BoardRecord, BoardChanges from pydantic import BaseModel, Field
from invokeai.app.services.image_record_storage import OffsetPaginatedResults
from invokeai.app.services.models.board_record import BoardDTO
from invokeai.app.services.models.image_record import ImageDTO
from ..dependencies import ApiDependencies from ..dependencies import ApiDependencies
board_images_router = APIRouter(prefix="/v1/board_images", tags=["boards"]) board_images_router = APIRouter(prefix="/v1/board_images", tags=["boards"])
class AddImagesToBoardResult(BaseModel):
board_id: str = Field(description="The id of the board the images were added to")
added_image_names: list[str] = Field(description="The image names that were added to the board")
class RemoveImagesFromBoardResult(BaseModel):
removed_image_names: list[str] = Field(description="The image names that were removed from their board")
@board_images_router.post( @board_images_router.post(
"/", "/",
operation_id="create_board_image", operation_id="add_image_to_board",
responses={ responses={
201: {"description": "The image was added to a board successfully"}, 201: {"description": "The image was added to a board successfully"},
}, },
status_code=201, status_code=201,
) )
async def create_board_image( async def add_image_to_board(
board_id: str = Body(description="The id of the board to add to"), board_id: str = Body(description="The id of the board to add to"),
image_name: str = Body(description="The name of the image to add"), image_name: str = Body(description="The name of the image to add"),
): ):
@ -29,26 +35,78 @@ async def create_board_image(
) )
return result return result
except Exception as e: except Exception as e:
raise HTTPException(status_code=500, detail="Failed to add to board") raise HTTPException(status_code=500, detail="Failed to add image to board")
@board_images_router.delete( @board_images_router.delete(
"/", "/",
operation_id="remove_board_image", operation_id="remove_image_from_board",
responses={ responses={
201: {"description": "The image was removed from the board successfully"}, 201: {"description": "The image was removed from the board successfully"},
}, },
status_code=201, status_code=201,
) )
async def remove_board_image( async def remove_image_from_board(
board_id: str = Body(description="The id of the board"), image_name: str = Body(description="The name of the image to remove", embed=True),
image_name: str = Body(description="The name of the image to remove"),
): ):
"""Deletes a board_image""" """Removes an image from its board, if it had one"""
try: try:
result = ApiDependencies.invoker.services.board_images.remove_image_from_board( result = ApiDependencies.invoker.services.board_images.remove_image_from_board(image_name=image_name)
board_id=board_id, image_name=image_name
)
return result return result
except Exception as e: except Exception as e:
raise HTTPException(status_code=500, detail="Failed to update board") raise HTTPException(status_code=500, detail="Failed to remove image from board")
@board_images_router.post(
"/batch",
operation_id="add_images_to_board",
responses={
201: {"description": "Images were added to board successfully"},
},
status_code=201,
response_model=AddImagesToBoardResult,
)
async def add_images_to_board(
board_id: str = Body(description="The id of the board to add to"),
image_names: list[str] = Body(description="The names of the images to add", embed=True),
) -> AddImagesToBoardResult:
"""Adds a list of images to a board"""
try:
added_image_names: list[str] = []
for image_name in image_names:
try:
ApiDependencies.invoker.services.board_images.add_image_to_board(
board_id=board_id, image_name=image_name
)
added_image_names.append(image_name)
except:
pass
return AddImagesToBoardResult(board_id=board_id, added_image_names=added_image_names)
except Exception as e:
raise HTTPException(status_code=500, detail="Failed to add images to board")
@board_images_router.post(
"/batch/delete",
operation_id="remove_images_from_board",
responses={
201: {"description": "Images were removed from board successfully"},
},
status_code=201,
response_model=RemoveImagesFromBoardResult,
)
async def remove_images_from_board(
image_names: list[str] = Body(description="The names of the images to remove", embed=True),
) -> RemoveImagesFromBoardResult:
"""Removes a list of images from their board, if they had one"""
try:
removed_image_names: list[str] = []
for image_name in image_names:
try:
ApiDependencies.invoker.services.board_images.remove_image_from_board(image_name=image_name)
removed_image_names.append(image_name)
except:
pass
return RemoveImagesFromBoardResult(removed_image_names=removed_image_names)
except Exception as e:
raise HTTPException(status_code=500, detail="Failed to remove images from board")

View File

@ -5,6 +5,7 @@ from fastapi import Body, HTTPException, Path, Query, Request, Response, UploadF
from fastapi.responses import FileResponse from fastapi.responses import FileResponse
from fastapi.routing import APIRouter from fastapi.routing import APIRouter
from PIL import Image from PIL import Image
from pydantic import BaseModel, Field
from invokeai.app.invocations.metadata import ImageMetadata from invokeai.app.invocations.metadata import ImageMetadata
from invokeai.app.models.image import ImageCategory, ResourceOrigin from invokeai.app.models.image import ImageCategory, ResourceOrigin
@ -25,7 +26,7 @@ IMAGE_MAX_AGE = 31536000
@images_router.post( @images_router.post(
"/", "/upload",
operation_id="upload_image", operation_id="upload_image",
responses={ responses={
201: {"description": "The image was uploaded successfully"}, 201: {"description": "The image was uploaded successfully"},
@ -77,7 +78,7 @@ async def upload_image(
raise HTTPException(status_code=500, detail="Failed to create image") raise HTTPException(status_code=500, detail="Failed to create image")
@images_router.delete("/{image_name}", operation_id="delete_image") @images_router.delete("/i/{image_name}", operation_id="delete_image")
async def delete_image( async def delete_image(
image_name: str = Path(description="The name of the image to delete"), image_name: str = Path(description="The name of the image to delete"),
) -> None: ) -> None:
@ -103,7 +104,7 @@ async def clear_intermediates() -> int:
@images_router.patch( @images_router.patch(
"/{image_name}", "/i/{image_name}",
operation_id="update_image", operation_id="update_image",
response_model=ImageDTO, response_model=ImageDTO,
) )
@ -120,7 +121,7 @@ async def update_image(
@images_router.get( @images_router.get(
"/{image_name}", "/i/{image_name}",
operation_id="get_image_dto", operation_id="get_image_dto",
response_model=ImageDTO, response_model=ImageDTO,
) )
@ -136,7 +137,7 @@ async def get_image_dto(
@images_router.get( @images_router.get(
"/{image_name}/metadata", "/i/{image_name}/metadata",
operation_id="get_image_metadata", operation_id="get_image_metadata",
response_model=ImageMetadata, response_model=ImageMetadata,
) )
@ -152,7 +153,7 @@ async def get_image_metadata(
@images_router.get( @images_router.get(
"/{image_name}/full", "/i/{image_name}/full",
operation_id="get_image_full", operation_id="get_image_full",
response_class=Response, response_class=Response,
responses={ responses={
@ -187,7 +188,7 @@ async def get_image_full(
@images_router.get( @images_router.get(
"/{image_name}/thumbnail", "/i/{image_name}/thumbnail",
operation_id="get_image_thumbnail", operation_id="get_image_thumbnail",
response_class=Response, response_class=Response,
responses={ responses={
@ -216,7 +217,7 @@ async def get_image_thumbnail(
@images_router.get( @images_router.get(
"/{image_name}/urls", "/i/{image_name}/urls",
operation_id="get_image_urls", operation_id="get_image_urls",
response_model=ImageUrlsDTO, response_model=ImageUrlsDTO,
) )
@ -265,3 +266,24 @@ async def list_image_dtos(
) )
return image_dtos return image_dtos
class DeleteImagesFromListResult(BaseModel):
deleted_images: list[str]
@images_router.post("/delete", operation_id="delete_images_from_list", response_model=DeleteImagesFromListResult)
async def delete_images_from_list(
image_names: list[str] = Body(description="The list of names of images to delete", embed=True),
) -> DeleteImagesFromListResult:
try:
deleted_images: list[str] = []
for image_name in image_names:
try:
ApiDependencies.invoker.services.images.delete(image_name)
deleted_images.append(image_name)
except:
pass
return DeleteImagesFromListResult(deleted_images=deleted_images)
except Exception as e:
raise HTTPException(status_code=500, detail="Failed to delete images")

View File

@ -1,6 +1,6 @@
from typing import Literal, Optional, Union from typing import Literal, Optional, Union
from pydantic import BaseModel, Field from pydantic import Field
from invokeai.app.invocations.baseinvocation import ( from invokeai.app.invocations.baseinvocation import (
BaseInvocation, BaseInvocation,
@ -10,16 +10,17 @@ from invokeai.app.invocations.baseinvocation import (
) )
from invokeai.app.invocations.controlnet_image_processors import ControlField from invokeai.app.invocations.controlnet_image_processors import ControlField
from invokeai.app.invocations.model import LoRAModelField, MainModelField, VAEModelField from invokeai.app.invocations.model import LoRAModelField, MainModelField, VAEModelField
from invokeai.app.util.model_exclude_null import BaseModelExcludeNull
class LoRAMetadataField(BaseModel): class LoRAMetadataField(BaseModelExcludeNull):
"""LoRA metadata for an image generated in InvokeAI.""" """LoRA metadata for an image generated in InvokeAI."""
lora: LoRAModelField = Field(description="The LoRA model") lora: LoRAModelField = Field(description="The LoRA model")
weight: float = Field(description="The weight of the LoRA model") weight: float = Field(description="The weight of the LoRA model")
class CoreMetadata(BaseModel): class CoreMetadata(BaseModelExcludeNull):
"""Core generation metadata for an image generated in InvokeAI.""" """Core generation metadata for an image generated in InvokeAI."""
generation_mode: str = Field( generation_mode: str = Field(
@ -70,7 +71,7 @@ class CoreMetadata(BaseModel):
refiner_start: Union[float, None] = Field(default=None, description="The start value used for refiner denoising") refiner_start: Union[float, None] = Field(default=None, description="The start value used for refiner denoising")
class ImageMetadata(BaseModel): class ImageMetadata(BaseModelExcludeNull):
"""An image's generation metadata""" """An image's generation metadata"""
metadata: Optional[dict] = Field( metadata: Optional[dict] = Field(

View File

@ -25,7 +25,6 @@ class BoardImageRecordStorageBase(ABC):
@abstractmethod @abstractmethod
def remove_image_from_board( def remove_image_from_board(
self, self,
board_id: str,
image_name: str, image_name: str,
) -> None: ) -> None:
"""Removes an image from a board.""" """Removes an image from a board."""
@ -154,7 +153,6 @@ class SqliteBoardImageRecordStorage(BoardImageRecordStorageBase):
def remove_image_from_board( def remove_image_from_board(
self, self,
board_id: str,
image_name: str, image_name: str,
) -> None: ) -> None:
try: try:
@ -162,9 +160,9 @@ class SqliteBoardImageRecordStorage(BoardImageRecordStorageBase):
self._cursor.execute( self._cursor.execute(
"""--sql """--sql
DELETE FROM board_images DELETE FROM board_images
WHERE board_id = ? AND image_name = ?; WHERE image_name = ?;
""", """,
(board_id, image_name), (image_name,),
) )
self._conn.commit() self._conn.commit()
except sqlite3.Error as e: except sqlite3.Error as e:

View File

@ -31,7 +31,6 @@ class BoardImagesServiceABC(ABC):
@abstractmethod @abstractmethod
def remove_image_from_board( def remove_image_from_board(
self, self,
board_id: str,
image_name: str, image_name: str,
) -> None: ) -> None:
"""Removes an image from a board.""" """Removes an image from a board."""
@ -93,10 +92,9 @@ class BoardImagesService(BoardImagesServiceABC):
def remove_image_from_board( def remove_image_from_board(
self, self,
board_id: str,
image_name: str, image_name: str,
) -> None: ) -> None:
self._services.board_image_records.remove_image_from_board(board_id, image_name) self._services.board_image_records.remove_image_from_board(image_name)
def get_all_board_image_names_for_board( def get_all_board_image_names_for_board(
self, self,

View File

@ -0,0 +1,8 @@
from pydantic import Field
from invokeai.app.util.model_exclude_null import BaseModelExcludeNull
class BoardImage(BaseModelExcludeNull):
board_id: str = Field(description="The id of the board")
image_name: str = Field(description="The name of the image")

View File

@ -1,10 +1,11 @@
from typing import Optional, Union from typing import Optional, Union
from datetime import datetime from datetime import datetime
from pydantic import BaseModel, Extra, Field, StrictBool, StrictStr from pydantic import Field
from invokeai.app.util.misc import get_iso_timestamp from invokeai.app.util.misc import get_iso_timestamp
from invokeai.app.util.model_exclude_null import BaseModelExcludeNull
class BoardRecord(BaseModel): class BoardRecord(BaseModelExcludeNull):
"""Deserialized board record.""" """Deserialized board record."""
board_id: str = Field(description="The unique ID of the board.") board_id: str = Field(description="The unique ID of the board.")

View File

@ -1,13 +1,14 @@
import datetime import datetime
from typing import Optional, Union from typing import Optional, Union
from pydantic import BaseModel, Extra, Field, StrictBool, StrictStr from pydantic import Extra, Field, StrictBool, StrictStr
from invokeai.app.models.image import ImageCategory, ResourceOrigin from invokeai.app.models.image import ImageCategory, ResourceOrigin
from invokeai.app.util.misc import get_iso_timestamp from invokeai.app.util.misc import get_iso_timestamp
from invokeai.app.util.model_exclude_null import BaseModelExcludeNull
class ImageRecord(BaseModel): class ImageRecord(BaseModelExcludeNull):
"""Deserialized image record without metadata.""" """Deserialized image record without metadata."""
image_name: str = Field(description="The unique name of the image.") image_name: str = Field(description="The unique name of the image.")
@ -40,7 +41,7 @@ class ImageRecord(BaseModel):
"""The node ID that generated this image, if it is a generated image.""" """The node ID that generated this image, if it is a generated image."""
class ImageRecordChanges(BaseModel, extra=Extra.forbid): class ImageRecordChanges(BaseModelExcludeNull, extra=Extra.forbid):
"""A set of changes to apply to an image record. """A set of changes to apply to an image record.
Only limited changes are valid: Only limited changes are valid:
@ -60,7 +61,7 @@ class ImageRecordChanges(BaseModel, extra=Extra.forbid):
"""The image's new `is_intermediate` flag.""" """The image's new `is_intermediate` flag."""
class ImageUrlsDTO(BaseModel): class ImageUrlsDTO(BaseModelExcludeNull):
"""The URLs for an image and its thumbnail.""" """The URLs for an image and its thumbnail."""
image_name: str = Field(description="The unique name of the image.") image_name: str = Field(description="The unique name of the image.")
@ -76,11 +77,15 @@ class ImageDTO(ImageRecord, ImageUrlsDTO):
board_id: Optional[str] = Field(description="The id of the board the image belongs to, if one exists.") board_id: Optional[str] = Field(description="The id of the board the image belongs to, if one exists.")
"""The id of the board the image belongs to, if one exists.""" """The id of the board the image belongs to, if one exists."""
pass pass
def image_record_to_dto( def image_record_to_dto(
image_record: ImageRecord, image_url: str, thumbnail_url: str, board_id: Optional[str] image_record: ImageRecord,
image_url: str,
thumbnail_url: str,
board_id: Optional[str],
) -> ImageDTO: ) -> ImageDTO:
"""Converts an image record to an image DTO.""" """Converts an image record to an image DTO."""
return ImageDTO( return ImageDTO(

View File

@ -20,6 +20,6 @@ class LocalUrlService(UrlServiceBase):
# These paths are determined by the routes in invokeai/app/api/routers/images.py # These paths are determined by the routes in invokeai/app/api/routers/images.py
if thumbnail: if thumbnail:
return f"{self._base_url}/images/{image_basename}/thumbnail" return f"{self._base_url}/images/i/{image_basename}/thumbnail"
return f"{self._base_url}/images/{image_basename}/full" return f"{self._base_url}/images/i/{image_basename}/full"

View File

@ -0,0 +1,23 @@
from typing import Any
from pydantic import BaseModel
"""
We want to exclude null values from objects that make their way to the client.
Unfortunately there is no built-in way to do this in pydantic, so we need to override the default
dict method to do this.
From https://github.com/tiangolo/fastapi/discussions/8882#discussioncomment-5154541
"""
class BaseModelExcludeNull(BaseModel):
def dict(self, *args, **kwargs) -> dict[str, Any]:
"""
Override the default dict method to exclude None values in the response
"""
kwargs.pop("exclude_none", None)
return super().dict(*args, exclude_none=True, **kwargs)
pass

View File

@ -23,7 +23,7 @@
"dev": "concurrently \"vite dev\" \"yarn run theme:watch\"", "dev": "concurrently \"vite dev\" \"yarn run theme:watch\"",
"dev:host": "concurrently \"vite dev --host\" \"yarn run theme:watch\"", "dev:host": "concurrently \"vite dev --host\" \"yarn run theme:watch\"",
"build": "yarn run lint && vite build", "build": "yarn run lint && vite build",
"typegen": "npx ts-node scripts/typegen.ts", "typegen": "node scripts/typegen.js",
"preview": "vite preview", "preview": "vite preview",
"lint:madge": "madge --circular src/main.tsx", "lint:madge": "madge --circular src/main.tsx",
"lint:eslint": "eslint --max-warnings=0 .", "lint:eslint": "eslint --max-warnings=0 .",

View File

@ -4,8 +4,9 @@ import { appStarted } from 'app/store/middleware/listenerMiddleware/listeners/ap
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { PartialAppConfig } from 'app/types/invokeai'; import { PartialAppConfig } from 'app/types/invokeai';
import ImageUploader from 'common/components/ImageUploader'; import ImageUploader from 'common/components/ImageUploader';
import ChangeBoardModal from 'features/changeBoardModal/components/ChangeBoardModal';
import DeleteImageModal from 'features/deleteImageModal/components/DeleteImageModal';
import GalleryDrawer from 'features/gallery/components/GalleryPanel'; import GalleryDrawer from 'features/gallery/components/GalleryPanel';
import DeleteImageModal from 'features/imageDeletion/components/DeleteImageModal';
import SiteHeader from 'features/system/components/SiteHeader'; import SiteHeader from 'features/system/components/SiteHeader';
import { configChanged } from 'features/system/store/configSlice'; import { configChanged } from 'features/system/store/configSlice';
import { languageSelector } from 'features/system/store/systemSelectors'; import { languageSelector } from 'features/system/store/systemSelectors';
@ -16,7 +17,6 @@ import ParametersDrawer from 'features/ui/components/ParametersDrawer';
import i18n from 'i18n'; import i18n from 'i18n';
import { size } from 'lodash-es'; import { size } from 'lodash-es';
import { ReactNode, memo, useEffect } from 'react'; import { ReactNode, memo, useEffect } from 'react';
import UpdateImageBoardModal from '../../features/gallery/components/Boards/UpdateImageBoardModal';
import GlobalHotkeys from './GlobalHotkeys'; import GlobalHotkeys from './GlobalHotkeys';
import Toaster from './Toaster'; import Toaster from './Toaster';
@ -84,7 +84,7 @@ const App = ({ config = DEFAULT_CONFIG, headerComponent }: Props) => {
</Portal> </Portal>
</Grid> </Grid>
<DeleteImageModal /> <DeleteImageModal />
<UpdateImageBoardModal /> <ChangeBoardModal />
<Toaster /> <Toaster />
<GlobalHotkeys /> <GlobalHotkeys />
</> </>

View File

@ -58,7 +58,7 @@ const DragPreview = (props: OverlayDragImageProps) => {
); );
} }
if (props.dragData.payloadType === 'IMAGE_NAMES') { if (props.dragData.payloadType === 'IMAGE_DTOS') {
return ( return (
<Flex <Flex
sx={{ sx={{
@ -71,7 +71,7 @@ const DragPreview = (props: OverlayDragImageProps) => {
...STYLES, ...STYLES,
}} }}
> >
<Heading>{props.dragData.payload.image_names.length}</Heading> <Heading>{props.dragData.payload.imageDTOs.length}</Heading>
<Heading size="sm">Images</Heading> <Heading size="sm">Images</Heading>
</Flex> </Flex>
); );

View File

@ -18,27 +18,32 @@ import {
DragStartEvent, DragStartEvent,
TypesafeDraggableData, TypesafeDraggableData,
} from './typesafeDnd'; } from './typesafeDnd';
import { logger } from 'app/logging/logger';
type ImageDndContextProps = PropsWithChildren; type ImageDndContextProps = PropsWithChildren;
const ImageDndContext = (props: ImageDndContextProps) => { const ImageDndContext = (props: ImageDndContextProps) => {
const [activeDragData, setActiveDragData] = const [activeDragData, setActiveDragData] =
useState<TypesafeDraggableData | null>(null); useState<TypesafeDraggableData | null>(null);
const log = logger('images');
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const handleDragStart = useCallback((event: DragStartEvent) => { const handleDragStart = useCallback(
console.log('dragStart', event.active.data.current); (event: DragStartEvent) => {
log.trace({ dragData: event.active.data.current }, 'Drag started');
const activeData = event.active.data.current; const activeData = event.active.data.current;
if (!activeData) { if (!activeData) {
return; return;
} }
setActiveDragData(activeData); setActiveDragData(activeData);
}, []); },
[log]
);
const handleDragEnd = useCallback( const handleDragEnd = useCallback(
(event: DragEndEvent) => { (event: DragEndEvent) => {
console.log('dragEnd', event.active.data.current); log.trace({ dragData: event.active.data.current }, 'Drag ended');
const overData = event.over?.data.current; const overData = event.over?.data.current;
if (!activeDragData || !overData) { if (!activeDragData || !overData) {
return; return;
@ -46,7 +51,7 @@ const ImageDndContext = (props: ImageDndContextProps) => {
dispatch(dndDropped({ overData, activeData: activeDragData })); dispatch(dndDropped({ overData, activeData: activeDragData }));
setActiveDragData(null); setActiveDragData(null);
}, },
[activeDragData, dispatch] [activeDragData, dispatch, log]
); );
const mouseSensor = useSensor(MouseSensor, { const mouseSensor = useSensor(MouseSensor, {

View File

@ -11,7 +11,6 @@ import {
useDraggable as useOriginalDraggable, useDraggable as useOriginalDraggable,
useDroppable as useOriginalDroppable, useDroppable as useOriginalDroppable,
} from '@dnd-kit/core'; } from '@dnd-kit/core';
import { BoardId } from 'features/gallery/store/types';
import { ImageDTO } from 'services/api/types'; import { ImageDTO } from 'services/api/types';
type BaseDropData = { type BaseDropData = {
@ -54,9 +53,13 @@ export type AddToBatchDropData = BaseDropData & {
actionType: 'ADD_TO_BATCH'; actionType: 'ADD_TO_BATCH';
}; };
export type MoveBoardDropData = BaseDropData & { export type AddToBoardDropData = BaseDropData & {
actionType: 'MOVE_BOARD'; actionType: 'ADD_TO_BOARD';
context: { boardId: BoardId }; context: { boardId: string };
};
export type RemoveFromBoardDropData = BaseDropData & {
actionType: 'REMOVE_FROM_BOARD';
}; };
export type TypesafeDroppableData = export type TypesafeDroppableData =
@ -67,7 +70,8 @@ export type TypesafeDroppableData =
| NodesImageDropData | NodesImageDropData
| AddToBatchDropData | AddToBatchDropData
| NodesMultiImageDropData | NodesMultiImageDropData
| MoveBoardDropData; | AddToBoardDropData
| RemoveFromBoardDropData;
type BaseDragData = { type BaseDragData = {
id: string; id: string;
@ -78,14 +82,12 @@ export type ImageDraggableData = BaseDragData & {
payload: { imageDTO: ImageDTO }; payload: { imageDTO: ImageDTO };
}; };
export type ImageNamesDraggableData = BaseDragData & { export type ImageDTOsDraggableData = BaseDragData & {
payloadType: 'IMAGE_NAMES'; payloadType: 'IMAGE_DTOS';
payload: { image_names: string[] }; payload: { imageDTOs: ImageDTO[] };
}; };
export type TypesafeDraggableData = export type TypesafeDraggableData = ImageDraggableData | ImageDTOsDraggableData;
| ImageDraggableData
| ImageNamesDraggableData;
interface UseDroppableTypesafeArguments interface UseDroppableTypesafeArguments
extends Omit<UseDroppableArguments, 'data'> { extends Omit<UseDroppableArguments, 'data'> {
@ -156,14 +158,39 @@ export const isValidDrop = (
case 'SET_NODES_IMAGE': case 'SET_NODES_IMAGE':
return payloadType === 'IMAGE_DTO'; return payloadType === 'IMAGE_DTO';
case 'SET_MULTI_NODES_IMAGE': case 'SET_MULTI_NODES_IMAGE':
return payloadType === 'IMAGE_DTO' || 'IMAGE_NAMES'; return payloadType === 'IMAGE_DTO' || 'IMAGE_DTOS';
case 'ADD_TO_BATCH': case 'ADD_TO_BATCH':
return payloadType === 'IMAGE_DTO' || 'IMAGE_NAMES'; return payloadType === 'IMAGE_DTO' || 'IMAGE_DTOS';
case 'MOVE_BOARD': { case 'ADD_TO_BOARD': {
// If the board is the same, don't allow the drop // If the board is the same, don't allow the drop
// Check the payload types // Check the payload types
const isPayloadValid = payloadType === 'IMAGE_DTO' || 'IMAGE_NAMES'; const isPayloadValid = payloadType === 'IMAGE_DTO' || 'IMAGE_DTOS';
if (!isPayloadValid) {
return false;
}
// Check if the image's board is the board we are dragging onto
if (payloadType === 'IMAGE_DTO') {
const { imageDTO } = active.data.current.payload;
const currentBoard = imageDTO.board_id ?? 'none';
const destinationBoard = overData.context.boardId;
return currentBoard !== destinationBoard;
}
if (payloadType === 'IMAGE_DTOS') {
// TODO (multi-select)
return true;
}
return false;
}
case 'REMOVE_FROM_BOARD': {
// If the board is the same, don't allow the drop
// Check the payload types
const isPayloadValid = payloadType === 'IMAGE_DTO' || 'IMAGE_DTOS';
if (!isPayloadValid) { if (!isPayloadValid) {
return false; return false;
} }
@ -172,20 +199,16 @@ export const isValidDrop = (
if (payloadType === 'IMAGE_DTO') { if (payloadType === 'IMAGE_DTO') {
const { imageDTO } = active.data.current.payload; const { imageDTO } = active.data.current.payload;
const currentBoard = imageDTO.board_id; const currentBoard = imageDTO.board_id;
const destinationBoard = overData.context.boardId;
const isSameBoard = currentBoard === destinationBoard; return currentBoard !== 'none';
const isDestinationValid = !currentBoard ? destinationBoard : true;
return !isSameBoard && isDestinationValid;
} }
if (payloadType === 'IMAGE_NAMES') { if (payloadType === 'IMAGE_DTOS') {
// TODO (multi-select) // TODO (multi-select)
return false; return true;
} }
return true; return false;
} }
default: default:
return false; return false;

View File

@ -1,4 +1,6 @@
import { Middleware } from '@reduxjs/toolkit';
import { store } from 'app/store/store'; import { store } from 'app/store/store';
import { PartialAppConfig } from 'app/types/invokeai';
import React, { import React, {
lazy, lazy,
memo, memo,
@ -7,16 +9,11 @@ import React, {
useEffect, useEffect,
} from 'react'; } from 'react';
import { Provider } from 'react-redux'; import { Provider } from 'react-redux';
import { PartialAppConfig } from 'app/types/invokeai';
import { addMiddleware, resetMiddlewares } from 'redux-dynamic-middlewares'; import { addMiddleware, resetMiddlewares } from 'redux-dynamic-middlewares';
import Loading from '../../common/components/Loading/Loading';
import { Middleware } from '@reduxjs/toolkit';
import { $authToken, $baseUrl, $projectId } from 'services/api/client'; import { $authToken, $baseUrl, $projectId } from 'services/api/client';
import { socketMiddleware } from 'services/events/middleware'; import { socketMiddleware } from 'services/events/middleware';
import Loading from '../../common/components/Loading/Loading';
import '../../i18n'; import '../../i18n';
import { AddImageToBoardContextProvider } from '../contexts/AddImageToBoardContext';
import ImageDndContext from './ImageDnd/ImageDndContext'; import ImageDndContext from './ImageDnd/ImageDndContext';
const App = lazy(() => import('./App')); const App = lazy(() => import('./App'));
@ -84,9 +81,7 @@ const InvokeAIUI = ({
<React.Suspense fallback={<Loading />}> <React.Suspense fallback={<Loading />}>
<ThemeLocaleProvider> <ThemeLocaleProvider>
<ImageDndContext> <ImageDndContext>
<AddImageToBoardContextProvider>
<App config={config} headerComponent={headerComponent} /> <App config={config} headerComponent={headerComponent} />
</AddImageToBoardContextProvider>
</ImageDndContext> </ImageDndContext>
</ThemeLocaleProvider> </ThemeLocaleProvider>
</React.Suspense> </React.Suspense>

View File

@ -1,91 +0,0 @@
import { useDisclosure } from '@chakra-ui/react';
import { PropsWithChildren, createContext, useCallback, useState } from 'react';
import { ImageDTO } from 'services/api/types';
import { imagesApi } from 'services/api/endpoints/images';
import { useAppDispatch } from '../store/storeHooks';
export type ImageUsage = {
isInitialImage: boolean;
isCanvasImage: boolean;
isNodesImage: boolean;
isControlNetImage: boolean;
};
type AddImageToBoardContextValue = {
/**
* Whether the move image dialog is open.
*/
isOpen: boolean;
/**
* Closes the move image dialog.
*/
onClose: () => void;
/**
* The image pending movement
*/
image?: ImageDTO;
onClickAddToBoard: (image: ImageDTO) => void;
handleAddToBoard: (boardId: string) => void;
};
export const AddImageToBoardContext =
createContext<AddImageToBoardContextValue>({
isOpen: false,
onClose: () => undefined,
onClickAddToBoard: () => undefined,
handleAddToBoard: () => undefined,
});
type Props = PropsWithChildren;
export const AddImageToBoardContextProvider = (props: Props) => {
const [imageToMove, setImageToMove] = useState<ImageDTO>();
const { isOpen, onOpen, onClose } = useDisclosure();
const dispatch = useAppDispatch();
// Clean up after deleting or dismissing the modal
const closeAndClearImageToDelete = useCallback(() => {
setImageToMove(undefined);
onClose();
}, [onClose]);
const onClickAddToBoard = useCallback(
(image?: ImageDTO) => {
if (!image) {
return;
}
setImageToMove(image);
onOpen();
},
[setImageToMove, onOpen]
);
const handleAddToBoard = useCallback(
(boardId: string) => {
if (imageToMove) {
dispatch(
imagesApi.endpoints.addImageToBoard.initiate({
imageDTO: imageToMove,
board_id: boardId,
})
);
closeAndClearImageToDelete();
}
},
[dispatch, closeAndClearImageToDelete, imageToMove]
);
return (
<AddImageToBoardContext.Provider
value={{
isOpen,
image: imageToMove,
onClose: closeAndClearImageToDelete,
onClickAddToBoard,
handleAddToBoard,
}}
>
{props.children}
</AddImageToBoardContext.Provider>
);
};

View File

@ -1,8 +0,0 @@
import { createContext } from 'react';
type VoidFunc = () => void;
type ImageUploaderTriggerContextType = VoidFunc | null;
export const ImageUploaderTriggerContext =
createContext<ImageUploaderTriggerContextType>(null);

View File

@ -23,6 +23,6 @@ const serializationDenylist: {
}; };
export const serialize: SerializeFunction = (data, key) => { export const serialize: SerializeFunction = (data, key) => {
const result = omit(data, serializationDenylist[key]); const result = omit(data, serializationDenylist[key] ?? []);
return JSON.stringify(result); return JSON.stringify(result);
}; };

View File

@ -27,7 +27,8 @@ import {
addImageDeletedFulfilledListener, addImageDeletedFulfilledListener,
addImageDeletedPendingListener, addImageDeletedPendingListener,
addImageDeletedRejectedListener, addImageDeletedRejectedListener,
addRequestedImageDeletionListener, addRequestedSingleImageDeletionListener,
addRequestedMultipleImageDeletionListener,
} from './listeners/imageDeleted'; } from './listeners/imageDeleted';
import { addImageDroppedListener } from './listeners/imageDropped'; import { addImageDroppedListener } from './listeners/imageDropped';
import { import {
@ -111,7 +112,8 @@ addImageUploadedRejectedListener();
addInitialImageSelectedListener(); addInitialImageSelectedListener();
// Image deleted // Image deleted
addRequestedImageDeletionListener(); addRequestedSingleImageDeletionListener();
addRequestedMultipleImageDeletionListener();
addImageDeletedPendingListener(); addImageDeletedPendingListener();
addImageDeletedFulfilledListener(); addImageDeletedFulfilledListener();
addImageDeletedRejectedListener(); addImageDeletedRejectedListener();

View File

@ -1,12 +1,10 @@
import { createAction } from '@reduxjs/toolkit'; import { createAction } from '@reduxjs/toolkit';
import { imageSelected } from 'features/gallery/store/gallerySlice'; import { imageSelected } from 'features/gallery/store/gallerySlice';
import { IMAGE_CATEGORIES } from 'features/gallery/store/types'; import { IMAGE_CATEGORIES } from 'features/gallery/store/types';
import { import { imagesApi } from 'services/api/endpoints/images';
ImageCache,
getListImagesUrl,
imagesApi,
} from 'services/api/endpoints/images';
import { startAppListening } from '..'; import { startAppListening } from '..';
import { getListImagesUrl, imagesAdapter } from 'services/api/util';
import { ImageCache } from 'services/api/types';
export const appStarted = createAction('app/appStarted'); export const appStarted = createAction('app/appStarted');
@ -34,7 +32,8 @@ export const addFirstListImagesListener = () => {
if (data.ids.length > 0) { if (data.ids.length > 0) {
// Select the first image // Select the first image
dispatch(imageSelected(data.ids[0] as string)); const firstImage = imagesAdapter.getSelectors().selectAll(data)[0];
dispatch(imageSelected(firstImage ?? null));
} }
}, },
}); });

View File

@ -18,7 +18,9 @@ export const addAppConfigReceivedListener = () => {
const infillMethod = getState().generation.infillMethod; const infillMethod = getState().generation.infillMethod;
if (!infill_methods.includes(infillMethod)) { if (!infill_methods.includes(infillMethod)) {
dispatch(setInfillMethod(infill_methods[0])); // if there is no infill method, set it to the first one
// if there is no first one... god help us
dispatch(setInfillMethod(infill_methods[0] as string));
} }
if (!nsfw_methods.includes('nsfw_checker')) { if (!nsfw_methods.includes('nsfw_checker')) {

View File

@ -1,14 +1,14 @@
import { resetCanvas } from 'features/canvas/store/canvasSlice'; import { resetCanvas } from 'features/canvas/store/canvasSlice';
import { controlNetReset } from 'features/controlNet/store/controlNetSlice'; import { controlNetReset } from 'features/controlNet/store/controlNetSlice';
import { getImageUsage } from 'features/imageDeletion/store/imageDeletionSelectors'; import { getImageUsage } from 'features/deleteImageModal/store/selectors';
import { nodeEditorReset } from 'features/nodes/store/nodesSlice'; import { nodeEditorReset } from 'features/nodes/store/nodesSlice';
import { clearInitialImage } from 'features/parameters/store/generationSlice'; import { clearInitialImage } from 'features/parameters/store/generationSlice';
import { imagesApi } from 'services/api/endpoints/images';
import { startAppListening } from '..'; import { startAppListening } from '..';
import { boardsApi } from '../../../../../services/api/endpoints/boards';
export const addDeleteBoardAndImagesFulfilledListener = () => { export const addDeleteBoardAndImagesFulfilledListener = () => {
startAppListening({ startAppListening({
matcher: boardsApi.endpoints.deleteBoardAndImages.matchFulfilled, matcher: imagesApi.endpoints.deleteBoardAndImages.matchFulfilled,
effect: async (action, { dispatch, getState }) => { effect: async (action, { dispatch, getState }) => {
const { deleted_images } = action.payload; const { deleted_images } = action.payload;

View File

@ -10,6 +10,7 @@ import {
} from 'features/gallery/store/types'; } from 'features/gallery/store/types';
import { imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
import { startAppListening } from '..'; import { startAppListening } from '..';
import { imagesSelectors } from 'services/api/util';
export const addBoardIdSelectedListener = () => { export const addBoardIdSelectedListener = () => {
startAppListening({ startAppListening({
@ -52,8 +53,9 @@ export const addBoardIdSelectedListener = () => {
queryArgs queryArgs
)(getState()); )(getState());
if (boardImagesData?.ids.length) { if (boardImagesData) {
dispatch(imageSelected((boardImagesData.ids[0] as string) ?? null)); const firstImage = imagesSelectors.selectAll(boardImagesData)[0];
dispatch(imageSelected(firstImage ?? null));
} else { } else {
// board has no images - deselect // board has no images - deselect
dispatch(imageSelected(null)); dispatch(imageSelected(null));

View File

@ -26,6 +26,8 @@ export const addCanvasSavedToGalleryListener = () => {
return; return;
} }
const { autoAddBoardId } = state.gallery;
dispatch( dispatch(
imagesApi.endpoints.uploadImage.initiate({ imagesApi.endpoints.uploadImage.initiate({
file: new File([blob], 'savedCanvas.png', { file: new File([blob], 'savedCanvas.png', {
@ -33,7 +35,7 @@ export const addCanvasSavedToGalleryListener = () => {
}), }),
image_category: 'general', image_category: 'general',
is_intermediate: false, is_intermediate: false,
board_id: state.gallery.autoAddBoardId, board_id: autoAddBoardId === 'none' ? undefined : autoAddBoardId,
crop_visible: true, crop_visible: true,
postUploadAction: { postUploadAction: {
type: 'TOAST', type: 'TOAST',

View File

@ -31,15 +31,20 @@ const predicate: AnyListenerPredicate<RootState> = (
// do not process if the user just disabled auto-config // do not process if the user just disabled auto-config
if ( if (
prevState.controlNet.controlNets[action.payload.controlNetId] prevState.controlNet.controlNets[action.payload.controlNetId]
.shouldAutoConfig === true ?.shouldAutoConfig === true
) { ) {
return false; return false;
} }
} }
const { controlImage, processorType, shouldAutoConfig } = const cn = state.controlNet.controlNets[action.payload.controlNetId];
state.controlNet.controlNets[action.payload.controlNetId];
if (!cn) {
// something is wrong, the controlNet should exist
return false;
}
const { controlImage, processorType, shouldAutoConfig } = cn;
if (controlNetModelChanged.match(action) && !shouldAutoConfig) { if (controlNetModelChanged.match(action) && !shouldAutoConfig) {
// do not process if the action is a model change but the processor settings are dirty // do not process if the action is a model change but the processor settings are dirty
return false; return false;

View File

@ -17,7 +17,7 @@ export const addControlNetImageProcessedListener = () => {
const { controlNetId } = action.payload; const { controlNetId } = action.payload;
const controlNet = getState().controlNet.controlNets[controlNetId]; const controlNet = getState().controlNet.controlNets[controlNetId];
if (!controlNet.controlImage) { if (!controlNet?.controlImage) {
log.error('Unable to process ControlNet image'); log.error('Unable to process ControlNet image');
return; return;
} }

View File

@ -1,57 +1,72 @@
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { resetCanvas } from 'features/canvas/store/canvasSlice'; import { resetCanvas } from 'features/canvas/store/canvasSlice';
import { controlNetReset } from 'features/controlNet/store/controlNetSlice'; import { controlNetReset } from 'features/controlNet/store/controlNetSlice';
import { imageDeletionConfirmed } from 'features/deleteImageModal/store/actions';
import { isModalOpenChanged } from 'features/deleteImageModal/store/slice';
import { selectListImagesBaseQueryArgs } from 'features/gallery/store/gallerySelectors'; import { selectListImagesBaseQueryArgs } from 'features/gallery/store/gallerySelectors';
import { imageSelected } from 'features/gallery/store/gallerySlice'; import { imageSelected } from 'features/gallery/store/gallerySlice';
import { imageDeletionConfirmed } from 'features/imageDeletion/store/actions';
import { isModalOpenChanged } from 'features/imageDeletion/store/imageDeletionSlice';
import { nodeEditorReset } from 'features/nodes/store/nodesSlice'; import { nodeEditorReset } from 'features/nodes/store/nodesSlice';
import { clearInitialImage } from 'features/parameters/store/generationSlice'; import { clearInitialImage } from 'features/parameters/store/generationSlice';
import { clamp } from 'lodash-es'; import { clamp } from 'lodash-es';
import { api } from 'services/api'; import { api } from 'services/api';
import { imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
import { imagesAdapter } from 'services/api/util';
import { startAppListening } from '..'; import { startAppListening } from '..';
/** export const addRequestedSingleImageDeletionListener = () => {
* Called when the user requests an image deletion
*/
export const addRequestedImageDeletionListener = () => {
startAppListening({ startAppListening({
actionCreator: imageDeletionConfirmed, actionCreator: imageDeletionConfirmed,
effect: async (action, { dispatch, getState, condition }) => { effect: async (action, { dispatch, getState, condition }) => {
const { imageDTO, imageUsage } = action.payload; const { imageDTOs, imagesUsage } = action.payload;
if (imageDTOs.length !== 1 || imagesUsage.length !== 1) {
// handle multiples in separate listener
return;
}
const imageDTO = imageDTOs[0];
const imageUsage = imagesUsage[0];
if (!imageDTO || !imageUsage) {
// satisfy noUncheckedIndexedAccess
return;
}
dispatch(isModalOpenChanged(false)); dispatch(isModalOpenChanged(false));
const { image_name } = imageDTO;
const state = getState(); const state = getState();
const lastSelectedImage = const lastSelectedImage =
state.gallery.selection[state.gallery.selection.length - 1]; state.gallery.selection[state.gallery.selection.length - 1]?.image_name;
if (imageDTO && imageDTO?.image_name === lastSelectedImage) {
const { image_name } = imageDTO;
if (lastSelectedImage === image_name) {
const baseQueryArgs = selectListImagesBaseQueryArgs(state); const baseQueryArgs = selectListImagesBaseQueryArgs(state);
const { data } = const { data } =
imagesApi.endpoints.listImages.select(baseQueryArgs)(state); imagesApi.endpoints.listImages.select(baseQueryArgs)(state);
const ids = data?.ids ?? []; const cachedImageDTOs = data
? imagesAdapter.getSelectors().selectAll(data)
: [];
const deletedImageIndex = ids.findIndex( const deletedImageIndex = cachedImageDTOs.findIndex(
(result) => result.toString() === image_name (i) => i.image_name === image_name
); );
const filteredIds = ids.filter((id) => id.toString() !== image_name); const filteredImageDTOs = cachedImageDTOs.filter(
(i) => i.image_name !== image_name
);
const newSelectedImageIndex = clamp( const newSelectedImageIndex = clamp(
deletedImageIndex, deletedImageIndex,
0, 0,
filteredIds.length - 1 filteredImageDTOs.length - 1
); );
const newSelectedImageId = filteredIds[newSelectedImageIndex]; const newSelectedImageDTO = filteredImageDTOs[newSelectedImageIndex];
if (newSelectedImageId) { if (newSelectedImageDTO) {
dispatch(imageSelected(newSelectedImageId as string)); dispatch(imageSelected(newSelectedImageDTO));
} else { } else {
dispatch(imageSelected(null)); dispatch(imageSelected(null));
} }
@ -97,6 +112,66 @@ export const addRequestedImageDeletionListener = () => {
}); });
}; };
/**
* Called when the user requests an image deletion
*/
export const addRequestedMultipleImageDeletionListener = () => {
startAppListening({
actionCreator: imageDeletionConfirmed,
effect: async (action, { dispatch, getState }) => {
const { imageDTOs, imagesUsage } = action.payload;
if (imageDTOs.length < 1 || imagesUsage.length < 1) {
// handle singles in separate listener
return;
}
try {
// Delete from server
await dispatch(
imagesApi.endpoints.deleteImages.initiate({ imageDTOs })
).unwrap();
const state = getState();
const baseQueryArgs = selectListImagesBaseQueryArgs(state);
const { data } =
imagesApi.endpoints.listImages.select(baseQueryArgs)(state);
const newSelectedImageDTO = data
? imagesAdapter.getSelectors().selectAll(data)[0]
: undefined;
if (newSelectedImageDTO) {
dispatch(imageSelected(newSelectedImageDTO));
} else {
dispatch(imageSelected(null));
}
dispatch(isModalOpenChanged(false));
// We need to reset the features where the image is in use - none of these work if their image(s) don't exist
if (imagesUsage.some((i) => i.isCanvasImage)) {
dispatch(resetCanvas());
}
if (imagesUsage.some((i) => i.isControlNetImage)) {
dispatch(controlNetReset());
}
if (imagesUsage.some((i) => i.isInitialImage)) {
dispatch(clearInitialImage());
}
if (imagesUsage.some((i) => i.isNodesImage)) {
dispatch(nodeEditorReset());
}
} catch {
// no-op
}
},
});
};
/** /**
* Called when the actual delete request is sent to the server * Called when the actual delete request is sent to the server
*/ */

View File

@ -6,10 +6,7 @@ import {
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { setInitialCanvasImage } from 'features/canvas/store/canvasSlice'; import { setInitialCanvasImage } from 'features/canvas/store/canvasSlice';
import { controlNetImageChanged } from 'features/controlNet/store/controlNetSlice'; import { controlNetImageChanged } from 'features/controlNet/store/controlNetSlice';
import { import { imageSelected } from 'features/gallery/store/gallerySlice';
imageSelected,
imagesAddedToBatch,
} from 'features/gallery/store/gallerySlice';
import { fieldValueChanged } from 'features/nodes/store/nodesSlice'; import { fieldValueChanged } from 'features/nodes/store/nodesSlice';
import { initialImageChanged } from 'features/parameters/store/generationSlice'; import { initialImageChanged } from 'features/parameters/store/generationSlice';
import { imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
@ -27,19 +24,32 @@ export const addImageDroppedListener = () => {
const log = logger('images'); const log = logger('images');
const { activeData, overData } = action.payload; const { activeData, overData } = action.payload;
log.debug({ activeData, overData }, 'Image or selection dropped'); if (activeData.payloadType === 'IMAGE_DTO') {
log.debug({ activeData, overData }, 'Image dropped');
} else if (activeData.payloadType === 'IMAGE_DTOS') {
log.debug(
{ activeData, overData },
`Images (${activeData.payload.imageDTOs.length}) dropped`
);
} else {
log.debug({ activeData, overData }, `Unknown payload dropped`);
}
// set current image /**
* Image dropped on current image
*/
if ( if (
overData.actionType === 'SET_CURRENT_IMAGE' && overData.actionType === 'SET_CURRENT_IMAGE' &&
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
activeData.payload.imageDTO activeData.payload.imageDTO
) { ) {
dispatch(imageSelected(activeData.payload.imageDTO.image_name)); dispatch(imageSelected(activeData.payload.imageDTO));
return; return;
} }
// set initial image /**
* Image dropped on initial image
*/
if ( if (
overData.actionType === 'SET_INITIAL_IMAGE' && overData.actionType === 'SET_INITIAL_IMAGE' &&
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
@ -49,27 +59,9 @@ export const addImageDroppedListener = () => {
return; return;
} }
// add image to batch /**
if ( * Image dropped on ControlNet
overData.actionType === 'ADD_TO_BATCH' && */
activeData.payloadType === 'IMAGE_DTO' &&
activeData.payload.imageDTO
) {
dispatch(imagesAddedToBatch([activeData.payload.imageDTO.image_name]));
return;
}
// add multiple images to batch
if (
overData.actionType === 'ADD_TO_BATCH' &&
activeData.payloadType === 'IMAGE_NAMES'
) {
dispatch(imagesAddedToBatch(activeData.payload.image_names));
return;
}
// set control image
if ( if (
overData.actionType === 'SET_CONTROLNET_IMAGE' && overData.actionType === 'SET_CONTROLNET_IMAGE' &&
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
@ -85,7 +77,9 @@ export const addImageDroppedListener = () => {
return; return;
} }
// set canvas image /**
* Image dropped on Canvas
*/
if ( if (
overData.actionType === 'SET_CANVAS_INITIAL_IMAGE' && overData.actionType === 'SET_CANVAS_INITIAL_IMAGE' &&
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
@ -95,7 +89,9 @@ export const addImageDroppedListener = () => {
return; return;
} }
// set nodes image /**
* Image dropped on node image field
*/
if ( if (
overData.actionType === 'SET_NODES_IMAGE' && overData.actionType === 'SET_NODES_IMAGE' &&
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
@ -112,61 +108,36 @@ export const addImageDroppedListener = () => {
return; return;
} }
// set multiple nodes images (single image handler) /**
if ( * TODO
overData.actionType === 'SET_MULTI_NODES_IMAGE' && * Image selection dropped on node image collection field
activeData.payloadType === 'IMAGE_DTO' && */
activeData.payload.imageDTO
) {
const { fieldName, nodeId } = overData.context;
dispatch(
fieldValueChanged({
nodeId,
fieldName,
value: [activeData.payload.imageDTO],
})
);
return;
}
// // set multiple nodes images (multiple images handler)
// if ( // if (
// overData.actionType === 'SET_MULTI_NODES_IMAGE' && // overData.actionType === 'SET_MULTI_NODES_IMAGE' &&
// activeData.payloadType === 'IMAGE_NAMES' // activeData.payloadType === 'IMAGE_DTO' &&
// activeData.payload.imageDTO
// ) { // ) {
// const { fieldName, nodeId } = overData.context; // const { fieldName, nodeId } = overData.context;
// dispatch( // dispatch(
// imageCollectionFieldValueChanged({ // fieldValueChanged({
// nodeId, // nodeId,
// fieldName, // fieldName,
// value: activeData.payload.image_names.map((image_name) => ({ // value: [activeData.payload.imageDTO],
// image_name,
// })),
// }) // })
// ); // );
// return; // return;
// } // }
// add image to board /**
* Image dropped on user board
*/
if ( if (
overData.actionType === 'MOVE_BOARD' && overData.actionType === 'ADD_TO_BOARD' &&
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
activeData.payload.imageDTO activeData.payload.imageDTO
) { ) {
const { imageDTO } = activeData.payload; const { imageDTO } = activeData.payload;
const { boardId } = overData.context; const { boardId } = overData.context;
// image was droppe on the "NoBoardBoard"
if (!boardId) {
dispatch(
imagesApi.endpoints.removeImageFromBoard.initiate({
imageDTO,
})
);
return;
}
// image was dropped on a user board
dispatch( dispatch(
imagesApi.endpoints.addImageToBoard.initiate({ imagesApi.endpoints.addImageToBoard.initiate({
imageDTO, imageDTO,
@ -176,67 +147,58 @@ export const addImageDroppedListener = () => {
return; return;
} }
// // add gallery selection to board /**
// if ( * Image dropped on 'none' board
// overData.actionType === 'MOVE_BOARD' && */
// activeData.payloadType === 'IMAGE_NAMES' && if (
// overData.context.boardId overData.actionType === 'REMOVE_FROM_BOARD' &&
// ) { activeData.payloadType === 'IMAGE_DTO' &&
// console.log('adding gallery selection to board'); activeData.payload.imageDTO
// const board_id = overData.context.boardId; ) {
// dispatch( const { imageDTO } = activeData.payload;
// boardImagesApi.endpoints.addManyBoardImages.initiate({ dispatch(
// board_id, imagesApi.endpoints.removeImageFromBoard.initiate({
// image_names: activeData.payload.image_names, imageDTO,
// }) })
// ); );
// return; return;
// } }
// // remove gallery selection from board /**
// if ( * Multiple images dropped on user board
// overData.actionType === 'MOVE_BOARD' && */
// activeData.payloadType === 'IMAGE_NAMES' && if (
// overData.context.boardId === null overData.actionType === 'ADD_TO_BOARD' &&
// ) { activeData.payloadType === 'IMAGE_DTOS' &&
// console.log('removing gallery selection to board'); activeData.payload.imageDTOs
// dispatch( ) {
// boardImagesApi.endpoints.deleteManyBoardImages.initiate({ const { imageDTOs } = activeData.payload;
// image_names: activeData.payload.image_names, const { boardId } = overData.context;
// }) dispatch(
// ); imagesApi.endpoints.addImagesToBoard.initiate({
// return; imageDTOs,
// } board_id: boardId,
})
);
return;
}
// // add batch selection to board /**
// if ( * Multiple images dropped on 'none' board
// overData.actionType === 'MOVE_BOARD' && */
// activeData.payloadType === 'IMAGE_NAMES' && if (
// overData.context.boardId overData.actionType === 'REMOVE_FROM_BOARD' &&
// ) { activeData.payloadType === 'IMAGE_DTOS' &&
// const board_id = overData.context.boardId; activeData.payload.imageDTOs
// dispatch( ) {
// boardImagesApi.endpoints.addManyBoardImages.initiate({ const { imageDTOs } = activeData.payload;
// board_id, dispatch(
// image_names: activeData.payload.image_names, imagesApi.endpoints.removeImagesFromBoard.initiate({
// }) imageDTOs,
// ); })
// return; );
// } return;
}
// // remove batch selection from board
// if (
// overData.actionType === 'MOVE_BOARD' &&
// activeData.payloadType === 'IMAGE_NAMES' &&
// overData.context.boardId === null
// ) {
// dispatch(
// boardImagesApi.endpoints.deleteManyBoardImages.initiate({
// image_names: activeData.payload.image_names,
// })
// );
// return;
// }
}, },
}); });
}; };

View File

@ -1,37 +1,32 @@
import { imageDeletionConfirmed } from 'features/imageDeletion/store/actions'; import { imageDeletionConfirmed } from 'features/deleteImageModal/store/actions';
import { selectImageUsage } from 'features/imageDeletion/store/imageDeletionSelectors'; import { selectImageUsage } from 'features/deleteImageModal/store/selectors';
import { import {
imageToDeleteSelected, imagesToDeleteSelected,
isModalOpenChanged, isModalOpenChanged,
} from 'features/imageDeletion/store/imageDeletionSlice'; } from 'features/deleteImageModal/store/slice';
import { startAppListening } from '..'; import { startAppListening } from '..';
export const addImageToDeleteSelectedListener = () => { export const addImageToDeleteSelectedListener = () => {
startAppListening({ startAppListening({
actionCreator: imageToDeleteSelected, actionCreator: imagesToDeleteSelected,
effect: async (action, { dispatch, getState }) => { effect: async (action, { dispatch, getState }) => {
const imageDTO = action.payload; const imageDTOs = action.payload;
const state = getState(); const state = getState();
const { shouldConfirmOnDelete } = state.system; const { shouldConfirmOnDelete } = state.system;
const imageUsage = selectImageUsage(getState()); const imagesUsage = selectImageUsage(getState());
if (!imageUsage) {
// should never happen
return;
}
const isImageInUse = const isImageInUse =
imageUsage.isCanvasImage || imagesUsage.some((i) => i.isCanvasImage) ||
imageUsage.isInitialImage || imagesUsage.some((i) => i.isInitialImage) ||
imageUsage.isControlNetImage || imagesUsage.some((i) => i.isControlNetImage) ||
imageUsage.isNodesImage; imagesUsage.some((i) => i.isNodesImage);
if (shouldConfirmOnDelete || isImageInUse) { if (shouldConfirmOnDelete || isImageInUse) {
dispatch(isModalOpenChanged(true)); dispatch(isModalOpenChanged(true));
return; return;
} }
dispatch(imageDeletionConfirmed({ imageDTO, imageUsage })); dispatch(imageDeletionConfirmed({ imageDTOs, imagesUsage }));
}, },
}); });
}; };

View File

@ -2,14 +2,13 @@ import { UseToastOptions } from '@chakra-ui/react';
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { setInitialCanvasImage } from 'features/canvas/store/canvasSlice'; import { setInitialCanvasImage } from 'features/canvas/store/canvasSlice';
import { controlNetImageChanged } from 'features/controlNet/store/controlNetSlice'; import { controlNetImageChanged } from 'features/controlNet/store/controlNetSlice';
import { imagesAddedToBatch } from 'features/gallery/store/gallerySlice';
import { fieldValueChanged } from 'features/nodes/store/nodesSlice'; import { fieldValueChanged } from 'features/nodes/store/nodesSlice';
import { initialImageChanged } from 'features/parameters/store/generationSlice'; import { initialImageChanged } from 'features/parameters/store/generationSlice';
import { addToast } from 'features/system/store/systemSlice'; import { addToast } from 'features/system/store/systemSlice';
import { omit } from 'lodash-es';
import { boardsApi } from 'services/api/endpoints/boards'; import { boardsApi } from 'services/api/endpoints/boards';
import { startAppListening } from '..'; import { startAppListening } from '..';
import { imagesApi } from '../../../../../services/api/endpoints/images'; import { imagesApi } from '../../../../../services/api/endpoints/images';
import { omit } from 'lodash-es';
const DEFAULT_UPLOADED_TOAST: UseToastOptions = { const DEFAULT_UPLOADED_TOAST: UseToastOptions = {
title: 'Image Uploaded', title: 'Image Uploaded',
@ -121,17 +120,6 @@ export const addImageUploadedFulfilledListener = () => {
); );
return; return;
} }
if (postUploadAction?.type === 'ADD_TO_BATCH') {
dispatch(imagesAddedToBatch([imageDTO.image_name]));
dispatch(
addToast({
...DEFAULT_UPLOADED_TOAST,
description: 'Added to batch',
})
);
return;
}
}, },
}); });
}; };

View File

@ -15,7 +15,7 @@ import {
setShouldUseSDXLRefiner, setShouldUseSDXLRefiner,
} from 'features/sdxl/store/sdxlSlice'; } from 'features/sdxl/store/sdxlSlice';
import { forEach, some } from 'lodash-es'; import { forEach, some } from 'lodash-es';
import { modelsApi } from 'services/api/endpoints/models'; import { modelsApi, vaeModelsAdapter } from 'services/api/endpoints/models';
import { startAppListening } from '..'; import { startAppListening } from '..';
export const addModelsLoadedListener = () => { export const addModelsLoadedListener = () => {
@ -144,8 +144,9 @@ export const addModelsLoadedListener = () => {
return; return;
} }
const firstModelId = action.payload.ids[0]; const firstModel = vaeModelsAdapter
const firstModel = action.payload.entities[firstModelId]; .getSelectors()
.selectAll(action.payload)[0];
if (!firstModel) { if (!firstModel) {
// No custom VAEs loaded at all; use the default // No custom VAEs loaded at all; use the default

View File

@ -8,9 +8,10 @@ import {
} from 'features/gallery/store/gallerySlice'; } from 'features/gallery/store/gallerySlice';
import { IMAGE_CATEGORIES } from 'features/gallery/store/types'; import { IMAGE_CATEGORIES } from 'features/gallery/store/types';
import { progressImageSet } from 'features/system/store/systemSlice'; import { progressImageSet } from 'features/system/store/systemSlice';
import { imagesAdapter, imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
import { isImageOutput } from 'services/api/guards'; import { isImageOutput } from 'services/api/guards';
import { sessionCanceled } from 'services/api/thunks/session'; import { sessionCanceled } from 'services/api/thunks/session';
import { imagesAdapter } from 'services/api/util';
import { import {
appSocketInvocationComplete, appSocketInvocationComplete,
socketInvocationComplete, socketInvocationComplete,
@ -67,7 +68,7 @@ export const addInvocationCompleteEventListener = () => {
*/ */
const { autoAddBoardId } = gallery; const { autoAddBoardId } = gallery;
if (autoAddBoardId) { if (autoAddBoardId && autoAddBoardId !== 'none') {
dispatch( dispatch(
imagesApi.endpoints.addImageToBoard.initiate({ imagesApi.endpoints.addImageToBoard.initiate({
board_id: autoAddBoardId, board_id: autoAddBoardId,
@ -83,10 +84,7 @@ export const addInvocationCompleteEventListener = () => {
categories: IMAGE_CATEGORIES, categories: IMAGE_CATEGORIES,
}, },
(draft) => { (draft) => {
const oldTotal = draft.total; imagesAdapter.addOne(draft, imageDTO);
const newState = imagesAdapter.addOne(draft, imageDTO);
const delta = newState.total - oldTotal;
draft.total = draft.total + delta;
} }
) )
); );
@ -94,8 +92,8 @@ export const addInvocationCompleteEventListener = () => {
dispatch( dispatch(
imagesApi.util.invalidateTags([ imagesApi.util.invalidateTags([
{ type: 'BoardImagesTotal', id: autoAddBoardId ?? 'none' }, { type: 'BoardImagesTotal', id: autoAddBoardId },
{ type: 'BoardAssetsTotal', id: autoAddBoardId ?? 'none' }, { type: 'BoardAssetsTotal', id: autoAddBoardId },
]) ])
); );
@ -110,7 +108,7 @@ export const addInvocationCompleteEventListener = () => {
} else if (!autoAddBoardId) { } else if (!autoAddBoardId) {
dispatch(galleryViewChanged('images')); dispatch(galleryViewChanged('images'));
} }
dispatch(imageSelected(imageDTO.image_name)); dispatch(imageSelected(imageDTO));
} }
} }

View File

@ -8,9 +8,9 @@ import {
import canvasReducer from 'features/canvas/store/canvasSlice'; import canvasReducer from 'features/canvas/store/canvasSlice';
import controlNetReducer from 'features/controlNet/store/controlNetSlice'; import controlNetReducer from 'features/controlNet/store/controlNetSlice';
import dynamicPromptsReducer from 'features/dynamicPrompts/store/dynamicPromptsSlice'; import dynamicPromptsReducer from 'features/dynamicPrompts/store/dynamicPromptsSlice';
import boardsReducer from 'features/gallery/store/boardSlice';
import galleryReducer from 'features/gallery/store/gallerySlice'; import galleryReducer from 'features/gallery/store/gallerySlice';
import imageDeletionReducer from 'features/imageDeletion/store/imageDeletionSlice'; import deleteImageModalReducer from 'features/deleteImageModal/store/slice';
import changeBoardModalReducer from 'features/changeBoardModal/store/slice';
import loraReducer from 'features/lora/store/loraSlice'; import loraReducer from 'features/lora/store/loraSlice';
import nodesReducer from 'features/nodes/store/nodesSlice'; import nodesReducer from 'features/nodes/store/nodesSlice';
import generationReducer from 'features/parameters/store/generationSlice'; import generationReducer from 'features/parameters/store/generationSlice';
@ -43,9 +43,9 @@ const allReducers = {
ui: uiReducer, ui: uiReducer,
hotkeys: hotkeysReducer, hotkeys: hotkeysReducer,
controlNet: controlNetReducer, controlNet: controlNetReducer,
boards: boardsReducer,
dynamicPrompts: dynamicPromptsReducer, dynamicPrompts: dynamicPromptsReducer,
imageDeletion: imageDeletionReducer, deleteImageModal: deleteImageModalReducer,
changeBoardModal: changeBoardModalReducer,
lora: loraReducer, lora: loraReducer,
modelmanager: modelmanagerReducer, modelmanager: modelmanagerReducer,
sdxl: sdxlReducer, sdxl: sdxlReducer,

View File

@ -1,4 +1,4 @@
import { Flex, Text, useColorMode } from '@chakra-ui/react'; import { Box, Flex, useColorMode } from '@chakra-ui/react';
import { motion } from 'framer-motion'; import { motion } from 'framer-motion';
import { ReactNode, memo, useRef } from 'react'; import { ReactNode, memo, useRef } from 'react';
import { mode } from 'theme/util/mode'; import { mode } from 'theme/util/mode';
@ -74,7 +74,7 @@ export const IAIDropOverlay = (props: Props) => {
justifyContent: 'center', justifyContent: 'center',
}} }}
> >
<Text <Box
sx={{ sx={{
fontSize: '2xl', fontSize: '2xl',
fontWeight: 600, fontWeight: 600,
@ -87,7 +87,7 @@ export const IAIDropOverlay = (props: Props) => {
}} }}
> >
{label} {label}
</Text> </Box>
</Flex> </Flex>
</Flex> </Flex>
</motion.div> </motion.div>

View File

@ -53,7 +53,9 @@ const IAIMantineSearchableSelect = (props: IAISelectProps) => {
// wrap onChange to clear search value on select // wrap onChange to clear search value on select
const handleChange = useCallback( const handleChange = useCallback(
(v: string | null) => { (v: string | null) => {
setSearchValue(''); // cannot figure out why we were doing this, but it was causing an issue where if you
// select the currently-selected item, it reset the search value to empty
// setSearchValue('');
if (!onChange) { if (!onChange) {
return; return;

View File

@ -78,7 +78,7 @@ const ImageUploader = (props: ImageUploaderProps) => {
image_category: 'user', image_category: 'user',
is_intermediate: false, is_intermediate: false,
postUploadAction, postUploadAction,
board_id: autoAddBoardId, board_id: autoAddBoardId === 'none' ? undefined : autoAddBoardId,
}); });
}, },
[autoAddBoardId, postUploadAction, uploadImage] [autoAddBoardId, postUploadAction, uploadImage]

View File

@ -49,7 +49,7 @@ export const useImageUploadButton = ({
image_category: 'user', image_category: 'user',
is_intermediate: false, is_intermediate: false,
postUploadAction: postUploadAction ?? { type: 'TOAST' }, postUploadAction: postUploadAction ?? { type: 'TOAST' },
board_id: autoAddBoardId, board_id: autoAddBoardId === 'none' ? undefined : autoAddBoardId,
}); });
}, },
[autoAddBoardId, postUploadAction, uploadImage] [autoAddBoardId, postUploadAction, uploadImage]

View File

@ -33,6 +33,10 @@ const useColorPicker = () => {
1 1
).data; ).data;
if (!(a && r && g && b)) {
return;
}
dispatch(setColorPickerColor({ r, g, b, a })); dispatch(setColorPickerColor({ r, g, b, a }));
}, },
commitColorUnderCursor: () => { commitColorUnderCursor: () => {

View File

@ -727,10 +727,13 @@ export const canvasSlice = createSlice({
state.pastLayerStates.shift(); state.pastLayerStates.shift();
} }
state.layerState.objects.push({ const imageToCommit = images[selectedImageIndex];
...images[selectedImageIndex],
});
if (imageToCommit) {
state.layerState.objects.push({
...imageToCommit,
});
}
state.layerState.stagingArea = { state.layerState.stagingArea = {
...initialLayerState.stagingArea, ...initialLayerState.stagingArea,
}; };

View File

@ -0,0 +1,132 @@
import {
AlertDialog,
AlertDialogBody,
AlertDialogContent,
AlertDialogFooter,
AlertDialogHeader,
AlertDialogOverlay,
Flex,
Text,
} from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIButton from 'common/components/IAIButton';
import IAIMantineSearchableSelect from 'common/components/IAIMantineSearchableSelect';
import { memo, useCallback, useMemo, useRef, useState } from 'react';
import { useListAllBoardsQuery } from 'services/api/endpoints/boards';
import {
useAddImagesToBoardMutation,
useRemoveImagesFromBoardMutation,
} from 'services/api/endpoints/images';
import { changeBoardReset, isModalOpenChanged } from '../store/slice';
const selector = createSelector(
[stateSelector],
({ changeBoardModal }) => {
const { isModalOpen, imagesToChange } = changeBoardModal;
return {
isModalOpen,
imagesToChange,
};
},
defaultSelectorOptions
);
const ChangeBoardModal = () => {
const dispatch = useAppDispatch();
const [selectedBoard, setSelectedBoard] = useState<string | null>();
const { data: boards, isFetching } = useListAllBoardsQuery();
const { imagesToChange, isModalOpen } = useAppSelector(selector);
const [addImagesToBoard] = useAddImagesToBoardMutation();
const [removeImagesFromBoard] = useRemoveImagesFromBoardMutation();
const data = useMemo(() => {
const data: { label: string; value: string }[] = [
{ label: 'Uncategorized', value: 'none' },
];
(boards ?? []).forEach((board) =>
data.push({
label: board.board_name,
value: board.board_id,
})
);
return data;
}, [boards]);
const handleClose = useCallback(() => {
dispatch(changeBoardReset());
dispatch(isModalOpenChanged(false));
}, [dispatch]);
const handleChangeBoard = useCallback(() => {
if (!imagesToChange.length || !selectedBoard) {
return;
}
if (selectedBoard === 'none') {
removeImagesFromBoard({ imageDTOs: imagesToChange });
} else {
addImagesToBoard({
imageDTOs: imagesToChange,
board_id: selectedBoard,
});
}
setSelectedBoard(null);
dispatch(changeBoardReset());
}, [
addImagesToBoard,
dispatch,
imagesToChange,
removeImagesFromBoard,
selectedBoard,
]);
const cancelRef = useRef<HTMLButtonElement>(null);
return (
<AlertDialog
isOpen={isModalOpen}
onClose={handleClose}
leastDestructiveRef={cancelRef}
isCentered
>
<AlertDialogOverlay>
<AlertDialogContent>
<AlertDialogHeader fontSize="lg" fontWeight="bold">
Change Board
</AlertDialogHeader>
<AlertDialogBody>
<Flex sx={{ flexDir: 'column', gap: 4 }}>
<Text>
Moving {`${imagesToChange.length}`} image
{`${imagesToChange.length > 1 ? 's' : ''}`} to board:
</Text>
<IAIMantineSearchableSelect
placeholder={isFetching ? 'Loading...' : 'Select Board'}
disabled={isFetching}
onChange={(v) => setSelectedBoard(v)}
value={selectedBoard}
data={data}
/>
</Flex>
</AlertDialogBody>
<AlertDialogFooter>
<IAIButton ref={cancelRef} onClick={handleClose}>
Cancel
</IAIButton>
<IAIButton colorScheme="accent" onClick={handleChangeBoard} ml={3}>
Move
</IAIButton>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialogOverlay>
</AlertDialog>
);
};
export default memo(ChangeBoardModal);

View File

@ -0,0 +1,6 @@
import { ChangeBoardModalState } from './types';
export const initialState: ChangeBoardModalState = {
isModalOpen: false,
imagesToChange: [],
};

View File

@ -0,0 +1,25 @@
import { PayloadAction, createSlice } from '@reduxjs/toolkit';
import { ImageDTO } from 'services/api/types';
import { initialState } from './initialState';
const changeBoardModal = createSlice({
name: 'changeBoardModal',
initialState,
reducers: {
isModalOpenChanged: (state, action: PayloadAction<boolean>) => {
state.isModalOpen = action.payload;
},
imagesToChangeSelected: (state, action: PayloadAction<ImageDTO[]>) => {
state.imagesToChange = action.payload;
},
changeBoardReset: (state) => {
state.imagesToChange = [];
state.isModalOpen = false;
},
},
});
export const { isModalOpenChanged, imagesToChangeSelected, changeBoardReset } =
changeBoardModal.actions;
export default changeBoardModal.reducer;

View File

@ -0,0 +1,6 @@
import { ImageDTO } from 'services/api/types';
export type ChangeBoardModalState = {
isModalOpen: boolean;
imagesToChange: ImageDTO[];
};

View File

@ -3,6 +3,7 @@ import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { memo, useCallback } from 'react'; import { memo, useCallback } from 'react';
import { FaCopy, FaTrash } from 'react-icons/fa'; import { FaCopy, FaTrash } from 'react-icons/fa';
import { import {
ControlNetConfig,
controlNetDuplicated, controlNetDuplicated,
controlNetRemoved, controlNetRemoved,
controlNetToggled, controlNetToggled,
@ -27,18 +28,27 @@ import ParamControlNetProcessorSelect from './parameters/ParamControlNetProcesso
import ParamControlNetResizeMode from './parameters/ParamControlNetResizeMode'; import ParamControlNetResizeMode from './parameters/ParamControlNetResizeMode';
type ControlNetProps = { type ControlNetProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const ControlNet = (props: ControlNetProps) => { const ControlNet = (props: ControlNetProps) => {
const { controlNetId } = props; const { controlNet } = props;
const { controlNetId } = controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const selector = createSelector( const selector = createSelector(
stateSelector, stateSelector,
({ controlNet }) => { ({ controlNet }) => {
const { isEnabled, shouldAutoConfig } = const cn = controlNet.controlNets[controlNetId];
controlNet.controlNets[controlNetId];
if (!cn) {
return {
isEnabled: false,
shouldAutoConfig: false,
};
}
const { isEnabled, shouldAutoConfig } = cn;
return { isEnabled, shouldAutoConfig }; return { isEnabled, shouldAutoConfig };
}, },
@ -96,7 +106,7 @@ const ControlNet = (props: ControlNetProps) => {
transitionDuration: '0.1s', transitionDuration: '0.1s',
}} }}
> >
<ParamControlNetModel controlNetId={controlNetId} /> <ParamControlNetModel controlNet={controlNet} />
</Box> </Box>
<IAIIconButton <IAIIconButton
size="sm" size="sm"
@ -171,8 +181,8 @@ const ControlNet = (props: ControlNetProps) => {
justifyContent: 'space-between', justifyContent: 'space-between',
}} }}
> >
<ParamControlNetWeight controlNetId={controlNetId} /> <ParamControlNetWeight controlNet={controlNet} />
<ParamControlNetBeginEnd controlNetId={controlNetId} /> <ParamControlNetBeginEnd controlNet={controlNet} />
</Flex> </Flex>
{!isExpanded && ( {!isExpanded && (
<Flex <Flex
@ -184,22 +194,22 @@ const ControlNet = (props: ControlNetProps) => {
aspectRatio: '1/1', aspectRatio: '1/1',
}} }}
> >
<ControlNetImagePreview controlNetId={controlNetId} height={28} /> <ControlNetImagePreview controlNet={controlNet} height={28} />
</Flex> </Flex>
)} )}
</Flex> </Flex>
<Flex sx={{ gap: 2 }}> <Flex sx={{ gap: 2 }}>
<ParamControlNetControlMode controlNetId={controlNetId} /> <ParamControlNetControlMode controlNet={controlNet} />
<ParamControlNetResizeMode controlNetId={controlNetId} /> <ParamControlNetResizeMode controlNet={controlNet} />
</Flex> </Flex>
<ParamControlNetProcessorSelect controlNetId={controlNetId} /> <ParamControlNetProcessorSelect controlNet={controlNet} />
</Flex> </Flex>
{isExpanded && ( {isExpanded && (
<> <>
<ControlNetImagePreview controlNetId={controlNetId} height="392px" /> <ControlNetImagePreview controlNet={controlNet} height="392px" />
<ParamControlNetShouldAutoConfig controlNetId={controlNetId} /> <ParamControlNetShouldAutoConfig controlNet={controlNet} />
<ControlNetProcessorComponent controlNetId={controlNetId} /> <ControlNetProcessorComponent controlNet={controlNet} />
</> </>
)} )}
</Flex> </Flex>

View File

@ -12,50 +12,41 @@ import IAIDndImage from 'common/components/IAIDndImage';
import { memo, useCallback, useMemo, useState } from 'react'; import { memo, useCallback, useMemo, useState } from 'react';
import { useGetImageDTOQuery } from 'services/api/endpoints/images'; import { useGetImageDTOQuery } from 'services/api/endpoints/images';
import { PostUploadAction } from 'services/api/types'; import { PostUploadAction } from 'services/api/types';
import { controlNetImageChanged } from '../store/controlNetSlice'; import {
ControlNetConfig,
controlNetImageChanged,
} from '../store/controlNetSlice';
type Props = { type Props = {
controlNetId: string; controlNet: ControlNetConfig;
height: SystemStyleObject['h']; height: SystemStyleObject['h'];
}; };
const ControlNetImagePreview = (props: Props) => { const selector = createSelector(
const { height, controlNetId } = props;
const dispatch = useAppDispatch();
const selector = useMemo(
() =>
createSelector(
stateSelector, stateSelector,
({ controlNet }) => { ({ controlNet }) => {
const { pendingControlImages } = controlNet; const { pendingControlImages } = controlNet;
const {
controlImage,
processedControlImage,
processorType,
isEnabled,
} = controlNet.controlNets[controlNetId];
return { return {
controlImageName: controlImage,
processedControlImageName: processedControlImage,
processorType,
isEnabled,
pendingControlImages, pendingControlImages,
}; };
}, },
defaultSelectorOptions defaultSelectorOptions
), );
[controlNetId]
);
const ControlNetImagePreview = (props: Props) => {
const { height } = props;
const { const {
controlImageName, controlImage: controlImageName,
processedControlImageName, processedControlImage: processedControlImageName,
processorType, processorType,
pendingControlImages,
isEnabled, isEnabled,
} = useAppSelector(selector); controlNetId,
} = props.controlNet;
const dispatch = useAppDispatch();
const { pendingControlImages } = useAppSelector(selector);
const [isMouseOverImage, setIsMouseOverImage] = useState(false); const [isMouseOverImage, setIsMouseOverImage] = useState(false);

View File

@ -1,8 +1,5 @@
import { createSelector } from '@reduxjs/toolkit'; import { memo } from 'react';
import { stateSelector } from 'app/store/store'; import { ControlNetConfig } from '../store/controlNetSlice';
import { useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { memo, useMemo } from 'react';
import CannyProcessor from './processors/CannyProcessor'; import CannyProcessor from './processors/CannyProcessor';
import ContentShuffleProcessor from './processors/ContentShuffleProcessor'; import ContentShuffleProcessor from './processors/ContentShuffleProcessor';
import HedProcessor from './processors/HedProcessor'; import HedProcessor from './processors/HedProcessor';
@ -17,28 +14,11 @@ import PidiProcessor from './processors/PidiProcessor';
import ZoeDepthProcessor from './processors/ZoeDepthProcessor'; import ZoeDepthProcessor from './processors/ZoeDepthProcessor';
export type ControlNetProcessorProps = { export type ControlNetProcessorProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const ControlNetProcessorComponent = (props: ControlNetProcessorProps) => { const ControlNetProcessorComponent = (props: ControlNetProcessorProps) => {
const { controlNetId } = props; const { controlNetId, isEnabled, processorNode } = props.controlNet;
const selector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { isEnabled, processorNode } =
controlNet.controlNets[controlNetId];
return { isEnabled, processorNode };
},
defaultSelectorOptions
),
[controlNetId]
);
const { isEnabled, processorNode } = useAppSelector(selector);
if (processorNode.type === 'canny_image_processor') { if (processorNode.type === 'canny_image_processor') {
return ( return (

View File

@ -1,34 +1,19 @@
import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAISwitch from 'common/components/IAISwitch'; import IAISwitch from 'common/components/IAISwitch';
import { controlNetAutoConfigToggled } from 'features/controlNet/store/controlNetSlice'; import {
ControlNetConfig,
controlNetAutoConfigToggled,
} from 'features/controlNet/store/controlNetSlice';
import { selectIsBusy } from 'features/system/store/systemSelectors'; import { selectIsBusy } from 'features/system/store/systemSelectors';
import { memo, useCallback, useMemo } from 'react'; import { memo, useCallback } from 'react';
type Props = { type Props = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const ParamControlNetShouldAutoConfig = (props: Props) => { const ParamControlNetShouldAutoConfig = (props: Props) => {
const { controlNetId } = props; const { controlNetId, isEnabled, shouldAutoConfig } = props.controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const selector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { isEnabled, shouldAutoConfig } =
controlNet.controlNets[controlNetId];
return { isEnabled, shouldAutoConfig };
},
defaultSelectorOptions
),
[controlNetId]
);
const { isEnabled, shouldAutoConfig } = useAppSelector(selector);
const isBusy = useAppSelector(selectIsBusy); const isBusy = useAppSelector(selectIsBusy);
const handleShouldAutoConfigChanged = useCallback(() => { const handleShouldAutoConfigChanged = useCallback(() => {

View File

@ -9,48 +9,39 @@ import {
RangeSliderTrack, RangeSliderTrack,
Tooltip, Tooltip,
} from '@chakra-ui/react'; } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit'; import { useAppDispatch } from 'app/store/storeHooks';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { import {
ControlNetConfig,
controlNetBeginStepPctChanged, controlNetBeginStepPctChanged,
controlNetEndStepPctChanged, controlNetEndStepPctChanged,
} from 'features/controlNet/store/controlNetSlice'; } from 'features/controlNet/store/controlNetSlice';
import { memo, useCallback, useMemo } from 'react'; import { memo, useCallback } from 'react';
type Props = { type Props = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const formatPct = (v: number) => `${Math.round(v * 100)}%`; const formatPct = (v: number) => `${Math.round(v * 100)}%`;
const ParamControlNetBeginEnd = (props: Props) => { const ParamControlNetBeginEnd = (props: Props) => {
const { controlNetId } = props; const { beginStepPct, endStepPct, isEnabled, controlNetId } =
props.controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const selector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { beginStepPct, endStepPct, isEnabled } =
controlNet.controlNets[controlNetId];
return { beginStepPct, endStepPct, isEnabled };
},
defaultSelectorOptions
),
[controlNetId]
);
const { beginStepPct, endStepPct, isEnabled } = useAppSelector(selector);
const handleStepPctChanged = useCallback( const handleStepPctChanged = useCallback(
(v: number[]) => { (v: number[]) => {
dispatch( dispatch(
controlNetBeginStepPctChanged({ controlNetId, beginStepPct: v[0] }) controlNetBeginStepPctChanged({
controlNetId,
beginStepPct: v[0] as number,
})
);
dispatch(
controlNetEndStepPctChanged({
controlNetId,
endStepPct: v[1] as number,
})
); );
dispatch(controlNetEndStepPctChanged({ controlNetId, endStepPct: v[1] }));
}, },
[controlNetId, dispatch] [controlNetId, dispatch]
); );

View File

@ -1,16 +1,14 @@
import { createSelector } from '@reduxjs/toolkit'; import { useAppDispatch } from 'app/store/storeHooks';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIMantineSelect from 'common/components/IAIMantineSelect'; import IAIMantineSelect from 'common/components/IAIMantineSelect';
import { import {
ControlModes, ControlModes,
ControlNetConfig,
controlNetControlModeChanged, controlNetControlModeChanged,
} from 'features/controlNet/store/controlNetSlice'; } from 'features/controlNet/store/controlNetSlice';
import { useCallback, useMemo } from 'react'; import { useCallback } from 'react';
type ParamControlNetControlModeProps = { type ParamControlNetControlModeProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const CONTROL_MODE_DATA = [ const CONTROL_MODE_DATA = [
@ -23,23 +21,8 @@ const CONTROL_MODE_DATA = [
export default function ParamControlNetControlMode( export default function ParamControlNetControlMode(
props: ParamControlNetControlModeProps props: ParamControlNetControlModeProps
) { ) {
const { controlNetId } = props; const { controlMode, isEnabled, controlNetId } = props.controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const selector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { controlMode, isEnabled } =
controlNet.controlNets[controlNetId];
return { controlMode, isEnabled };
},
defaultSelectorOptions
),
[controlNetId]
);
const { controlMode, isEnabled } = useAppSelector(selector);
const handleControlModeChange = useCallback( const handleControlModeChange = useCallback(
(controlMode: ControlModes) => { (controlMode: ControlModes) => {

View File

@ -5,7 +5,10 @@ import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIMantineSearchableSelect from 'common/components/IAIMantineSearchableSelect'; import IAIMantineSearchableSelect from 'common/components/IAIMantineSearchableSelect';
import IAIMantineSelectItemWithTooltip from 'common/components/IAIMantineSelectItemWithTooltip'; import IAIMantineSelectItemWithTooltip from 'common/components/IAIMantineSelectItemWithTooltip';
import { controlNetModelChanged } from 'features/controlNet/store/controlNetSlice'; import {
ControlNetConfig,
controlNetModelChanged,
} from 'features/controlNet/store/controlNetSlice';
import { MODEL_TYPE_MAP } from 'features/parameters/types/constants'; import { MODEL_TYPE_MAP } from 'features/parameters/types/constants';
import { modelIdToControlNetModelParam } from 'features/parameters/util/modelIdToControlNetModelParam'; import { modelIdToControlNetModelParam } from 'features/parameters/util/modelIdToControlNetModelParam';
import { selectIsBusy } from 'features/system/store/systemSelectors'; import { selectIsBusy } from 'features/system/store/systemSelectors';
@ -14,30 +17,24 @@ import { memo, useCallback, useMemo } from 'react';
import { useGetControlNetModelsQuery } from 'services/api/endpoints/models'; import { useGetControlNetModelsQuery } from 'services/api/endpoints/models';
type ParamControlNetModelProps = { type ParamControlNetModelProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const selector = createSelector(
stateSelector,
({ generation }) => {
const { model } = generation;
return { mainModel: model };
},
defaultSelectorOptions
);
const ParamControlNetModel = (props: ParamControlNetModelProps) => { const ParamControlNetModel = (props: ParamControlNetModelProps) => {
const { controlNetId } = props; const { controlNetId, model: controlNetModel, isEnabled } = props.controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const isBusy = useAppSelector(selectIsBusy); const isBusy = useAppSelector(selectIsBusy);
const selector = useMemo( const { mainModel } = useAppSelector(selector);
() =>
createSelector(
stateSelector,
({ generation, controlNet }) => {
const { model } = generation;
const controlNetModel = controlNet.controlNets[controlNetId]?.model;
const isEnabled = controlNet.controlNets[controlNetId]?.isEnabled;
return { mainModel: model, controlNetModel, isEnabled };
},
defaultSelectorOptions
),
[controlNetId]
);
const { mainModel, controlNetModel, isEnabled } = useAppSelector(selector);
const { data: controlNetModels } = useGetControlNetModelsQuery(); const { data: controlNetModels } = useGetControlNetModelsQuery();

View File

@ -1,7 +1,6 @@
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { createSelector } from '@reduxjs/toolkit'; import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIMantineSearchableSelect, { import IAIMantineSearchableSelect, {
IAISelectDataType, IAISelectDataType,
@ -9,13 +8,16 @@ import IAIMantineSearchableSelect, {
import { configSelector } from 'features/system/store/configSelectors'; import { configSelector } from 'features/system/store/configSelectors';
import { selectIsBusy } from 'features/system/store/systemSelectors'; import { selectIsBusy } from 'features/system/store/systemSelectors';
import { map } from 'lodash-es'; import { map } from 'lodash-es';
import { memo, useCallback, useMemo } from 'react'; import { memo, useCallback } from 'react';
import { CONTROLNET_PROCESSORS } from '../../store/constants'; import { CONTROLNET_PROCESSORS } from '../../store/constants';
import { controlNetProcessorTypeChanged } from '../../store/controlNetSlice'; import {
ControlNetConfig,
controlNetProcessorTypeChanged,
} from '../../store/controlNetSlice';
import { ControlNetProcessorType } from '../../store/types'; import { ControlNetProcessorType } from '../../store/types';
type ParamControlNetProcessorSelectProps = { type ParamControlNetProcessorSelectProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const selector = createSelector( const selector = createSelector(
@ -52,23 +54,9 @@ const ParamControlNetProcessorSelect = (
props: ParamControlNetProcessorSelectProps props: ParamControlNetProcessorSelectProps
) => { ) => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { controlNetId } = props; const { controlNetId, isEnabled, processorNode } = props.controlNet;
const processorNodeSelector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { isEnabled, processorNode } =
controlNet.controlNets[controlNetId];
return { isEnabled, processorNode };
},
defaultSelectorOptions
),
[controlNetId]
);
const isBusy = useAppSelector(selectIsBusy); const isBusy = useAppSelector(selectIsBusy);
const controlNetProcessors = useAppSelector(selector); const controlNetProcessors = useAppSelector(selector);
const { isEnabled, processorNode } = useAppSelector(processorNodeSelector);
const handleProcessorTypeChanged = useCallback( const handleProcessorTypeChanged = useCallback(
(v: string | null) => { (v: string | null) => {

View File

@ -1,16 +1,14 @@
import { createSelector } from '@reduxjs/toolkit'; import { useAppDispatch } from 'app/store/storeHooks';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIMantineSelect from 'common/components/IAIMantineSelect'; import IAIMantineSelect from 'common/components/IAIMantineSelect';
import { import {
ControlNetConfig,
ResizeModes, ResizeModes,
controlNetResizeModeChanged, controlNetResizeModeChanged,
} from 'features/controlNet/store/controlNetSlice'; } from 'features/controlNet/store/controlNetSlice';
import { useCallback, useMemo } from 'react'; import { useCallback } from 'react';
type ParamControlNetResizeModeProps = { type ParamControlNetResizeModeProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const RESIZE_MODE_DATA = [ const RESIZE_MODE_DATA = [
@ -22,23 +20,8 @@ const RESIZE_MODE_DATA = [
export default function ParamControlNetResizeMode( export default function ParamControlNetResizeMode(
props: ParamControlNetResizeModeProps props: ParamControlNetResizeModeProps
) { ) {
const { controlNetId } = props; const { resizeMode, isEnabled, controlNetId } = props.controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const selector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { resizeMode, isEnabled } =
controlNet.controlNets[controlNetId];
return { resizeMode, isEnabled };
},
defaultSelectorOptions
),
[controlNetId]
);
const { resizeMode, isEnabled } = useAppSelector(selector);
const handleResizeModeChange = useCallback( const handleResizeModeChange = useCallback(
(resizeMode: ResizeModes) => { (resizeMode: ResizeModes) => {

View File

@ -1,32 +1,18 @@
import { createSelector } from '@reduxjs/toolkit'; import { useAppDispatch } from 'app/store/storeHooks';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAISlider from 'common/components/IAISlider'; import IAISlider from 'common/components/IAISlider';
import { controlNetWeightChanged } from 'features/controlNet/store/controlNetSlice'; import {
import { memo, useCallback, useMemo } from 'react'; ControlNetConfig,
controlNetWeightChanged,
} from 'features/controlNet/store/controlNetSlice';
import { memo, useCallback } from 'react';
type ParamControlNetWeightProps = { type ParamControlNetWeightProps = {
controlNetId: string; controlNet: ControlNetConfig;
}; };
const ParamControlNetWeight = (props: ParamControlNetWeightProps) => { const ParamControlNetWeight = (props: ParamControlNetWeightProps) => {
const { controlNetId } = props; const { weight, isEnabled, controlNetId } = props.controlNet;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const selector = useMemo(
() =>
createSelector(
stateSelector,
({ controlNet }) => {
const { weight, isEnabled } = controlNet.controlNets[controlNetId];
return { weight, isEnabled };
},
defaultSelectorOptions
),
[controlNetId]
);
const { weight, isEnabled } = useAppSelector(selector);
const handleWeightChanged = useCallback( const handleWeightChanged = useCallback(
(weight: number) => { (weight: number) => {
dispatch(controlNetWeightChanged({ controlNetId, weight })); dispatch(controlNetWeightChanged({ controlNetId, weight }));

View File

@ -4,7 +4,7 @@ import {
} from './types'; } from './types';
type ControlNetProcessorsDict = Record< type ControlNetProcessorsDict = Record<
string, ControlNetProcessorType,
{ {
type: ControlNetProcessorType | 'none'; type: ControlNetProcessorType | 'none';
label: string; label: string;

View File

@ -96,8 +96,11 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { sourceControlNetId, newControlNetId } = action.payload; const { sourceControlNetId, newControlNetId } = action.payload;
const oldControlNet = state.controlNets[sourceControlNetId];
const newControlnet = cloneDeep(state.controlNets[sourceControlNetId]); if (!oldControlNet) {
return;
}
const newControlnet = cloneDeep(oldControlNet);
newControlnet.controlNetId = newControlNetId; newControlnet.controlNetId = newControlNetId;
state.controlNets[newControlNetId] = newControlnet; state.controlNets[newControlNetId] = newControlnet;
}, },
@ -124,8 +127,11 @@ export const controlNetSlice = createSlice({
action: PayloadAction<{ controlNetId: string }> action: PayloadAction<{ controlNetId: string }>
) => { ) => {
const { controlNetId } = action.payload; const { controlNetId } = action.payload;
state.controlNets[controlNetId].isEnabled = const cn = state.controlNets[controlNetId];
!state.controlNets[controlNetId].isEnabled; if (!cn) {
return;
}
cn.isEnabled = !cn.isEnabled;
}, },
controlNetImageChanged: ( controlNetImageChanged: (
state, state,
@ -135,12 +141,14 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId, controlImage } = action.payload; const { controlNetId, controlImage } = action.payload;
state.controlNets[controlNetId].controlImage = controlImage; const cn = state.controlNets[controlNetId];
state.controlNets[controlNetId].processedControlImage = null; if (!cn) {
if ( return;
controlImage !== null && }
state.controlNets[controlNetId].processorType !== 'none'
) { cn.controlImage = controlImage;
cn.processedControlImage = null;
if (controlImage !== null && cn.processorType !== 'none') {
state.pendingControlImages.push(controlNetId); state.pendingControlImages.push(controlNetId);
} }
}, },
@ -152,8 +160,12 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId, processedControlImage } = action.payload; const { controlNetId, processedControlImage } = action.payload;
state.controlNets[controlNetId].processedControlImage = const cn = state.controlNets[controlNetId];
processedControlImage; if (!cn) {
return;
}
cn.processedControlImage = processedControlImage;
state.pendingControlImages = state.pendingControlImages.filter( state.pendingControlImages = state.pendingControlImages.filter(
(id) => id !== controlNetId (id) => id !== controlNetId
); );
@ -166,10 +178,15 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId, model } = action.payload; const { controlNetId, model } = action.payload;
state.controlNets[controlNetId].model = model; const cn = state.controlNets[controlNetId];
state.controlNets[controlNetId].processedControlImage = null; if (!cn) {
return;
}
if (state.controlNets[controlNetId].shouldAutoConfig) { cn.model = model;
cn.processedControlImage = null;
if (cn.shouldAutoConfig) {
let processorType: ControlNetProcessorType | undefined = undefined; let processorType: ControlNetProcessorType | undefined = undefined;
for (const modelSubstring in CONTROLNET_MODEL_DEFAULT_PROCESSORS) { for (const modelSubstring in CONTROLNET_MODEL_DEFAULT_PROCESSORS) {
@ -180,14 +197,13 @@ export const controlNetSlice = createSlice({
} }
if (processorType) { if (processorType) {
state.controlNets[controlNetId].processorType = processorType; cn.processorType = processorType;
state.controlNets[controlNetId].processorNode = CONTROLNET_PROCESSORS[ cn.processorNode = CONTROLNET_PROCESSORS[processorType]
processorType .default as RequiredControlNetProcessorNode;
].default as RequiredControlNetProcessorNode;
} else { } else {
state.controlNets[controlNetId].processorType = 'none'; cn.processorType = 'none';
state.controlNets[controlNetId].processorNode = CONTROLNET_PROCESSORS cn.processorNode = CONTROLNET_PROCESSORS.none
.none.default as RequiredControlNetProcessorNode; .default as RequiredControlNetProcessorNode;
} }
} }
}, },
@ -196,28 +212,48 @@ export const controlNetSlice = createSlice({
action: PayloadAction<{ controlNetId: string; weight: number }> action: PayloadAction<{ controlNetId: string; weight: number }>
) => { ) => {
const { controlNetId, weight } = action.payload; const { controlNetId, weight } = action.payload;
state.controlNets[controlNetId].weight = weight; const cn = state.controlNets[controlNetId];
if (!cn) {
return;
}
cn.weight = weight;
}, },
controlNetBeginStepPctChanged: ( controlNetBeginStepPctChanged: (
state, state,
action: PayloadAction<{ controlNetId: string; beginStepPct: number }> action: PayloadAction<{ controlNetId: string; beginStepPct: number }>
) => { ) => {
const { controlNetId, beginStepPct } = action.payload; const { controlNetId, beginStepPct } = action.payload;
state.controlNets[controlNetId].beginStepPct = beginStepPct; const cn = state.controlNets[controlNetId];
if (!cn) {
return;
}
cn.beginStepPct = beginStepPct;
}, },
controlNetEndStepPctChanged: ( controlNetEndStepPctChanged: (
state, state,
action: PayloadAction<{ controlNetId: string; endStepPct: number }> action: PayloadAction<{ controlNetId: string; endStepPct: number }>
) => { ) => {
const { controlNetId, endStepPct } = action.payload; const { controlNetId, endStepPct } = action.payload;
state.controlNets[controlNetId].endStepPct = endStepPct; const cn = state.controlNets[controlNetId];
if (!cn) {
return;
}
cn.endStepPct = endStepPct;
}, },
controlNetControlModeChanged: ( controlNetControlModeChanged: (
state, state,
action: PayloadAction<{ controlNetId: string; controlMode: ControlModes }> action: PayloadAction<{ controlNetId: string; controlMode: ControlModes }>
) => { ) => {
const { controlNetId, controlMode } = action.payload; const { controlNetId, controlMode } = action.payload;
state.controlNets[controlNetId].controlMode = controlMode; const cn = state.controlNets[controlNetId];
if (!cn) {
return;
}
cn.controlMode = controlMode;
}, },
controlNetResizeModeChanged: ( controlNetResizeModeChanged: (
state, state,
@ -227,7 +263,12 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId, resizeMode } = action.payload; const { controlNetId, resizeMode } = action.payload;
state.controlNets[controlNetId].resizeMode = resizeMode; const cn = state.controlNets[controlNetId];
if (!cn) {
return;
}
cn.resizeMode = resizeMode;
}, },
controlNetProcessorParamsChanged: ( controlNetProcessorParamsChanged: (
state, state,
@ -240,12 +281,17 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId, changes } = action.payload; const { controlNetId, changes } = action.payload;
const processorNode = state.controlNets[controlNetId].processorNode; const cn = state.controlNets[controlNetId];
state.controlNets[controlNetId].processorNode = { if (!cn) {
return;
}
const processorNode = cn.processorNode;
cn.processorNode = {
...processorNode, ...processorNode,
...changes, ...changes,
}; };
state.controlNets[controlNetId].shouldAutoConfig = false; cn.shouldAutoConfig = false;
}, },
controlNetProcessorTypeChanged: ( controlNetProcessorTypeChanged: (
state, state,
@ -255,12 +301,16 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId, processorType } = action.payload; const { controlNetId, processorType } = action.payload;
state.controlNets[controlNetId].processedControlImage = null; const cn = state.controlNets[controlNetId];
state.controlNets[controlNetId].processorType = processorType; if (!cn) {
state.controlNets[controlNetId].processorNode = CONTROLNET_PROCESSORS[ return;
processorType }
].default as RequiredControlNetProcessorNode;
state.controlNets[controlNetId].shouldAutoConfig = false; cn.processedControlImage = null;
cn.processorType = processorType;
cn.processorNode = CONTROLNET_PROCESSORS[processorType]
.default as RequiredControlNetProcessorNode;
cn.shouldAutoConfig = false;
}, },
controlNetAutoConfigToggled: ( controlNetAutoConfigToggled: (
state, state,
@ -269,37 +319,36 @@ export const controlNetSlice = createSlice({
}> }>
) => { ) => {
const { controlNetId } = action.payload; const { controlNetId } = action.payload;
const newShouldAutoConfig = const cn = state.controlNets[controlNetId];
!state.controlNets[controlNetId].shouldAutoConfig; if (!cn) {
return;
}
const newShouldAutoConfig = !cn.shouldAutoConfig;
if (newShouldAutoConfig) { if (newShouldAutoConfig) {
// manage the processor for the user // manage the processor for the user
let processorType: ControlNetProcessorType | undefined = undefined; let processorType: ControlNetProcessorType | undefined = undefined;
for (const modelSubstring in CONTROLNET_MODEL_DEFAULT_PROCESSORS) { for (const modelSubstring in CONTROLNET_MODEL_DEFAULT_PROCESSORS) {
if ( if (cn.model?.model_name.includes(modelSubstring)) {
state.controlNets[controlNetId].model?.model_name.includes(
modelSubstring
)
) {
processorType = CONTROLNET_MODEL_DEFAULT_PROCESSORS[modelSubstring]; processorType = CONTROLNET_MODEL_DEFAULT_PROCESSORS[modelSubstring];
break; break;
} }
} }
if (processorType) { if (processorType) {
state.controlNets[controlNetId].processorType = processorType; cn.processorType = processorType;
state.controlNets[controlNetId].processorNode = CONTROLNET_PROCESSORS[ cn.processorNode = CONTROLNET_PROCESSORS[processorType]
processorType .default as RequiredControlNetProcessorNode;
].default as RequiredControlNetProcessorNode;
} else { } else {
state.controlNets[controlNetId].processorType = 'none'; cn.processorType = 'none';
state.controlNets[controlNetId].processorNode = CONTROLNET_PROCESSORS cn.processorNode = CONTROLNET_PROCESSORS.none
.none.default as RequiredControlNetProcessorNode; .default as RequiredControlNetProcessorNode;
} }
} }
state.controlNets[controlNetId].shouldAutoConfig = newShouldAutoConfig; cn.shouldAutoConfig = newShouldAutoConfig;
}, },
controlNetReset: () => { controlNetReset: () => {
return { ...initialControlNetState }; return { ...initialControlNetState };
@ -307,9 +356,11 @@ export const controlNetSlice = createSlice({
}, },
extraReducers: (builder) => { extraReducers: (builder) => {
builder.addCase(controlNetImageProcessed, (state, action) => { builder.addCase(controlNetImageProcessed, (state, action) => {
if ( const cn = state.controlNets[action.payload.controlNetId];
state.controlNets[action.payload.controlNetId].controlImage !== null if (!cn) {
) { return;
}
if (cn.controlImage !== null) {
state.pendingControlImages.push(action.payload.controlNetId); state.pendingControlImages.push(action.payload.controlNetId);
} }
}); });

View File

@ -15,30 +15,42 @@ import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIButton from 'common/components/IAIButton'; import IAIButton from 'common/components/IAIButton';
import IAISwitch from 'common/components/IAISwitch'; import IAISwitch from 'common/components/IAISwitch';
import { setShouldConfirmOnDelete } from 'features/system/store/systemSlice'; import { setShouldConfirmOnDelete } from 'features/system/store/systemSlice';
import { stateSelector } from 'app/store/store'; import { stateSelector } from 'app/store/store';
import { some } from 'lodash-es';
import { ChangeEvent, memo, useCallback, useRef } from 'react'; import { ChangeEvent, memo, useCallback, useRef } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { imageDeletionConfirmed } from '../store/actions'; import { imageDeletionConfirmed } from '../store/actions';
import { selectImageUsage } from '../store/imageDeletionSelectors'; import { getImageUsage, selectImageUsage } from '../store/selectors';
import { import { imageDeletionCanceled, isModalOpenChanged } from '../store/slice';
imageToDeleteCleared,
isModalOpenChanged,
} from '../store/imageDeletionSlice';
import ImageUsageMessage from './ImageUsageMessage'; import ImageUsageMessage from './ImageUsageMessage';
import { ImageUsage } from '../store/types';
const selector = createSelector( const selector = createSelector(
[stateSelector, selectImageUsage], [stateSelector, selectImageUsage],
({ system, config, imageDeletion }, imageUsage) => { (state, imagesUsage) => {
const { system, config, deleteImageModal } = state;
const { shouldConfirmOnDelete } = system; const { shouldConfirmOnDelete } = system;
const { canRestoreDeletedImagesFromBin } = config; const { canRestoreDeletedImagesFromBin } = config;
const { imageToDelete, isModalOpen } = imageDeletion; const { imagesToDelete, isModalOpen } = deleteImageModal;
const allImageUsage = (imagesToDelete ?? []).map(({ image_name }) =>
getImageUsage(state, image_name)
);
const imageUsageSummary: ImageUsage = {
isInitialImage: some(allImageUsage, (i) => i.isInitialImage),
isCanvasImage: some(allImageUsage, (i) => i.isCanvasImage),
isNodesImage: some(allImageUsage, (i) => i.isNodesImage),
isControlNetImage: some(allImageUsage, (i) => i.isControlNetImage),
};
return { return {
shouldConfirmOnDelete, shouldConfirmOnDelete,
canRestoreDeletedImagesFromBin, canRestoreDeletedImagesFromBin,
imageToDelete, imagesToDelete,
imageUsage, imagesUsage,
isModalOpen, isModalOpen,
imageUsageSummary,
}; };
}, },
defaultSelectorOptions defaultSelectorOptions
@ -51,9 +63,10 @@ const DeleteImageModal = () => {
const { const {
shouldConfirmOnDelete, shouldConfirmOnDelete,
canRestoreDeletedImagesFromBin, canRestoreDeletedImagesFromBin,
imageToDelete, imagesToDelete,
imageUsage, imagesUsage,
isModalOpen, isModalOpen,
imageUsageSummary,
} = useAppSelector(selector); } = useAppSelector(selector);
const handleChangeShouldConfirmOnDelete = useCallback( const handleChangeShouldConfirmOnDelete = useCallback(
@ -63,17 +76,19 @@ const DeleteImageModal = () => {
); );
const handleClose = useCallback(() => { const handleClose = useCallback(() => {
dispatch(imageToDeleteCleared()); dispatch(imageDeletionCanceled());
dispatch(isModalOpenChanged(false)); dispatch(isModalOpenChanged(false));
}, [dispatch]); }, [dispatch]);
const handleDelete = useCallback(() => { const handleDelete = useCallback(() => {
if (!imageToDelete || !imageUsage) { if (!imagesToDelete.length || !imagesUsage.length) {
return; return;
} }
dispatch(imageToDeleteCleared()); dispatch(imageDeletionCanceled());
dispatch(imageDeletionConfirmed({ imageDTO: imageToDelete, imageUsage })); dispatch(
}, [dispatch, imageToDelete, imageUsage]); imageDeletionConfirmed({ imageDTOs: imagesToDelete, imagesUsage })
);
}, [dispatch, imagesToDelete, imagesUsage]);
const cancelRef = useRef<HTMLButtonElement>(null); const cancelRef = useRef<HTMLButtonElement>(null);
@ -92,7 +107,7 @@ const DeleteImageModal = () => {
<AlertDialogBody> <AlertDialogBody>
<Flex direction="column" gap={3}> <Flex direction="column" gap={3}>
<ImageUsageMessage imageUsage={imageUsage} /> <ImageUsageMessage imageUsage={imageUsageSummary} />
<Divider /> <Divider />
<Text> <Text>
{canRestoreDeletedImagesFromBin {canRestoreDeletedImagesFromBin

View File

@ -3,6 +3,6 @@ import { ImageDTO } from 'services/api/types';
import { ImageUsage } from './types'; import { ImageUsage } from './types';
export const imageDeletionConfirmed = createAction<{ export const imageDeletionConfirmed = createAction<{
imageDTO: ImageDTO; imageDTOs: ImageDTO[];
imageUsage: ImageUsage; imagesUsage: ImageUsage[];
}>('imageDeletion/imageDeletionConfirmed'); }>('deleteImageModal/imageDeletionConfirmed');

View File

@ -0,0 +1,6 @@
import { DeleteImageState } from './types';
export const initialDeleteImageState: DeleteImageState = {
imagesToDelete: [],
isModalOpen: false,
};

View File

@ -39,17 +39,17 @@ export const getImageUsage = (state: RootState, image_name: string) => {
export const selectImageUsage = createSelector( export const selectImageUsage = createSelector(
[(state: RootState) => state], [(state: RootState) => state],
(state) => { (state) => {
const { imageToDelete } = state.imageDeletion; const { imagesToDelete } = state.deleteImageModal;
if (!imageToDelete) { if (!imagesToDelete.length) {
return; return [];
} }
const { image_name } = imageToDelete; const imagesUsage = imagesToDelete.map((i) =>
getImageUsage(state, i.image_name)
);
const imageUsage = getImageUsage(state, image_name); return imagesUsage;
return imageUsage;
}, },
defaultSelectorOptions defaultSelectorOptions
); );

View File

@ -0,0 +1,28 @@
import { PayloadAction, createSlice } from '@reduxjs/toolkit';
import { ImageDTO } from 'services/api/types';
import { initialDeleteImageState } from './initialState';
const deleteImageModal = createSlice({
name: 'deleteImageModal',
initialState: initialDeleteImageState,
reducers: {
isModalOpenChanged: (state, action: PayloadAction<boolean>) => {
state.isModalOpen = action.payload;
},
imagesToDeleteSelected: (state, action: PayloadAction<ImageDTO[]>) => {
state.imagesToDelete = action.payload;
},
imageDeletionCanceled: (state) => {
state.imagesToDelete = [];
state.isModalOpen = false;
},
},
});
export const {
isModalOpenChanged,
imagesToDeleteSelected,
imageDeletionCanceled,
} = deleteImageModal.actions;
export default deleteImageModal.reducer;

View File

@ -0,0 +1,13 @@
import { ImageDTO } from 'services/api/types';
export type DeleteImageState = {
imagesToDelete: ImageDTO[];
isModalOpen: boolean;
};
export type ImageUsage = {
isInitialImage: boolean;
isCanvasImage: boolean;
isNodesImage: boolean;
isControlNetImage: boolean;
};

View File

@ -56,7 +56,7 @@ const BoardAutoAddSelect = () => {
return; return;
} }
dispatch(autoAddBoardIdChanged(v === 'none' ? undefined : v)); dispatch(autoAddBoardIdChanged(v));
}, },
[dispatch] [dispatch]
); );

View File

@ -11,10 +11,11 @@ import { BoardDTO } from 'services/api/types';
import { menuListMotionProps } from 'theme/components/menu'; import { menuListMotionProps } from 'theme/components/menu';
import GalleryBoardContextMenuItems from './GalleryBoardContextMenuItems'; import GalleryBoardContextMenuItems from './GalleryBoardContextMenuItems';
import NoBoardContextMenuItems from './NoBoardContextMenuItems'; import NoBoardContextMenuItems from './NoBoardContextMenuItems';
import { BoardId } from 'features/gallery/store/types';
type Props = { type Props = {
board?: BoardDTO; board?: BoardDTO;
board_id?: string; board_id: BoardId;
children: ContextMenuProps<HTMLDivElement>['children']; children: ContextMenuProps<HTMLDivElement>['children'];
setBoardToDelete?: (board?: BoardDTO) => void; setBoardToDelete?: (board?: BoardDTO) => void;
}; };

View File

@ -1,43 +0,0 @@
import { createSelector } from '@reduxjs/toolkit';
import { AddToBatchDropData } from 'app/components/ImageDnd/typesafeDnd';
import { stateSelector } from 'app/store/store';
import { useAppSelector } from 'app/store/storeHooks';
import { boardIdSelected } from 'features/gallery/store/gallerySlice';
import { useCallback } from 'react';
import { FaLayerGroup } from 'react-icons/fa';
import { useDispatch } from 'react-redux';
import GenericBoard from './GenericBoard';
const selector = createSelector(stateSelector, (state) => {
return {
count: state.gallery.batchImageNames.length,
};
});
const BatchBoard = ({ isSelected }: { isSelected: boolean }) => {
const dispatch = useDispatch();
const { count } = useAppSelector(selector);
const handleBatchBoardClick = useCallback(() => {
dispatch(boardIdSelected('batch'));
}, [dispatch]);
const droppableData: AddToBatchDropData = {
id: 'batch-board',
actionType: 'ADD_TO_BATCH',
};
return (
<GenericBoard
board_id="batch"
droppableData={droppableData}
onClick={handleBatchBoardClick}
isSelected={isSelected}
icon={FaLayerGroup}
label="Batch"
badgeCount={count}
/>
);
};
export default BatchBoard;

View File

@ -15,10 +15,9 @@ import NoBoardBoard from './NoBoardBoard';
const selector = createSelector( const selector = createSelector(
[stateSelector], [stateSelector],
({ boards, gallery }) => { ({ gallery }) => {
const { searchText } = boards; const { selectedBoardId, boardSearchText } = gallery;
const { selectedBoardId } = gallery; return { selectedBoardId, boardSearchText };
return { selectedBoardId, searchText };
}, },
defaultSelectorOptions defaultSelectorOptions
); );
@ -29,11 +28,11 @@ type Props = {
const BoardsList = (props: Props) => { const BoardsList = (props: Props) => {
const { isOpen } = props; const { isOpen } = props;
const { selectedBoardId, searchText } = useAppSelector(selector); const { selectedBoardId, boardSearchText } = useAppSelector(selector);
const { data: boards } = useListAllBoardsQuery(); const { data: boards } = useListAllBoardsQuery();
const filteredBoards = searchText const filteredBoards = boardSearchText
? boards?.filter((board) => ? boards?.filter((board) =>
board.board_name.toLowerCase().includes(searchText.toLowerCase()) board.board_name.toLowerCase().includes(boardSearchText.toLowerCase())
) )
: boards; : boards;
const [boardToDelete, setBoardToDelete] = useState<BoardDTO>(); const [boardToDelete, setBoardToDelete] = useState<BoardDTO>();
@ -75,7 +74,7 @@ const BoardsList = (props: Props) => {
}} }}
> >
<GridItem sx={{ p: 1.5 }}> <GridItem sx={{ p: 1.5 }}>
<NoBoardBoard isSelected={selectedBoardId === undefined} /> <NoBoardBoard isSelected={selectedBoardId === 'none'} />
</GridItem> </GridItem>
{filteredBoards && {filteredBoards &&
filteredBoards.map((board) => ( filteredBoards.map((board) => (

View File

@ -9,7 +9,7 @@ import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store'; import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { setBoardSearchText } from 'features/gallery/store/boardSlice'; import { boardSearchTextChanged } from 'features/gallery/store/gallerySlice';
import { import {
ChangeEvent, ChangeEvent,
KeyboardEvent, KeyboardEvent,
@ -21,27 +21,27 @@ import {
const selector = createSelector( const selector = createSelector(
[stateSelector], [stateSelector],
({ boards }) => { ({ gallery }) => {
const { searchText } = boards; const { boardSearchText } = gallery;
return { searchText }; return { boardSearchText };
}, },
defaultSelectorOptions defaultSelectorOptions
); );
const BoardsSearch = () => { const BoardsSearch = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { searchText } = useAppSelector(selector); const { boardSearchText } = useAppSelector(selector);
const inputRef = useRef<HTMLInputElement>(null); const inputRef = useRef<HTMLInputElement>(null);
const handleBoardSearch = useCallback( const handleBoardSearch = useCallback(
(searchTerm: string) => { (searchTerm: string) => {
dispatch(setBoardSearchText(searchTerm)); dispatch(boardSearchTextChanged(searchTerm));
}, },
[dispatch] [dispatch]
); );
const clearBoardSearch = useCallback(() => { const clearBoardSearch = useCallback(() => {
dispatch(setBoardSearchText('')); dispatch(boardSearchTextChanged(''));
}, [dispatch]); }, [dispatch]);
const handleKeydown = useCallback( const handleKeydown = useCallback(
@ -74,11 +74,11 @@ const BoardsSearch = () => {
<Input <Input
ref={inputRef} ref={inputRef}
placeholder="Search Boards..." placeholder="Search Boards..."
value={searchText} value={boardSearchText}
onKeyDown={handleKeydown} onKeyDown={handleKeydown}
onChange={handleChange} onChange={handleChange}
/> />
{searchText && searchText.length && ( {boardSearchText && boardSearchText.length && (
<InputRightElement> <InputRightElement>
<IconButton <IconButton
onClick={clearBoardSearch} onClick={clearBoardSearch}

View File

@ -7,10 +7,11 @@ import {
Icon, Icon,
Image, Image,
Text, Text,
Tooltip,
} from '@chakra-ui/react'; } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit'; import { createSelector } from '@reduxjs/toolkit';
import { skipToken } from '@reduxjs/toolkit/dist/query'; import { skipToken } from '@reduxjs/toolkit/dist/query';
import { MoveBoardDropData } from 'app/components/ImageDnd/typesafeDnd'; import { AddToBoardDropData } from 'app/components/ImageDnd/typesafeDnd';
import { stateSelector } from 'app/store/store'; import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
@ -22,7 +23,11 @@ import {
} from 'features/gallery/store/gallerySlice'; } from 'features/gallery/store/gallerySlice';
import { memo, useCallback, useMemo, useState } from 'react'; import { memo, useCallback, useMemo, useState } from 'react';
import { FaUser } from 'react-icons/fa'; import { FaUser } from 'react-icons/fa';
import { useUpdateBoardMutation } from 'services/api/endpoints/boards'; import {
useGetBoardAssetsTotalQuery,
useGetBoardImagesTotalQuery,
useUpdateBoardMutation,
} from 'services/api/endpoints/boards';
import { useGetImageDTOQuery } from 'services/api/endpoints/images'; import { useGetImageDTOQuery } from 'services/api/endpoints/images';
import { BoardDTO } from 'services/api/types'; import { BoardDTO } from 'services/api/types';
import AutoAddIcon from '../AutoAddIcon'; import AutoAddIcon from '../AutoAddIcon';
@ -67,6 +72,18 @@ const GalleryBoard = memo(
const handleMouseOut = useCallback(() => { const handleMouseOut = useCallback(() => {
setIsHovered(false); setIsHovered(false);
}, []); }, []);
const { data: imagesTotal } = useGetBoardImagesTotalQuery(board.board_id);
const { data: assetsTotal } = useGetBoardAssetsTotalQuery(board.board_id);
const tooltip = useMemo(() => {
if (!imagesTotal || !assetsTotal) {
return undefined;
}
return `${imagesTotal} image${
imagesTotal > 1 ? 's' : ''
}, ${assetsTotal} asset${assetsTotal > 1 ? 's' : ''}`;
}, [assetsTotal, imagesTotal]);
const { currentData: coverImage } = useGetImageDTOQuery( const { currentData: coverImage } = useGetImageDTOQuery(
board.cover_image_name ?? skipToken board.cover_image_name ?? skipToken
); );
@ -84,10 +101,10 @@ const GalleryBoard = memo(
const [updateBoard, { isLoading: isUpdateBoardLoading }] = const [updateBoard, { isLoading: isUpdateBoardLoading }] =
useUpdateBoardMutation(); useUpdateBoardMutation();
const droppableData: MoveBoardDropData = useMemo( const droppableData: AddToBoardDropData = useMemo(
() => ({ () => ({
id: board_id, id: board_id,
actionType: 'MOVE_BOARD', actionType: 'ADD_TO_BOARD',
context: { boardId: board_id }, context: { boardId: board_id },
}), }),
[board_id] [board_id]
@ -148,6 +165,7 @@ const GalleryBoard = memo(
setBoardToDelete={setBoardToDelete} setBoardToDelete={setBoardToDelete}
> >
{(ref) => ( {(ref) => (
<Tooltip label={tooltip} openDelay={1000} hasArrow>
<Flex <Flex
ref={ref} ref={ref}
onClick={handleSelectBoard} onClick={handleSelectBoard}
@ -278,6 +296,7 @@ const GalleryBoard = memo(
dropLabel={<Text fontSize="md">Move</Text>} dropLabel={<Text fontSize="md">Move</Text>}
/> />
</Flex> </Flex>
</Tooltip>
)} )}
</BoardContextMenu> </BoardContextMenu>
</Flex> </Flex>

View File

@ -1,6 +1,6 @@
import { Box, Flex, Image, Text } from '@chakra-ui/react'; import { Box, Flex, Image, Text } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit'; import { createSelector } from '@reduxjs/toolkit';
import { MoveBoardDropData } from 'app/components/ImageDnd/typesafeDnd'; import { RemoveFromBoardDropData } from 'app/components/ImageDnd/typesafeDnd';
import { stateSelector } from 'app/store/store'; import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
@ -15,6 +15,7 @@ import { memo, useCallback, useMemo, useState } from 'react';
import { useBoardName } from 'services/api/hooks/useBoardName'; import { useBoardName } from 'services/api/hooks/useBoardName';
import AutoAddIcon from '../AutoAddIcon'; import AutoAddIcon from '../AutoAddIcon';
import BoardContextMenu from '../BoardContextMenu'; import BoardContextMenu from '../BoardContextMenu';
interface Props { interface Props {
isSelected: boolean; isSelected: boolean;
} }
@ -33,26 +34,27 @@ const NoBoardBoard = memo(({ isSelected }: Props) => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { autoAddBoardId, autoAssignBoardOnClick, isProcessing } = const { autoAddBoardId, autoAssignBoardOnClick, isProcessing } =
useAppSelector(selector); useAppSelector(selector);
const boardName = useBoardName(undefined); const boardName = useBoardName('none');
const handleSelectBoard = useCallback(() => { const handleSelectBoard = useCallback(() => {
dispatch(boardIdSelected(undefined)); dispatch(boardIdSelected('none'));
if (autoAssignBoardOnClick && !isProcessing) { if (autoAssignBoardOnClick && !isProcessing) {
dispatch(autoAddBoardIdChanged(undefined)); dispatch(autoAddBoardIdChanged('none'));
} }
}, [dispatch, autoAssignBoardOnClick, isProcessing]); }, [dispatch, autoAssignBoardOnClick, isProcessing]);
const [isHovered, setIsHovered] = useState(false); const [isHovered, setIsHovered] = useState(false);
const handleMouseOver = useCallback(() => { const handleMouseOver = useCallback(() => {
setIsHovered(true); setIsHovered(true);
}, []); }, []);
const handleMouseOut = useCallback(() => { const handleMouseOut = useCallback(() => {
setIsHovered(false); setIsHovered(false);
}, []); }, []);
const droppableData: MoveBoardDropData = useMemo( const droppableData: RemoveFromBoardDropData = useMemo(
() => ({ () => ({
id: 'no_board', id: 'no_board',
actionType: 'MOVE_BOARD', actionType: 'REMOVE_FROM_BOARD',
context: { boardId: undefined },
}), }),
[] []
); );
@ -72,7 +74,7 @@ const NoBoardBoard = memo(({ isSelected }: Props) => {
h: 'full', h: 'full',
}} }}
> >
<BoardContextMenu> <BoardContextMenu board_id="none">
{(ref) => ( {(ref) => (
<Flex <Flex
ref={ref} ref={ref}
@ -99,17 +101,6 @@ const NoBoardBoard = memo(({ isSelected }: Props) => {
alignItems: 'center', alignItems: 'center',
}} }}
> >
{/* <Icon
boxSize={12}
as={FaBucket}
sx={{
opacity: 0.7,
color: 'base.500',
_dark: {
color: 'base.500',
},
}}
/> */}
<Image <Image
src={InvokeAILogoImage} src={InvokeAILogoImage}
alt="invoke-ai-logo" alt="invoke-ai-logo"
@ -125,19 +116,7 @@ const NoBoardBoard = memo(({ isSelected }: Props) => {
}} }}
/> />
</Flex> </Flex>
{/* <Flex {autoAddBoardId === 'none' && <AutoAddIcon />}
sx={{
position: 'absolute',
insetInlineEnd: 0,
top: 0,
p: 1,
}}
>
<Badge variant="solid" sx={BASE_BADGE_STYLES}>
{totalImages}/{totalAssets}
</Badge>
</Flex> */}
{!autoAddBoardId && <AutoAddIcon />}
<Flex <Flex
sx={{ sx={{
position: 'absolute', position: 'absolute',

View File

@ -11,20 +11,20 @@ import {
} from '@chakra-ui/react'; } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit'; import { createSelector } from '@reduxjs/toolkit';
import { skipToken } from '@reduxjs/toolkit/dist/query'; import { skipToken } from '@reduxjs/toolkit/dist/query';
import { ImageUsage } from 'app/contexts/AddImageToBoardContext';
import { stateSelector } from 'app/store/store'; import { stateSelector } from 'app/store/store';
import { useAppSelector } from 'app/store/storeHooks'; import { useAppSelector } from 'app/store/storeHooks';
import IAIButton from 'common/components/IAIButton'; import IAIButton from 'common/components/IAIButton';
import ImageUsageMessage from 'features/imageDeletion/components/ImageUsageMessage'; import ImageUsageMessage from 'features/deleteImageModal/components/ImageUsageMessage';
import { getImageUsage } from 'features/imageDeletion/store/imageDeletionSelectors'; import { getImageUsage } from 'features/deleteImageModal/store/selectors';
import { ImageUsage } from 'features/deleteImageModal/store/types';
import { some } from 'lodash-es'; import { some } from 'lodash-es';
import { memo, useCallback, useMemo, useRef } from 'react'; import { memo, useCallback, useMemo, useRef } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { useListAllImageNamesForBoardQuery } from 'services/api/endpoints/boards';
import { import {
useDeleteBoardAndImagesMutation, useDeleteBoardAndImagesMutation,
useDeleteBoardMutation, useDeleteBoardMutation,
useListAllImageNamesForBoardQuery, } from 'services/api/endpoints/images';
} from 'services/api/endpoints/boards';
import { BoardDTO } from 'services/api/types'; import { BoardDTO } from 'services/api/types';
type Props = { type Props = {
@ -32,7 +32,7 @@ type Props = {
setBoardToDelete: (board?: BoardDTO) => void; setBoardToDelete: (board?: BoardDTO) => void;
}; };
const DeleteImageModal = (props: Props) => { const DeleteBoardModal = (props: Props) => {
const { boardToDelete, setBoardToDelete } = props; const { boardToDelete, setBoardToDelete } = props;
const { t } = useTranslation(); const { t } = useTranslation();
const canRestoreDeletedImagesFromBin = useAppSelector( const canRestoreDeletedImagesFromBin = useAppSelector(
@ -49,13 +49,10 @@ const DeleteImageModal = (props: Props) => {
); );
const imageUsageSummary: ImageUsage = { const imageUsageSummary: ImageUsage = {
isInitialImage: some(allImageUsage, (usage) => usage.isInitialImage), isInitialImage: some(allImageUsage, (i) => i.isInitialImage),
isCanvasImage: some(allImageUsage, (usage) => usage.isCanvasImage), isCanvasImage: some(allImageUsage, (i) => i.isCanvasImage),
isNodesImage: some(allImageUsage, (usage) => usage.isNodesImage), isNodesImage: some(allImageUsage, (i) => i.isNodesImage),
isControlNetImage: some( isControlNetImage: some(allImageUsage, (i) => i.isControlNetImage),
allImageUsage,
(usage) => usage.isControlNetImage
),
}; };
return { imageUsageSummary }; return { imageUsageSummary };
}), }),
@ -176,4 +173,4 @@ const DeleteImageModal = (props: Props) => {
); );
}; };
export default memo(DeleteImageModal); export default memo(DeleteBoardModal);

View File

@ -1,93 +0,0 @@
import {
AlertDialog,
AlertDialogBody,
AlertDialogContent,
AlertDialogFooter,
AlertDialogHeader,
AlertDialogOverlay,
Box,
Flex,
Spinner,
Text,
} from '@chakra-ui/react';
import IAIButton from 'common/components/IAIButton';
import IAIMantineSearchableSelect from 'common/components/IAIMantineSearchableSelect';
import { memo, useContext, useRef, useState } from 'react';
import { useListAllBoardsQuery } from 'services/api/endpoints/boards';
import { AddImageToBoardContext } from '../../../../app/contexts/AddImageToBoardContext';
const UpdateImageBoardModal = () => {
// const boards = useSelector(selectBoardsAll);
const { data: boards, isFetching } = useListAllBoardsQuery();
const { isOpen, onClose, handleAddToBoard, image } = useContext(
AddImageToBoardContext
);
const [selectedBoard, setSelectedBoard] = useState<string | null>();
const cancelRef = useRef<HTMLButtonElement>(null);
const currentBoard = boards?.find(
(board) => board.board_id === image?.board_id
);
return (
<AlertDialog
isOpen={isOpen}
leastDestructiveRef={cancelRef}
onClose={onClose}
isCentered
>
<AlertDialogOverlay>
<AlertDialogContent>
<AlertDialogHeader fontSize="lg" fontWeight="bold">
{currentBoard ? 'Move Image to Board' : 'Add Image to Board'}
</AlertDialogHeader>
<AlertDialogBody>
<Box>
<Flex direction="column" gap={3}>
{currentBoard && (
<Text>
Moving this image from{' '}
<strong>{currentBoard.board_name}</strong> to
</Text>
)}
{isFetching ? (
<Spinner />
) : (
<IAIMantineSearchableSelect
placeholder="Select Board"
onChange={(v) => setSelectedBoard(v)}
value={selectedBoard}
data={(boards ?? []).map((board) => ({
label: board.board_name,
value: board.board_id,
}))}
/>
)}
</Flex>
</Box>
</AlertDialogBody>
<AlertDialogFooter>
<IAIButton onClick={onClose}>Cancel</IAIButton>
<IAIButton
isDisabled={!selectedBoard}
colorScheme="accent"
onClick={() => {
if (selectedBoard) {
handleAddToBoard(selectedBoard);
}
}}
ml={3}
>
{currentBoard ? 'Move' : 'Add'}
</IAIButton>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialogOverlay>
</AlertDialog>
);
};
export default memo(UpdateImageBoardModal);

View File

@ -9,16 +9,14 @@ import {
MenuButton, MenuButton,
MenuList, MenuList,
} from '@chakra-ui/react'; } from '@chakra-ui/react';
// import { runESRGAN, runFacetool } from 'app/socketio/actions';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import IAIIconButton from 'common/components/IAIIconButton'; import IAIIconButton from 'common/components/IAIIconButton';
import { skipToken } from '@reduxjs/toolkit/dist/query'; import { skipToken } from '@reduxjs/toolkit/dist/query';
import { useAppToaster } from 'app/components/Toaster'; import { useAppToaster } from 'app/components/Toaster';
import { upscaleRequested } from 'app/store/middleware/listenerMiddleware/listeners/upscaleRequested'; import { upscaleRequested } from 'app/store/middleware/listenerMiddleware/listeners/upscaleRequested';
import { stateSelector } from 'app/store/store'; import { stateSelector } from 'app/store/store';
import { DeleteImageButton } from 'features/imageDeletion/components/DeleteImageButton'; import { DeleteImageButton } from 'features/deleteImageModal/components/DeleteImageButton';
import { imageToDeleteSelected } from 'features/imageDeletion/store/imageDeletionSlice'; import { imagesToDeleteSelected } from 'features/deleteImageModal/store/slice';
import ParamUpscalePopover from 'features/parameters/components/Parameters/Upscale/ParamUpscaleSettings'; import ParamUpscalePopover from 'features/parameters/components/Parameters/Upscale/ParamUpscaleSettings';
import { useRecallParameters } from 'features/parameters/hooks/useRecallParameters'; import { useRecallParameters } from 'features/parameters/hooks/useRecallParameters';
import { initialImageSelected } from 'features/parameters/store/actions'; import { initialImageSelected } from 'features/parameters/store/actions';
@ -109,13 +107,13 @@ const CurrentImageButtons = (props: CurrentImageButtonsProps) => {
); );
const { currentData: imageDTO } = useGetImageDTOQuery( const { currentData: imageDTO } = useGetImageDTOQuery(
lastSelectedImage ?? skipToken lastSelectedImage?.image_name ?? skipToken
); );
const { currentData: metadataData } = useGetImageMetadataQuery( const { currentData: metadataData } = useGetImageMetadataQuery(
debounceState.isPending() debounceState.isPending()
? skipToken ? skipToken
: debouncedMetadataQueryArg ?? skipToken : debouncedMetadataQueryArg?.image_name ?? skipToken
); );
const metadata = metadataData?.metadata; const metadata = metadataData?.metadata;
@ -173,7 +171,7 @@ const CurrentImageButtons = (props: CurrentImageButtonsProps) => {
if (!imageDTO) { if (!imageDTO) {
return; return;
} }
dispatch(imageToDeleteSelected(imageDTO)); dispatch(imagesToDeleteSelected([imageDTO]));
}, [dispatch, imageDTO]); }, [dispatch, imageDTO]);
useHotkeys( useHotkeys(

View File

@ -32,7 +32,7 @@ export const imagesSelector = createSelector(
return { return {
shouldShowImageDetails, shouldShowImageDetails,
shouldHidePreview, shouldHidePreview,
imageName: lastSelectedImage, imageName: lastSelectedImage?.image_name,
progressImage, progressImage,
shouldShowProgressInViewer, shouldShowProgressInViewer,
shouldAntialiasProgressImage, shouldAntialiasProgressImage,
@ -57,8 +57,6 @@ const CurrentImagePreview = () => {
const { const {
handlePrevImage, handlePrevImage,
handleNextImage, handleNextImage,
prevImageId,
nextImageId,
isOnLastImage, isOnLastImage,
handleLoadMoreImages, handleLoadMoreImages,
areMoreImagesAvailable, areMoreImagesAvailable,
@ -70,7 +68,7 @@ const CurrentImagePreview = () => {
() => { () => {
handlePrevImage(); handlePrevImage();
}, },
[prevImageId] [handlePrevImage]
); );
useHotkeys( useHotkeys(
@ -85,11 +83,11 @@ const CurrentImagePreview = () => {
} }
}, },
[ [
nextImageId,
isOnLastImage, isOnLastImage,
areMoreImagesAvailable, areMoreImagesAvailable,
handleLoadMoreImages, handleLoadMoreImages,
isFetching, isFetching,
handleNextImage,
] ]
); );

View File

@ -5,17 +5,19 @@ import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIIconButton from 'common/components/IAIIconButton'; import IAIIconButton from 'common/components/IAIIconButton';
import IAIPopover from 'common/components/IAIPopover'; import IAIPopover from 'common/components/IAIPopover';
import IAISimpleCheckbox from 'common/components/IAISimpleCheckbox';
import IAISlider from 'common/components/IAISlider'; import IAISlider from 'common/components/IAISlider';
import IAISwitch from 'common/components/IAISwitch';
import { import {
autoAssignBoardOnClickChanged, autoAssignBoardOnClickChanged,
setGalleryImageMinimumWidth, setGalleryImageMinimumWidth,
shouldAutoSwitchChanged, shouldAutoSwitchChanged,
shouldShowDeleteButtonChanged,
} from 'features/gallery/store/gallerySlice'; } from 'features/gallery/store/gallerySlice';
import { ChangeEvent } from 'react'; import { ChangeEvent, useCallback } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { FaWrench } from 'react-icons/fa'; import { FaWrench } from 'react-icons/fa';
import BoardAutoAddSelect from './Boards/BoardAutoAddSelect'; import BoardAutoAddSelect from './Boards/BoardAutoAddSelect';
import IAISimpleCheckbox from 'common/components/IAISimpleCheckbox';
const selector = createSelector( const selector = createSelector(
[stateSelector], [stateSelector],
@ -24,12 +26,14 @@ const selector = createSelector(
galleryImageMinimumWidth, galleryImageMinimumWidth,
shouldAutoSwitch, shouldAutoSwitch,
autoAssignBoardOnClick, autoAssignBoardOnClick,
shouldShowDeleteButton,
} = state.gallery; } = state.gallery;
return { return {
galleryImageMinimumWidth, galleryImageMinimumWidth,
shouldAutoSwitch, shouldAutoSwitch,
autoAssignBoardOnClick, autoAssignBoardOnClick,
shouldShowDeleteButton,
}; };
}, },
defaultSelectorOptions defaultSelectorOptions
@ -39,12 +43,37 @@ const GallerySettingsPopover = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { t } = useTranslation(); const { t } = useTranslation();
const { galleryImageMinimumWidth, shouldAutoSwitch, autoAssignBoardOnClick } = const {
useAppSelector(selector); galleryImageMinimumWidth,
shouldAutoSwitch,
autoAssignBoardOnClick,
shouldShowDeleteButton,
} = useAppSelector(selector);
const handleChangeGalleryImageMinimumWidth = (v: number) => { const handleChangeGalleryImageMinimumWidth = useCallback(
(v: number) => {
dispatch(setGalleryImageMinimumWidth(v)); dispatch(setGalleryImageMinimumWidth(v));
}; },
[dispatch]
);
const handleResetGalleryImageMinimumWidth = useCallback(() => {
dispatch(setGalleryImageMinimumWidth(64));
}, [dispatch]);
const handleChangeAutoSwitch = useCallback(
(e: ChangeEvent<HTMLInputElement>) => {
dispatch(shouldAutoSwitchChanged(e.target.checked));
},
[dispatch]
);
const handleChangeShowDeleteButton = useCallback(
(e: ChangeEvent<HTMLInputElement>) => {
dispatch(shouldShowDeleteButtonChanged(e.target.checked));
},
[dispatch]
);
return ( return (
<IAIPopover <IAIPopover
@ -57,7 +86,7 @@ const GallerySettingsPopover = () => {
/> />
} }
> >
<Flex direction="column" gap={4}> <Flex direction="column" gap={2}>
<IAISlider <IAISlider
value={galleryImageMinimumWidth} value={galleryImageMinimumWidth}
onChange={handleChangeGalleryImageMinimumWidth} onChange={handleChangeGalleryImageMinimumWidth}
@ -66,14 +95,17 @@ const GallerySettingsPopover = () => {
hideTooltip={true} hideTooltip={true}
label={t('gallery.galleryImageSize')} label={t('gallery.galleryImageSize')}
withReset withReset
handleReset={() => dispatch(setGalleryImageMinimumWidth(64))} handleReset={handleResetGalleryImageMinimumWidth}
/> />
<IAISimpleCheckbox <IAISwitch
label={t('gallery.autoSwitchNewImages')} label={t('gallery.autoSwitchNewImages')}
isChecked={shouldAutoSwitch} isChecked={shouldAutoSwitch}
onChange={(e: ChangeEvent<HTMLInputElement>) => onChange={handleChangeAutoSwitch}
dispatch(shouldAutoSwitchChanged(e.target.checked)) />
} <IAISwitch
label="Show Delete Button"
isChecked={shouldShowDeleteButton}
onChange={handleChangeShowDeleteButton}
/> />
<IAISimpleCheckbox <IAISimpleCheckbox
label={t('gallery.autoAssignBoardOnClick')} label={t('gallery.autoAssignBoardOnClick')}

View File

@ -4,28 +4,29 @@ import { MouseEvent, memo, useCallback } from 'react';
import { ImageDTO } from 'services/api/types'; import { ImageDTO } from 'services/api/types';
import { menuListMotionProps } from 'theme/components/menu'; import { menuListMotionProps } from 'theme/components/menu';
import SingleSelectionMenuItems from './SingleSelectionMenuItems'; import SingleSelectionMenuItems from './SingleSelectionMenuItems';
import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { useAppSelector } from 'app/store/storeHooks';
import MultipleSelectionMenuItems from './MultipleSelectionMenuItems';
type Props = { type Props = {
imageDTO: ImageDTO | undefined; imageDTO: ImageDTO | undefined;
children: ContextMenuProps<HTMLDivElement>['children']; children: ContextMenuProps<HTMLDivElement>['children'];
}; };
const selector = createSelector(
[stateSelector],
({ gallery }) => {
const selectionCount = gallery.selection.length;
return { selectionCount };
},
defaultSelectorOptions
);
const ImageContextMenu = ({ imageDTO, children }: Props) => { const ImageContextMenu = ({ imageDTO, children }: Props) => {
// const selector = useMemo( const { selectionCount } = useAppSelector(selector);
// () =>
// createSelector(
// [stateSelector],
// ({ gallery }) => {
// const selectionCount = gallery.selection.length;
// return { selectionCount };
// },
// defaultSelectorOptions
// ),
// []
// );
// const { selectionCount } = useAppSelector(selector);
const skipEvent = useCallback((e: MouseEvent<HTMLDivElement>) => { const skipEvent = useCallback((e: MouseEvent<HTMLDivElement>) => {
e.preventDefault(); e.preventDefault();
@ -38,8 +39,24 @@ const ImageContextMenu = ({ imageDTO, children }: Props) => {
bg: 'transparent', bg: 'transparent',
_hover: { bg: 'transparent' }, _hover: { bg: 'transparent' },
}} }}
renderMenu={() => renderMenu={() => {
imageDTO ? ( if (!imageDTO) {
return null;
}
if (selectionCount > 1) {
return (
<MenuList
sx={{ visibility: 'visible !important' }}
motionProps={menuListMotionProps}
onContextMenu={skipEvent}
>
<MultipleSelectionMenuItems />
</MenuList>
);
}
return (
<MenuList <MenuList
sx={{ visibility: 'visible !important' }} sx={{ visibility: 'visible !important' }}
motionProps={menuListMotionProps} motionProps={menuListMotionProps}
@ -47,8 +64,8 @@ const ImageContextMenu = ({ imageDTO, children }: Props) => {
> >
<SingleSelectionMenuItems imageDTO={imageDTO} /> <SingleSelectionMenuItems imageDTO={imageDTO} />
</MenuList> </MenuList>
) : null );
} }}
> >
{children} {children}
</ContextMenu> </ContextMenu>

View File

@ -1,30 +1,30 @@
import { MenuItem } from '@chakra-ui/react'; import { MenuItem } from '@chakra-ui/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import {
imagesToChangeSelected,
isModalOpenChanged,
} from 'features/changeBoardModal/store/slice';
import { imagesToDeleteSelected } from 'features/deleteImageModal/store/slice';
import { useCallback } from 'react'; import { useCallback } from 'react';
import { FaFolder, FaFolderPlus, FaTrash } from 'react-icons/fa'; import { FaFolder, FaTrash } from 'react-icons/fa';
const MultipleSelectionMenuItems = () => { const MultipleSelectionMenuItems = () => {
const handleAddSelectionToBoard = useCallback(() => { const dispatch = useAppDispatch();
// TODO: add selection to board const selection = useAppSelector((state) => state.gallery.selection);
}, []);
const handleChangeBoard = useCallback(() => {
dispatch(imagesToChangeSelected(selection));
dispatch(isModalOpenChanged(true));
}, [dispatch, selection]);
const handleDeleteSelection = useCallback(() => { const handleDeleteSelection = useCallback(() => {
// TODO: delete all selected images dispatch(imagesToDeleteSelected(selection));
}, []); }, [dispatch, selection]);
const handleAddSelectionToBatch = useCallback(() => {
// TODO: add selection to batch
}, []);
return ( return (
<> <>
<MenuItem icon={<FaFolder />} onClickCapture={handleAddSelectionToBoard}> <MenuItem icon={<FaFolder />} onClickCapture={handleChangeBoard}>
Move Selection to Board Change Board
</MenuItem>
<MenuItem
icon={<FaFolderPlus />}
onClickCapture={handleAddSelectionToBatch}
>
Add Selection to Batch
</MenuItem> </MenuItem>
<MenuItem <MenuItem
sx={{ color: 'error.600', _dark: { color: 'error.300' } }} sx={{ color: 'error.600', _dark: { color: 'error.300' } }}

View File

@ -1,22 +1,22 @@
import { MenuItem } from '@chakra-ui/react'; import { MenuItem } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit';
import { skipToken } from '@reduxjs/toolkit/dist/query'; import { skipToken } from '@reduxjs/toolkit/dist/query';
import { useAppToaster } from 'app/components/Toaster'; import { useAppToaster } from 'app/components/Toaster';
import { stateSelector } from 'app/store/store'; import { useAppDispatch } from 'app/store/storeHooks';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { import {
resizeAndScaleCanvas, resizeAndScaleCanvas,
setInitialCanvasImage, setInitialCanvasImage,
} from 'features/canvas/store/canvasSlice'; } from 'features/canvas/store/canvasSlice';
import { imagesAddedToBatch } from 'features/gallery/store/gallerySlice'; import {
import { imageToDeleteSelected } from 'features/imageDeletion/store/imageDeletionSlice'; imagesToChangeSelected,
isModalOpenChanged,
} from 'features/changeBoardModal/store/slice';
import { imagesToDeleteSelected } from 'features/deleteImageModal/store/slice';
import { useRecallParameters } from 'features/parameters/hooks/useRecallParameters'; import { useRecallParameters } from 'features/parameters/hooks/useRecallParameters';
import { initialImageSelected } from 'features/parameters/store/actions'; import { initialImageSelected } from 'features/parameters/store/actions';
import { useFeatureStatus } from 'features/system/hooks/useFeatureStatus'; import { useFeatureStatus } from 'features/system/hooks/useFeatureStatus';
import { useCopyImageToClipboard } from 'features/ui/hooks/useCopyImageToClipboard'; import { useCopyImageToClipboard } from 'features/ui/hooks/useCopyImageToClipboard';
import { setActiveTab } from 'features/ui/store/uiSlice'; import { setActiveTab } from 'features/ui/store/uiSlice';
import { memo, useCallback, useContext, useMemo } from 'react'; import { memo, useCallback } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { import {
FaAsterisk, FaAsterisk,
@ -29,13 +29,9 @@ import {
FaShare, FaShare,
FaTrash, FaTrash,
} from 'react-icons/fa'; } from 'react-icons/fa';
import { import { useGetImageMetadataQuery } from 'services/api/endpoints/images';
useGetImageMetadataQuery,
useRemoveImageFromBoardMutation,
} from 'services/api/endpoints/images';
import { ImageDTO } from 'services/api/types'; import { ImageDTO } from 'services/api/types';
import { useDebounce } from 'use-debounce'; import { useDebounce } from 'use-debounce';
import { AddImageToBoardContext } from '../../../../app/contexts/AddImageToBoardContext';
import { sentImageToCanvas, sentImageToImg2Img } from '../../store/actions'; import { sentImageToCanvas, sentImageToImg2Img } from '../../store/actions';
type SingleSelectionMenuItemsProps = { type SingleSelectionMenuItemsProps = {
@ -45,32 +41,12 @@ type SingleSelectionMenuItemsProps = {
const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => { const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => {
const { imageDTO } = props; const { imageDTO } = props;
const selector = useMemo(
() =>
createSelector(
[stateSelector],
({ gallery }) => {
const isInBatch = gallery.batchImageNames.includes(
imageDTO.image_name
);
return { isInBatch };
},
defaultSelectorOptions
),
[imageDTO.image_name]
);
const { isInBatch } = useAppSelector(selector);
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { t } = useTranslation(); const { t } = useTranslation();
const toaster = useAppToaster(); const toaster = useAppToaster();
const isCanvasEnabled = useFeatureStatus('unifiedCanvas').isFeatureEnabled; const isCanvasEnabled = useFeatureStatus('unifiedCanvas').isFeatureEnabled;
const isBatchEnabled = useFeatureStatus('batches').isFeatureEnabled;
const { onClickAddToBoard } = useContext(AddImageToBoardContext);
const [debouncedMetadataQueryArg, debounceState] = useDebounce( const [debouncedMetadataQueryArg, debounceState] = useDebounce(
imageDTO.image_name, imageDTO.image_name,
@ -92,14 +68,12 @@ const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => {
if (!imageDTO) { if (!imageDTO) {
return; return;
} }
dispatch(imageToDeleteSelected(imageDTO)); dispatch(imagesToDeleteSelected([imageDTO]));
}, [dispatch, imageDTO]); }, [dispatch, imageDTO]);
const { recallBothPrompts, recallSeed, recallAllParameters } = const { recallBothPrompts, recallSeed, recallAllParameters } =
useRecallParameters(); useRecallParameters();
const [removeFromBoard] = useRemoveImageFromBoardMutation();
// Recall parameters handlers // Recall parameters handlers
const handleRecallPrompt = useCallback(() => { const handleRecallPrompt = useCallback(() => {
recallBothPrompts( recallBothPrompts(
@ -144,20 +118,10 @@ const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => {
recallAllParameters(metadata); recallAllParameters(metadata);
}, [metadata, recallAllParameters]); }, [metadata, recallAllParameters]);
const handleAddToBoard = useCallback(() => { const handleChangeBoard = useCallback(() => {
onClickAddToBoard(imageDTO); dispatch(imagesToChangeSelected([imageDTO]));
}, [imageDTO, onClickAddToBoard]); dispatch(isModalOpenChanged(true));
}, [dispatch, imageDTO]);
const handleRemoveFromBoard = useCallback(() => {
if (!imageDTO.board_id) {
return;
}
removeFromBoard({ imageDTO });
}, [imageDTO, removeFromBoard]);
const handleAddToBatch = useCallback(() => {
dispatch(imagesAddedToBatch([imageDTO.image_name]));
}, [dispatch, imageDTO.image_name]);
const handleCopyImage = useCallback(() => { const handleCopyImage = useCallback(() => {
copyImageToClipboard(imageDTO.image_url); copyImageToClipboard(imageDTO.image_url);
@ -229,23 +193,9 @@ const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => {
{t('parameters.sendToUnifiedCanvas')} {t('parameters.sendToUnifiedCanvas')}
</MenuItem> </MenuItem>
)} )}
{isBatchEnabled && ( <MenuItem icon={<FaFolder />} onClickCapture={handleChangeBoard}>
<MenuItem Change Board
icon={<FaFolder />}
isDisabled={isInBatch}
onClickCapture={handleAddToBatch}
>
Add to Batch
</MenuItem> </MenuItem>
)}
<MenuItem icon={<FaFolder />} onClickCapture={handleAddToBoard}>
{imageDTO.board_id ? 'Change Board' : 'Add to Board'}
</MenuItem>
{imageDTO.board_id && (
<MenuItem icon={<FaFolder />} onClickCapture={handleRemoveFromBoard}>
Remove from Board
</MenuItem>
)}
<MenuItem <MenuItem
sx={{ color: 'error.600', _dark: { color: 'error.300' } }} sx={{ color: 'error.600', _dark: { color: 'error.300' } }}
icon={<FaTrash />} icon={<FaTrash />}

View File

@ -20,16 +20,14 @@ import BoardsList from './Boards/BoardsList/BoardsList';
import GalleryBoardName from './GalleryBoardName'; import GalleryBoardName from './GalleryBoardName';
import GalleryPinButton from './GalleryPinButton'; import GalleryPinButton from './GalleryPinButton';
import GallerySettingsPopover from './GallerySettingsPopover'; import GallerySettingsPopover from './GallerySettingsPopover';
import BatchImageGrid from './ImageGrid/BatchImageGrid';
import GalleryImageGrid from './ImageGrid/GalleryImageGrid'; import GalleryImageGrid from './ImageGrid/GalleryImageGrid';
const selector = createSelector( const selector = createSelector(
[stateSelector], [stateSelector],
(state) => { (state) => {
const { selectedBoardId, galleryView } = state.gallery; const { galleryView } = state.gallery;
return { return {
selectedBoardId,
galleryView, galleryView,
}; };
}, },
@ -39,7 +37,7 @@ const selector = createSelector(
const ImageGalleryContent = () => { const ImageGalleryContent = () => {
const resizeObserverRef = useRef<HTMLDivElement>(null); const resizeObserverRef = useRef<HTMLDivElement>(null);
const galleryGridRef = useRef<HTMLDivElement>(null); const galleryGridRef = useRef<HTMLDivElement>(null);
const { selectedBoardId, galleryView } = useAppSelector(selector); const { galleryView } = useAppSelector(selector);
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { isOpen: isBoardListOpen, onToggle: onToggleBoardList } = const { isOpen: isBoardListOpen, onToggle: onToggleBoardList } =
useDisclosure(); useDisclosure();
@ -130,12 +128,7 @@ const ImageGalleryContent = () => {
</TabList> </TabList>
</Tabs> </Tabs>
</Flex> </Flex>
{selectedBoardId === 'batch' ? (
<BatchImageGrid />
) : (
<GalleryImageGrid /> <GalleryImageGrid />
)}
</Flex> </Flex>
</VStack> </VStack>
); );

View File

@ -1,122 +0,0 @@
import { Box } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit';
import { TypesafeDraggableData } from 'app/components/ImageDnd/typesafeDnd';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIDndImage from 'common/components/IAIDndImage';
import IAIErrorLoadingImageFallback from 'common/components/IAIErrorLoadingImageFallback';
import IAIFillSkeleton from 'common/components/IAIFillSkeleton';
import ImageContextMenu from 'features/gallery/components/ImageContextMenu/ImageContextMenu';
import { imagesRemovedFromBatch } from 'features/gallery/store/gallerySlice';
import { memo, useCallback, useMemo } from 'react';
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
const makeSelector = (image_name: string) =>
createSelector(
[stateSelector],
(state) => ({
selectionCount: state.gallery.selection.length,
selection: state.gallery.selection,
isSelected: state.gallery.selection.includes(image_name),
}),
defaultSelectorOptions
);
type BatchImageProps = {
imageName: string;
};
const BatchImage = (props: BatchImageProps) => {
const dispatch = useAppDispatch();
const { imageName } = props;
const {
currentData: imageDTO,
isLoading,
isError,
} = useGetImageDTOQuery(imageName);
const selector = useMemo(() => makeSelector(imageName), [imageName]);
const { isSelected, selectionCount, selection } = useAppSelector(selector);
const handleClickRemove = useCallback(() => {
dispatch(imagesRemovedFromBatch([imageName]));
}, [dispatch, imageName]);
// const handleClick = useCallback(
// (e: MouseEvent<HTMLDivElement>) => {
// if (e.shiftKey) {
// dispatch(imageRangeEndSelected(imageName));
// } else if (e.ctrlKey || e.metaKey) {
// dispatch(imageSelectionToggled(imageName));
// } else {
// dispatch(imageSelected(imageName));
// }
// },
// [dispatch, imageName]
// );
const draggableData = useMemo<TypesafeDraggableData | undefined>(() => {
if (selectionCount > 1) {
return {
id: 'batch',
payloadType: 'IMAGE_NAMES',
payload: { image_names: selection },
};
}
if (imageDTO) {
return {
id: 'batch',
payloadType: 'IMAGE_DTO',
payload: { imageDTO },
};
}
}, [imageDTO, selection, selectionCount]);
if (isLoading) {
return <IAIFillSkeleton />;
}
if (isError || !imageDTO) {
return <IAIErrorLoadingImageFallback />;
}
return (
<Box sx={{ w: 'full', h: 'full', touchAction: 'none' }}>
<ImageContextMenu imageDTO={imageDTO}>
{(ref) => (
<Box
position="relative"
key={imageName}
userSelect="none"
ref={ref}
sx={{
display: 'flex',
justifyContent: 'center',
alignItems: 'center',
aspectRatio: '1/1',
}}
>
<IAIDndImage
// onClick={handleClick}
imageDTO={imageDTO}
draggableData={draggableData}
isSelected={isSelected}
minSize={0}
onClickReset={handleClickRemove}
isDropDisabled={true}
imageSx={{ w: 'full', h: 'full' }}
isUploadDisabled={true}
resetTooltip="Remove from batch"
withResetIcon
thumbnail
/>
</Box>
)}
</ImageContextMenu>
</Box>
);
};
export default memo(BatchImage);

View File

@ -1,87 +0,0 @@
import { Box } from '@chakra-ui/react';
import { useAppSelector } from 'app/store/storeHooks';
import { useOverlayScrollbars } from 'overlayscrollbars-react';
import { memo, useEffect, useRef, useState } from 'react';
import { useTranslation } from 'react-i18next';
import { FaImage } from 'react-icons/fa';
import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { IAINoContentFallback } from 'common/components/IAIImageFallback';
import { VirtuosoGrid } from 'react-virtuoso';
import BatchImage from './BatchImage';
import ItemContainer from './ImageGridItemContainer';
import ListContainer from './ImageGridListContainer';
const selector = createSelector(
[stateSelector],
(state) => {
return {
imageNames: state.gallery.batchImageNames,
};
},
defaultSelectorOptions
);
const BatchImageGrid = () => {
const { t } = useTranslation();
const rootRef = useRef(null);
const [scroller, setScroller] = useState<HTMLElement | null>(null);
const [initialize, osInstance] = useOverlayScrollbars({
defer: true,
options: {
scrollbars: {
visibility: 'auto',
autoHide: 'leave',
autoHideDelay: 1300,
theme: 'os-theme-dark',
},
overflow: { x: 'hidden' },
},
});
const { imageNames } = useAppSelector(selector);
useEffect(() => {
const { current: root } = rootRef;
if (scroller && root) {
initialize({
target: root,
elements: {
viewport: scroller,
},
});
}
return () => osInstance()?.destroy();
}, [scroller, initialize, osInstance]);
if (imageNames.length) {
return (
<Box ref={rootRef} data-overlayscrollbars="" h="100%">
<VirtuosoGrid
style={{ height: '100%' }}
data={imageNames}
components={{
Item: ItemContainer,
List: ListContainer,
}}
scrollerRef={setScroller}
itemContent={(index, imageName) => (
<BatchImage key={imageName} imageName={imageName} />
)}
/>
</Box>
);
}
return (
<IAINoContentFallback
label={t('gallery.noImagesInGallery')}
icon={FaImage}
/>
);
};
export default memo(BatchImageGrid);

View File

@ -1,27 +1,18 @@
import { Box, Flex } from '@chakra-ui/react'; import { Box, Flex } from '@chakra-ui/react';
import { createSelector } from '@reduxjs/toolkit'; import {
import { TypesafeDraggableData } from 'app/components/ImageDnd/typesafeDnd'; ImageDTOsDraggableData,
import { stateSelector } from 'app/store/store'; ImageDraggableData,
TypesafeDraggableData,
} from 'app/components/ImageDnd/typesafeDnd';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import IAIDndImage from 'common/components/IAIDndImage'; import IAIDndImage from 'common/components/IAIDndImage';
import IAIFillSkeleton from 'common/components/IAIFillSkeleton'; import IAIFillSkeleton from 'common/components/IAIFillSkeleton';
import { imageSelected } from 'features/gallery/store/gallerySlice'; import { useMultiselect } from 'features/gallery/hooks/useMultiselect.ts';
import { imageToDeleteSelected } from 'features/imageDeletion/store/imageDeletionSlice'; import { imagesToDeleteSelected } from 'features/deleteImageModal/store/slice';
import { MouseEvent, memo, useCallback, useMemo } from 'react'; import { MouseEvent, memo, useCallback, useMemo } from 'react';
import { FaTrash } from 'react-icons/fa';
import { useGetImageDTOQuery } from 'services/api/endpoints/images'; import { useGetImageDTOQuery } from 'services/api/endpoints/images';
export const makeSelector = (image_name: string) =>
createSelector(
[stateSelector],
({ gallery }) => ({
isSelected: gallery.selection.includes(image_name),
selectionCount: gallery.selection.length,
selection: gallery.selection,
}),
defaultSelectorOptions
);
interface HoverableImageProps { interface HoverableImageProps {
imageName: string; imageName: string;
} }
@ -30,22 +21,12 @@ const GalleryImage = (props: HoverableImageProps) => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { imageName } = props; const { imageName } = props;
const { currentData: imageDTO } = useGetImageDTOQuery(imageName); const { currentData: imageDTO } = useGetImageDTOQuery(imageName);
const localSelector = useMemo(() => makeSelector(imageName), [imageName]); const shouldShowDeleteButton = useAppSelector(
(state) => state.gallery.shouldShowDeleteButton
);
const { isSelected, selectionCount, selection } = const { handleClick, isSelected, selection, selectionCount } =
useAppSelector(localSelector); useMultiselect(imageDTO);
const handleClick = useCallback(() => {
// disable multiselect for now
// if (e.shiftKey) {
// dispatch(imageRangeEndSelected(imageName));
// } else if (e.ctrlKey || e.metaKey) {
// dispatch(imageSelectionToggled(imageName));
// } else {
// dispatch(imageSelected(imageName));
// }
dispatch(imageSelected(imageName));
}, [dispatch, imageName]);
const handleDelete = useCallback( const handleDelete = useCallback(
(e: MouseEvent<HTMLButtonElement>) => { (e: MouseEvent<HTMLButtonElement>) => {
@ -53,26 +34,28 @@ const GalleryImage = (props: HoverableImageProps) => {
if (!imageDTO) { if (!imageDTO) {
return; return;
} }
dispatch(imageToDeleteSelected(imageDTO)); dispatch(imagesToDeleteSelected([imageDTO]));
}, },
[dispatch, imageDTO] [dispatch, imageDTO]
); );
const draggableData = useMemo<TypesafeDraggableData | undefined>(() => { const draggableData = useMemo<TypesafeDraggableData | undefined>(() => {
if (selectionCount > 1) { if (selectionCount > 1) {
return { const data: ImageDTOsDraggableData = {
id: 'gallery-image', id: 'gallery-image',
payloadType: 'IMAGE_NAMES', payloadType: 'IMAGE_DTOS',
payload: { image_names: selection }, payload: { imageDTOs: selection },
}; };
return data;
} }
if (imageDTO) { if (imageDTO) {
return { const data: ImageDraggableData = {
id: 'gallery-image', id: 'gallery-image',
payloadType: 'IMAGE_DTO', payloadType: 'IMAGE_DTO',
payload: { imageDTO }, payload: { imageDTO },
}; };
return data;
} }
}, [imageDTO, selection, selectionCount]); }, [imageDTO, selection, selectionCount]);
@ -103,9 +86,9 @@ const GalleryImage = (props: HoverableImageProps) => {
isUploadDisabled={true} isUploadDisabled={true}
thumbnail={true} thumbnail={true}
withHoverOverlay withHoverOverlay
// resetIcon={<FaTrash />} resetIcon={<FaTrash />}
// resetTooltip="Delete image" resetTooltip="Delete image"
// withResetIcon // removed bc it's too easy to accidentally delete images withResetIcon={shouldShowDeleteButton} // removed bc it's too easy to accidentally delete images
/> />
</Flex> </Flex>
</Box> </Box>

View File

@ -1,6 +1,6 @@
import { useRecallParameters } from 'features/parameters/hooks/useRecallParameters'; import { useRecallParameters } from 'features/parameters/hooks/useRecallParameters';
import { useCallback } from 'react'; import { useCallback } from 'react';
import { UnsafeImageMetadata } from 'services/api/endpoints/images'; import { UnsafeImageMetadata } from 'services/api/types';
import ImageMetadataItem from './ImageMetadataItem'; import ImageMetadataItem from './ImageMetadataItem';
type Props = { type Props = {

View File

@ -0,0 +1,93 @@
import { createSelector } from '@reduxjs/toolkit';
import { stateSelector } from 'app/store/store';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { selectListImagesBaseQueryArgs } from 'features/gallery/store/gallerySelectors';
import { uniq } from 'lodash-es';
import { MouseEvent, useCallback, useMemo } from 'react';
import { useListImagesQuery } from 'services/api/endpoints/images';
import { ImageDTO } from 'services/api/types';
import { selectionChanged } from '../store/gallerySlice';
import { imagesSelectors } from 'services/api/util';
const selector = createSelector(
[stateSelector, selectListImagesBaseQueryArgs],
({ gallery }, queryArgs) => {
const selection = gallery.selection;
return {
queryArgs,
selection,
};
},
defaultSelectorOptions
);
export const useMultiselect = (imageDTO?: ImageDTO) => {
const dispatch = useAppDispatch();
const { queryArgs, selection } = useAppSelector(selector);
const { imageDTOs } = useListImagesQuery(queryArgs, {
selectFromResult: (result) => ({
imageDTOs: result.data ? imagesSelectors.selectAll(result.data) : [],
}),
});
const handleClick = useCallback(
(e: MouseEvent<HTMLDivElement>) => {
if (!imageDTO) {
return;
}
if (e.shiftKey) {
const rangeEndImageName = imageDTO.image_name;
const lastSelectedImage = selection[selection.length - 1]?.image_name;
const lastClickedIndex = imageDTOs.findIndex(
(n) => n.image_name === lastSelectedImage
);
const currentClickedIndex = imageDTOs.findIndex(
(n) => n.image_name === rangeEndImageName
);
if (lastClickedIndex > -1 && currentClickedIndex > -1) {
// We have a valid range!
const start = Math.min(lastClickedIndex, currentClickedIndex);
const end = Math.max(lastClickedIndex, currentClickedIndex);
const imagesToSelect = imageDTOs.slice(start, end + 1);
dispatch(selectionChanged(uniq(selection.concat(imagesToSelect))));
}
} else if (e.ctrlKey || e.metaKey) {
if (
selection.some((i) => i.image_name === imageDTO.image_name) &&
selection.length > 1
) {
dispatch(
selectionChanged(
selection.filter((n) => n.image_name !== imageDTO.image_name)
)
);
} else {
dispatch(selectionChanged(uniq(selection.concat(imageDTO))));
}
} else {
dispatch(selectionChanged([imageDTO]));
}
},
[dispatch, imageDTO, imageDTOs, selection]
);
const isSelected = useMemo(
() =>
imageDTO
? selection.some((i) => i.image_name === imageDTO.image_name)
: false,
[imageDTO, selection]
);
const selectionCount = useMemo(() => selection.length, [selection.length]);
return {
selection,
selectionCount,
isSelected,
handleClick,
};
};

View File

@ -4,14 +4,15 @@ import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { imageSelected } from 'features/gallery/store/gallerySlice'; import { imageSelected } from 'features/gallery/store/gallerySlice';
import { clamp, isEqual } from 'lodash-es'; import { clamp, isEqual } from 'lodash-es';
import { useCallback } from 'react'; import { useCallback } from 'react';
import { boardsApi } from 'services/api/endpoints/boards';
import { import {
ListImagesArgs,
imagesAdapter,
imagesApi, imagesApi,
useLazyListImagesQuery, useLazyListImagesQuery,
} from 'services/api/endpoints/images'; } from 'services/api/endpoints/images';
import { selectListImagesBaseQueryArgs } from '../store/gallerySelectors'; import { selectListImagesBaseQueryArgs } from '../store/gallerySelectors';
import { IMAGE_LIMIT } from '../store/types'; import { IMAGE_LIMIT } from '../store/types';
import { ListImagesArgs } from 'services/api/types';
import { imagesAdapter } from 'services/api/util';
export const nextPrevImageButtonsSelector = createSelector( export const nextPrevImageButtonsSelector = createSelector(
[stateSelector, selectListImagesBaseQueryArgs], [stateSelector, selectListImagesBaseQueryArgs],
@ -19,12 +20,21 @@ export const nextPrevImageButtonsSelector = createSelector(
const { data, status } = const { data, status } =
imagesApi.endpoints.listImages.select(baseQueryArgs)(state); imagesApi.endpoints.listImages.select(baseQueryArgs)(state);
const { data: total } =
state.gallery.galleryView === 'images'
? boardsApi.endpoints.getBoardImagesTotal.select(
baseQueryArgs.board_id ?? 'none'
)(state)
: boardsApi.endpoints.getBoardAssetsTotal.select(
baseQueryArgs.board_id ?? 'none'
)(state);
const lastSelectedImage = const lastSelectedImage =
state.gallery.selection[state.gallery.selection.length - 1]; state.gallery.selection[state.gallery.selection.length - 1];
const isFetching = status === 'pending'; const isFetching = status === 'pending';
if (!data || !lastSelectedImage || data.total === 0) { if (!data || !lastSelectedImage || total === 0) {
return { return {
isFetching, isFetching,
queryArgs: baseQueryArgs, queryArgs: baseQueryArgs,
@ -44,30 +54,30 @@ export const nextPrevImageButtonsSelector = createSelector(
const images = selectors.selectAll(data); const images = selectors.selectAll(data);
const currentImageIndex = images.findIndex( const currentImageIndex = images.findIndex(
(i) => i.image_name === lastSelectedImage (i) => i.image_name === lastSelectedImage.image_name
); );
const nextImageIndex = clamp(currentImageIndex + 1, 0, images.length - 1); const nextImageIndex = clamp(currentImageIndex + 1, 0, images.length - 1);
const prevImageIndex = clamp(currentImageIndex - 1, 0, images.length - 1); const prevImageIndex = clamp(currentImageIndex - 1, 0, images.length - 1);
const nextImageId = images[nextImageIndex]?.image_name; const nextImageId = images[nextImageIndex]?.image_name;
const prevImageId = images[prevImageIndex]?.image_name; const prevImageId = images[prevImageIndex]?.image_name;
const nextImage = selectors.selectById(data, nextImageId); const nextImage = nextImageId
const prevImage = selectors.selectById(data, prevImageId); ? selectors.selectById(data, nextImageId)
: undefined;
const prevImage = prevImageId
? selectors.selectById(data, prevImageId)
: undefined;
const imagesLength = images.length; const imagesLength = images.length;
return { return {
isOnFirstImage: currentImageIndex === 0, loadedImagesCount: images.length,
isOnLastImage: currentImageIndex,
!isNaN(currentImageIndex) && currentImageIndex === imagesLength - 1, areMoreImagesAvailable: (total ?? 0) > imagesLength,
areMoreImagesAvailable: (data?.total ?? 0) > imagesLength,
isFetching: status === 'pending', isFetching: status === 'pending',
nextImage, nextImage,
prevImage, prevImage,
nextImageId,
prevImageId,
queryArgs, queryArgs,
}; };
}, },
@ -82,22 +92,22 @@ export const useNextPrevImage = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { const {
isOnFirstImage, nextImage,
isOnLastImage, prevImage,
nextImageId,
prevImageId,
areMoreImagesAvailable, areMoreImagesAvailable,
isFetching, isFetching,
queryArgs, queryArgs,
loadedImagesCount,
currentImageIndex,
} = useAppSelector(nextPrevImageButtonsSelector); } = useAppSelector(nextPrevImageButtonsSelector);
const handlePrevImage = useCallback(() => { const handlePrevImage = useCallback(() => {
prevImageId && dispatch(imageSelected(prevImageId)); prevImage && dispatch(imageSelected(prevImage));
}, [dispatch, prevImageId]); }, [dispatch, prevImage]);
const handleNextImage = useCallback(() => { const handleNextImage = useCallback(() => {
nextImageId && dispatch(imageSelected(nextImageId)); nextImage && dispatch(imageSelected(nextImage));
}, [dispatch, nextImageId]); }, [dispatch, nextImage]);
const [listImages] = useLazyListImagesQuery(); const [listImages] = useLazyListImagesQuery();
@ -108,10 +118,12 @@ export const useNextPrevImage = () => {
return { return {
handlePrevImage, handlePrevImage,
handleNextImage, handleNextImage,
isOnFirstImage, isOnFirstImage: currentImageIndex === 0,
isOnLastImage, isOnLastImage:
nextImageId, currentImageIndex !== undefined &&
prevImageId, currentImageIndex === loadedImagesCount - 1,
nextImage,
prevImage,
areMoreImagesAvailable, areMoreImagesAvailable,
handleLoadMoreImages, handleLoadMoreImages,
isFetching, isFetching,

View File

@ -1,5 +1,5 @@
import { createAction } from '@reduxjs/toolkit'; import { createAction } from '@reduxjs/toolkit';
import { ImageUsage } from 'app/contexts/AddImageToBoardContext'; import { ImageUsage } from 'features/deleteImageModal/store/types';
import { BoardDTO } from 'services/api/types'; import { BoardDTO } from 'services/api/types';
export type RequestedBoardImagesDeletionArg = { export type RequestedBoardImagesDeletionArg = {

View File

@ -1,29 +0,0 @@
import { PayloadAction, createSlice } from '@reduxjs/toolkit';
type BoardsState = {
searchText: string;
updateBoardModalOpen: boolean;
};
export const initialBoardsState: BoardsState = {
updateBoardModalOpen: false,
searchText: '',
};
const boardsSlice = createSlice({
name: 'boards',
initialState: initialBoardsState,
reducers: {
setBoardSearchText: (state, action: PayloadAction<string>) => {
state.searchText = action.payload;
},
setUpdateBoardModalOpen: (state, action: PayloadAction<boolean>) => {
state.updateBoardModalOpen = action.payload;
},
},
});
export const { setBoardSearchText, setUpdateBoardModalOpen } =
boardsSlice.actions;
export default boardsSlice.reducer;

View File

@ -1,7 +1,7 @@
import { createSelector } from '@reduxjs/toolkit'; import { createSelector } from '@reduxjs/toolkit';
import { RootState } from 'app/store/store'; import { RootState } from 'app/store/store';
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
import { ListImagesArgs } from 'services/api/endpoints/images'; import { ListImagesArgs } from 'services/api/types';
import { import {
ASSETS_CATEGORIES, ASSETS_CATEGORIES,
IMAGE_CATEGORIES, IMAGE_CATEGORIES,
@ -24,7 +24,7 @@ export const selectListImagesBaseQueryArgs = createSelector(
galleryView === 'images' ? IMAGE_CATEGORIES : ASSETS_CATEGORIES; galleryView === 'images' ? IMAGE_CATEGORIES : ASSETS_CATEGORIES;
const listImagesBaseQueryArgs: ListImagesArgs = { const listImagesBaseQueryArgs: ListImagesArgs = {
board_id: selectedBoardId ?? 'none', board_id: selectedBoardId,
categories, categories,
offset: 0, offset: 0,
limit: INITIAL_IMAGE_LIMIT, limit: INITIAL_IMAGE_LIMIT,

View File

@ -1,66 +1,32 @@
import type { PayloadAction } from '@reduxjs/toolkit'; import type { PayloadAction } from '@reduxjs/toolkit';
import { createSlice, isAnyOf } from '@reduxjs/toolkit'; import { createSlice, isAnyOf } from '@reduxjs/toolkit';
import { uniq } from 'lodash-es';
import { boardsApi } from 'services/api/endpoints/boards'; import { boardsApi } from 'services/api/endpoints/boards';
import { imagesApi } from 'services/api/endpoints/images';
import { ImageDTO } from 'services/api/types';
import { BoardId, GalleryState, GalleryView } from './types'; import { BoardId, GalleryState, GalleryView } from './types';
export const initialGalleryState: GalleryState = { export const initialGalleryState: GalleryState = {
selection: [], selection: [],
shouldAutoSwitch: true, shouldAutoSwitch: true,
autoAddBoardId: undefined,
autoAssignBoardOnClick: true, autoAssignBoardOnClick: true,
autoAddBoardId: 'none',
galleryImageMinimumWidth: 96, galleryImageMinimumWidth: 96,
selectedBoardId: undefined, selectedBoardId: 'none',
galleryView: 'images', galleryView: 'images',
batchImageNames: [], shouldShowDeleteButton: false,
isBatchEnabled: false, boardSearchText: '',
}; };
export const gallerySlice = createSlice({ export const gallerySlice = createSlice({
name: 'gallery', name: 'gallery',
initialState: initialGalleryState, initialState: initialGalleryState,
reducers: { reducers: {
imageRangeEndSelected: () => { imageSelected: (state, action: PayloadAction<ImageDTO | null>) => {
// TODO
},
// imageRangeEndSelected: (state, action: PayloadAction<string>) => {
// const rangeEndImageName = action.payload;
// const lastSelectedImage = state.selection[state.selection.length - 1];
// const filteredImages = selectFilteredImagesLocal(state);
// const lastClickedIndex = filteredImages.findIndex(
// (n) => n.image_name === lastSelectedImage
// );
// const currentClickedIndex = filteredImages.findIndex(
// (n) => n.image_name === rangeEndImageName
// );
// if (lastClickedIndex > -1 && currentClickedIndex > -1) {
// // We have a valid range!
// const start = Math.min(lastClickedIndex, currentClickedIndex);
// const end = Math.max(lastClickedIndex, currentClickedIndex);
// const imagesToSelect = filteredImages
// .slice(start, end + 1)
// .map((i) => i.image_name);
// state.selection = uniq(state.selection.concat(imagesToSelect));
// }
// },
imageSelectionToggled: () => {
// TODO
},
// imageSelectionToggled: (state, action: PayloadAction<string>) => {
// TODO: multiselect
// if (
// state.selection.includes(action.payload) &&
// state.selection.length > 1
// ) {
// state.selection = state.selection.filter(
// (imageName) => imageName !== action.payload
// );
// } else {
// state.selection = uniq(state.selection.concat(action.payload));
// }
imageSelected: (state, action: PayloadAction<string | null>) => {
state.selection = action.payload ? [action.payload] : []; state.selection = action.payload ? [action.payload] : [];
}, },
selectionChanged: (state, action: PayloadAction<ImageDTO[]>) => {
state.selection = action.payload;
},
shouldAutoSwitchChanged: (state, action: PayloadAction<boolean>) => { shouldAutoSwitchChanged: (state, action: PayloadAction<boolean>) => {
state.shouldAutoSwitch = action.payload; state.shouldAutoSwitch = action.payload;
}, },
@ -74,53 +40,28 @@ export const gallerySlice = createSlice({
state.selectedBoardId = action.payload; state.selectedBoardId = action.payload;
state.galleryView = 'images'; state.galleryView = 'images';
}, },
isBatchEnabledChanged: (state, action: PayloadAction<boolean>) => { autoAddBoardIdChanged: (state, action: PayloadAction<BoardId>) => {
state.isBatchEnabled = action.payload;
},
imagesAddedToBatch: (state, action: PayloadAction<string[]>) => {
state.batchImageNames = uniq(
state.batchImageNames.concat(action.payload)
);
},
imagesRemovedFromBatch: (state, action: PayloadAction<string[]>) => {
state.batchImageNames = state.batchImageNames.filter(
(imageName) => !action.payload.includes(imageName)
);
const newSelection = state.selection.filter(
(imageName) => !action.payload.includes(imageName)
);
if (newSelection.length) {
state.selection = newSelection;
return;
}
state.selection = [state.batchImageNames[0]] ?? [];
},
batchReset: (state) => {
state.batchImageNames = [];
state.selection = [];
},
autoAddBoardIdChanged: (
state,
action: PayloadAction<string | undefined>
) => {
state.autoAddBoardId = action.payload; state.autoAddBoardId = action.payload;
}, },
galleryViewChanged: (state, action: PayloadAction<GalleryView>) => { galleryViewChanged: (state, action: PayloadAction<GalleryView>) => {
state.galleryView = action.payload; state.galleryView = action.payload;
}, },
shouldShowDeleteButtonChanged: (state, action: PayloadAction<boolean>) => {
state.shouldShowDeleteButton = action.payload;
},
boardSearchTextChanged: (state, action: PayloadAction<string>) => {
state.boardSearchText = action.payload;
},
}, },
extraReducers: (builder) => { extraReducers: (builder) => {
builder.addMatcher(isAnyBoardDeleted, (state, action) => { builder.addMatcher(isAnyBoardDeleted, (state, action) => {
const deletedBoardId = action.meta.arg.originalArgs; const deletedBoardId = action.meta.arg.originalArgs;
if (deletedBoardId === state.selectedBoardId) { if (deletedBoardId === state.selectedBoardId) {
state.selectedBoardId = undefined; state.selectedBoardId = 'none';
state.galleryView = 'images'; state.galleryView = 'images';
} }
if (deletedBoardId === state.autoAddBoardId) { if (deletedBoardId === state.autoAddBoardId) {
state.autoAddBoardId = undefined; state.autoAddBoardId = 'none';
} }
}); });
builder.addMatcher( builder.addMatcher(
@ -132,7 +73,7 @@ export const gallerySlice = createSlice({
} }
if (!boards.map((b) => b.board_id).includes(state.autoAddBoardId)) { if (!boards.map((b) => b.board_id).includes(state.autoAddBoardId)) {
state.autoAddBoardId = undefined; state.autoAddBoardId = 'none';
} }
} }
); );
@ -140,23 +81,21 @@ export const gallerySlice = createSlice({
}); });
export const { export const {
imageRangeEndSelected,
imageSelectionToggled,
imageSelected, imageSelected,
shouldAutoSwitchChanged, shouldAutoSwitchChanged,
autoAssignBoardOnClickChanged, autoAssignBoardOnClickChanged,
setGalleryImageMinimumWidth, setGalleryImageMinimumWidth,
boardIdSelected, boardIdSelected,
isBatchEnabledChanged,
imagesAddedToBatch,
imagesRemovedFromBatch,
autoAddBoardIdChanged, autoAddBoardIdChanged,
galleryViewChanged, galleryViewChanged,
selectionChanged,
shouldShowDeleteButtonChanged,
boardSearchTextChanged,
} = gallerySlice.actions; } = gallerySlice.actions;
export default gallerySlice.reducer; export default gallerySlice.reducer;
const isAnyBoardDeleted = isAnyOf( const isAnyBoardDeleted = isAnyOf(
boardsApi.endpoints.deleteBoard.matchFulfilled, imagesApi.endpoints.deleteBoard.matchFulfilled,
boardsApi.endpoints.deleteBoardAndImages.matchFulfilled imagesApi.endpoints.deleteBoardAndImages.matchFulfilled
); );

View File

@ -1,4 +1,4 @@
import { ImageCategory } from 'services/api/types'; import { ImageCategory, ImageDTO } from 'services/api/types';
export const IMAGE_CATEGORIES: ImageCategory[] = ['general']; export const IMAGE_CATEGORIES: ImageCategory[] = ['general'];
export const ASSETS_CATEGORIES: ImageCategory[] = [ export const ASSETS_CATEGORIES: ImageCategory[] = [
@ -11,17 +11,16 @@ export const INITIAL_IMAGE_LIMIT = 100;
export const IMAGE_LIMIT = 20; export const IMAGE_LIMIT = 20;
export type GalleryView = 'images' | 'assets'; export type GalleryView = 'images' | 'assets';
// export type BoardId = 'no_board' | (string & Record<never, never>); export type BoardId = 'none' | (string & Record<never, never>);
export type BoardId = string | undefined;
export type GalleryState = { export type GalleryState = {
selection: string[]; selection: ImageDTO[];
shouldAutoSwitch: boolean; shouldAutoSwitch: boolean;
autoAddBoardId: string | undefined;
autoAssignBoardOnClick: boolean; autoAssignBoardOnClick: boolean;
autoAddBoardId: BoardId;
galleryImageMinimumWidth: number; galleryImageMinimumWidth: number;
selectedBoardId: BoardId; selectedBoardId: BoardId;
galleryView: GalleryView; galleryView: GalleryView;
batchImageNames: string[]; shouldShowDeleteButton: boolean;
isBatchEnabled: boolean; boardSearchText: string;
}; };

View File

@ -1,37 +0,0 @@
import { PayloadAction, createSlice } from '@reduxjs/toolkit';
import { ImageDTO } from 'services/api/types';
type DeleteImageState = {
imageToDelete: ImageDTO | null;
isModalOpen: boolean;
};
export const initialDeleteImageState: DeleteImageState = {
imageToDelete: null,
isModalOpen: false,
};
const imageDeletion = createSlice({
name: 'imageDeletion',
initialState: initialDeleteImageState,
reducers: {
isModalOpenChanged: (state, action: PayloadAction<boolean>) => {
state.isModalOpen = action.payload;
},
imageToDeleteSelected: (state, action: PayloadAction<ImageDTO>) => {
state.imageToDelete = action.payload;
},
imageToDeleteCleared: (state) => {
state.imageToDelete = null;
state.isModalOpen = false;
},
},
});
export const {
isModalOpenChanged,
imageToDeleteSelected,
imageToDeleteCleared,
} = imageDeletion.actions;
export default imageDeletion.reducer;

View File

@ -1,6 +0,0 @@
export type ImageUsage = {
isInitialImage: boolean;
isCanvasImage: boolean;
isNodesImage: boolean;
isControlNetImage: boolean;
};

View File

@ -39,11 +39,19 @@ export const loraSlice = createSlice({
action: PayloadAction<{ id: string; weight: number }> action: PayloadAction<{ id: string; weight: number }>
) => { ) => {
const { id, weight } = action.payload; const { id, weight } = action.payload;
state.loras[id].weight = weight; const lora = state.loras[id];
if (!lora) {
return;
}
lora.weight = weight;
}, },
loraWeightReset: (state, action: PayloadAction<string>) => { loraWeightReset: (state, action: PayloadAction<string>) => {
const id = action.payload; const id = action.payload;
state.loras[id].weight = defaultLoRAConfig.weight; const lora = state.loras[id];
if (!lora) {
return;
}
lora.weight = defaultLoRAConfig.weight;
}, },
}, },
}); });

View File

@ -170,15 +170,17 @@ const NodeSearch = () => {
// } // }
if (key === 'Enter') { if (key === 'Enter') {
let selectedNodeType: AnyInvocationType; let selectedNodeType: AnyInvocationType | undefined;
if (searchText.length > 0) { if (searchText.length > 0) {
selectedNodeType = filteredNodes[focusedIndex].item.type; selectedNodeType = filteredNodes[focusedIndex]?.item.type;
} else { } else {
selectedNodeType = nodes[focusedIndex].type; selectedNodeType = nodes[focusedIndex]?.type;
} }
if (selectedNodeType) {
addNode(selectedNodeType); addNode(selectedNodeType);
}
setShowNodeList(false); setShowNodeList(false);
} }

View File

@ -79,9 +79,12 @@ const nodesSlice = createSlice({
) => { ) => {
const { nodeId, fieldName, value } = action.payload; const { nodeId, fieldName, value } = action.payload;
const nodeIndex = state.nodes.findIndex((n) => n.id === nodeId); const nodeIndex = state.nodes.findIndex((n) => n.id === nodeId);
const input = state.nodes?.[nodeIndex]?.data?.inputs[fieldName];
if (!input) {
return;
}
if (nodeIndex > -1) { if (nodeIndex > -1) {
state.nodes[nodeIndex].data.inputs[fieldName].value = value; input.value = value;
} }
}, },
imageCollectionFieldValueChanged: ( imageCollectionFieldValueChanged: (
@ -99,16 +102,19 @@ const nodesSlice = createSlice({
return; return;
} }
const currentValue = cloneDeep( const input = state.nodes?.[nodeIndex]?.data?.inputs[fieldName];
state.nodes[nodeIndex].data.inputs[fieldName].value if (!input) {
);
if (!currentValue) {
state.nodes[nodeIndex].data.inputs[fieldName].value = value;
return; return;
} }
state.nodes[nodeIndex].data.inputs[fieldName].value = uniqBy( const currentValue = cloneDeep(input.value);
if (!currentValue) {
input.value = value;
return;
}
input.value = uniqBy(
(currentValue as ImageField[]).concat(value), (currentValue as ImageField[]).concat(value),
'image_name' 'image_name'
); );

View File

@ -29,6 +29,8 @@ import {
VaeInputFieldTemplate, VaeInputFieldTemplate,
VaeModelInputFieldTemplate, VaeModelInputFieldTemplate,
} from '../types/types'; } from '../types/types';
import { logger } from 'app/logging/logger';
import { parseify } from 'common/util/serialize';
export type BaseFieldProperties = 'name' | 'title' | 'description'; export type BaseFieldProperties = 'name' | 'title' | 'description';
@ -50,7 +52,13 @@ export type BuildInputFieldArg = {
*/ */
export const refObjectToFieldType = ( export const refObjectToFieldType = (
refObject: OpenAPIV3.ReferenceObject refObject: OpenAPIV3.ReferenceObject
): keyof typeof FIELD_TYPE_MAP => refObject.$ref.split('/').slice(-1)[0]; ): keyof typeof FIELD_TYPE_MAP => {
const name = refObject.$ref.split('/').slice(-1)[0];
if (!name) {
return 'UNKNOWN FIELD TYPE';
}
return name;
};
const buildIntegerInputFieldTemplate = ({ const buildIntegerInputFieldTemplate = ({
schemaObject, schemaObject,
@ -428,7 +436,7 @@ export const getFieldType = (
let rawFieldType = ''; let rawFieldType = '';
if (typeHints && name in typeHints) { if (typeHints && name in typeHints) {
rawFieldType = typeHints[name]; rawFieldType = typeHints[name] ?? 'UNKNOWN FIELD TYPE';
} else if (!schemaObject.type) { } else if (!schemaObject.type) {
// if schemaObject has no type, then it should have one of allOf, anyOf, oneOf // if schemaObject has no type, then it should have one of allOf, anyOf, oneOf
if (schemaObject.allOf) { if (schemaObject.allOf) {
@ -568,10 +576,23 @@ export const buildOutputFieldTemplates = (
// extract output schema name from ref // extract output schema name from ref
const outputSchemaName = refObject.$ref.split('/').slice(-1)[0]; const outputSchemaName = refObject.$ref.split('/').slice(-1)[0];
if (!outputSchemaName) {
logger('nodes').error(
{ refObject: parseify(refObject) },
'No output schema name found in ref object'
);
throw 'No output schema name found in ref object';
}
// get the output schema itself // get the output schema itself
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
const outputSchema = openAPI.components!.schemas![outputSchemaName]; const outputSchema = openAPI.components!.schemas![outputSchemaName];
if (!outputSchema) {
logger('nodes').error({ outputSchemaName }, 'Output schema not found');
throw 'Output schema not found';
}
if (isSchemaObject(outputSchema)) { if (isSchemaObject(outputSchema)) {
const outputFields = reduce( const outputFields = reduce(
outputSchema.properties as OpenAPIV3.SchemaObject, outputSchema.properties as OpenAPIV3.SchemaObject,

View File

@ -16,7 +16,10 @@ import { useFeatureStatus } from 'features/system/hooks/useFeatureStatus';
import { map } from 'lodash-es'; import { map } from 'lodash-es';
import { Fragment, memo, useCallback } from 'react'; import { Fragment, memo, useCallback } from 'react';
import { FaPlus } from 'react-icons/fa'; import { FaPlus } from 'react-icons/fa';
import { useGetControlNetModelsQuery } from 'services/api/endpoints/models'; import {
controlNetModelsAdapter,
useGetControlNetModelsQuery,
} from 'services/api/endpoints/models';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
const selector = createSelector( const selector = createSelector(
@ -42,7 +45,9 @@ const ParamControlNetCollapse = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { firstModel } = useGetControlNetModelsQuery(undefined, { const { firstModel } = useGetControlNetModelsQuery(undefined, {
selectFromResult: (result) => { selectFromResult: (result) => {
const firstModel = result.data?.entities[result.data?.ids[0]]; const firstModel = result.data
? controlNetModelsAdapter.getSelectors().selectAll(result.data)[0]
: undefined;
return { return {
firstModel, firstModel,
}; };
@ -95,7 +100,7 @@ const ParamControlNetCollapse = () => {
{controlNetsArray.map((c, i) => ( {controlNetsArray.map((c, i) => (
<Fragment key={c.controlNetId}> <Fragment key={c.controlNetId}>
{i > 0 && <Divider />} {i > 0 && <Divider />}
<ControlNet controlNetId={c.controlNetId} /> <ControlNet controlNet={c} />
</Fragment> </Fragment>
))} ))}
</Flex> </Flex>

View File

@ -12,7 +12,7 @@ import {
} from 'features/sdxl/store/sdxlSlice'; } from 'features/sdxl/store/sdxlSlice';
import { useCallback } from 'react'; import { useCallback } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { UnsafeImageMetadata } from 'services/api/endpoints/images'; import { UnsafeImageMetadata } from 'services/api/types';
import { ImageDTO } from 'services/api/types'; import { ImageDTO } from 'services/api/types';
import { initialImageSelected, modelSelected } from '../store/actions'; import { initialImageSelected, modelSelected } from '../store/actions';
import { import {

Some files were not shown because too many files have changed in this diff Show More