mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
refactor(ui): canvas v2 (wip)
This commit is contained in:
parent
ca9090d070
commit
d135c48319
@ -19,9 +19,9 @@ export const addDeleteBoardAndImagesFulfilledListener = (startAppListening: AppS
|
|||||||
let wereControlAdaptersReset = false;
|
let wereControlAdaptersReset = false;
|
||||||
let wereControlLayersReset = false;
|
let wereControlLayersReset = false;
|
||||||
|
|
||||||
const { canvas, nodes, controlAdapters, controlLayers } = getState();
|
const { canvas, nodes, controlAdapters, canvasV2 } = getState();
|
||||||
deleted_images.forEach((image_name) => {
|
deleted_images.forEach((image_name) => {
|
||||||
const imageUsage = getImageUsage(canvas, nodes.present, controlAdapters, controlLayers.present, image_name);
|
const imageUsage = getImageUsage(canvas, nodes.present, controlAdapters, canvasV2, image_name);
|
||||||
|
|
||||||
if (imageUsage.isCanvasImage && !wasCanvasReset) {
|
if (imageUsage.isCanvasImage && !wasCanvasReset) {
|
||||||
dispatch(resetCanvas());
|
dispatch(resetCanvas());
|
||||||
|
@ -65,14 +65,14 @@ export const addControlAdapterPreprocessor = (startAppListening: AppStartListeni
|
|||||||
// Delay before starting actual work
|
// Delay before starting actual work
|
||||||
await delay(DEBOUNCE_MS);
|
await delay(DEBOUNCE_MS);
|
||||||
|
|
||||||
const layer = state.controlLayers.present.layers.filter(isControlAdapterLayer).find((l) => l.id === layerId);
|
const layer = state.canvasV2.layers.filter(isControlAdapterLayer).find((l) => l.id === layerId);
|
||||||
|
|
||||||
if (!layer) {
|
if (!layer) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// We should only process if the processor settings or image have changed
|
// We should only process if the processor settings or image have changed
|
||||||
const originalLayer = originalState.controlLayers.present.layers
|
const originalLayer = originalState.canvasV2.layers
|
||||||
.filter(isControlAdapterLayer)
|
.filter(isControlAdapterLayer)
|
||||||
.find((l) => l.id === layerId);
|
.find((l) => l.id === layerId);
|
||||||
const originalImage = originalLayer?.controlAdapter.image;
|
const originalImage = originalLayer?.controlAdapter.image;
|
||||||
@ -161,7 +161,7 @@ export const addControlAdapterPreprocessor = (startAppListening: AppStartListeni
|
|||||||
if (signal.aborted) {
|
if (signal.aborted) {
|
||||||
// The listener was canceled - we need to cancel the pending processor batch, if there is one (could have changed by now).
|
// The listener was canceled - we need to cancel the pending processor batch, if there is one (could have changed by now).
|
||||||
const pendingBatchId = getState()
|
const pendingBatchId = getState()
|
||||||
.controlLayers.present.layers.filter(isControlAdapterLayer)
|
.canvasV2.layers.filter(isControlAdapterLayer)
|
||||||
.find((l) => l.id === layerId)?.controlAdapter.processorPendingBatchId;
|
.find((l) => l.id === layerId)?.controlAdapter.processorPendingBatchId;
|
||||||
if (pendingBatchId) {
|
if (pendingBatchId) {
|
||||||
cancelProcessorBatch(dispatch, layerId, pendingBatchId);
|
cancelProcessorBatch(dispatch, layerId, pendingBatchId);
|
||||||
|
@ -70,7 +70,7 @@ const deleteControlAdapterImages = (state: RootState, dispatch: AppDispatch, ima
|
|||||||
};
|
};
|
||||||
|
|
||||||
const deleteControlLayerImages = (state: RootState, dispatch: AppDispatch, imageDTO: ImageDTO) => {
|
const deleteControlLayerImages = (state: RootState, dispatch: AppDispatch, imageDTO: ImageDTO) => {
|
||||||
state.controlLayers.present.layers.forEach((l) => {
|
state.canvasV2.layers.forEach((l) => {
|
||||||
if (isRegionalGuidanceLayer(l)) {
|
if (isRegionalGuidanceLayer(l)) {
|
||||||
if (l.ipAdapters.some((ipa) => ipa.image?.name === imageDTO.image_name)) {
|
if (l.ipAdapters.some((ipa) => ipa.image?.name === imageDTO.image_name)) {
|
||||||
dispatch(layerDeleted(l.id));
|
dispatch(layerDeleted(l.id));
|
||||||
|
@ -79,15 +79,15 @@ const handleMainModels: ModelHandler = (models, state, dispatch, log) => {
|
|||||||
const optimalDimension = getOptimalDimension(defaultModelInList);
|
const optimalDimension = getOptimalDimension(defaultModelInList);
|
||||||
if (
|
if (
|
||||||
getIsSizeOptimal(
|
getIsSizeOptimal(
|
||||||
state.controlLayers.present.size.width,
|
state.canvasV2.size.width,
|
||||||
state.controlLayers.present.size.height,
|
state.canvasV2.size.height,
|
||||||
optimalDimension
|
optimalDimension
|
||||||
)
|
)
|
||||||
) {
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const { width, height } = calculateNewSize(
|
const { width, height } = calculateNewSize(
|
||||||
state.controlLayers.present.size.aspectRatio.value,
|
state.canvasV2.size.aspectRatio.value,
|
||||||
optimalDimension * optimalDimension
|
optimalDimension * optimalDimension
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -4,7 +4,7 @@ import { logger } from 'app/logging/logger';
|
|||||||
import { idbKeyValDriver } from 'app/store/enhancers/reduxRemember/driver';
|
import { idbKeyValDriver } from 'app/store/enhancers/reduxRemember/driver';
|
||||||
import { errorHandler } from 'app/store/enhancers/reduxRemember/errors';
|
import { errorHandler } from 'app/store/enhancers/reduxRemember/errors';
|
||||||
import type { JSONObject } from 'common/types';
|
import type { JSONObject } from 'common/types';
|
||||||
import { canvasPersistConfig, canvasSlice } from 'features/canvas/store/canvasSlice';
|
import { canvasPersistConfig } from 'features/canvas/store/canvasSlice';
|
||||||
import { changeBoardModalSlice } from 'features/changeBoardModal/store/slice';
|
import { changeBoardModalSlice } from 'features/changeBoardModal/store/slice';
|
||||||
import {
|
import {
|
||||||
controlAdaptersV2PersistConfig,
|
controlAdaptersV2PersistConfig,
|
||||||
@ -52,7 +52,6 @@ import { listenerMiddleware } from './middleware/listenerMiddleware';
|
|||||||
|
|
||||||
const allReducers = {
|
const allReducers = {
|
||||||
[api.reducerPath]: api.reducer,
|
[api.reducerPath]: api.reducer,
|
||||||
[canvasSlice.name]: canvasSlice.reducer,
|
|
||||||
[gallerySlice.name]: gallerySlice.reducer,
|
[gallerySlice.name]: gallerySlice.reducer,
|
||||||
[generationSlice.name]: generationSlice.reducer,
|
[generationSlice.name]: generationSlice.reducer,
|
||||||
[nodesSlice.name]: undoable(nodesSlice.reducer, nodesUndoableConfig),
|
[nodesSlice.name]: undoable(nodesSlice.reducer, nodesUndoableConfig),
|
||||||
|
@ -59,8 +59,8 @@ const createSelector = (templates: Templates) =>
|
|||||||
config
|
config
|
||||||
) => {
|
) => {
|
||||||
const { model } = generation;
|
const { model } = generation;
|
||||||
const { size } = controlLayers.present;
|
const { size } = canvasV2;
|
||||||
const { positivePrompt } = controlLayers.present;
|
const { positivePrompt } = canvasV2;
|
||||||
|
|
||||||
const { isConnected } = system;
|
const { isConnected } = system;
|
||||||
|
|
||||||
@ -126,7 +126,7 @@ const createSelector = (templates: Templates) =>
|
|||||||
|
|
||||||
if (activeTabName === 'generation') {
|
if (activeTabName === 'generation') {
|
||||||
// Handling for generation tab
|
// Handling for generation tab
|
||||||
controlLayers.present.layers
|
canvasV2.layers
|
||||||
.filter((l) => l.isEnabled)
|
.filter((l) => l.isEnabled)
|
||||||
.forEach((l, i) => {
|
.forEach((l, i) => {
|
||||||
const layerLiteral = i18n.t('controlLayers.layers_one');
|
const layerLiteral = i18n.t('controlLayers.layers_one');
|
||||||
|
@ -23,7 +23,7 @@ export const AddPromptButtons = ({ layerId }: AddPromptButtonProps) => {
|
|||||||
const selectValidActions = useMemo(
|
const selectValidActions = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
return {
|
return {
|
||||||
canAddPositivePrompt: layer.positivePrompt === null,
|
canAddPositivePrompt: layer.positivePrompt === null,
|
||||||
|
@ -9,7 +9,7 @@ import { useTranslation } from 'react-i18next';
|
|||||||
|
|
||||||
export const BrushColorPicker = memo(() => {
|
export const BrushColorPicker = memo(() => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const brushColor = useAppSelector((s) => s.controlLayers.present.brushColor);
|
const brushColor = useAppSelector((s) => s.canvasV2.brushColor);
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const onChange = useCallback(
|
const onChange = useCallback(
|
||||||
(color: RgbaColor) => {
|
(color: RgbaColor) => {
|
||||||
|
@ -20,7 +20,7 @@ const formatPx = (v: number | string) => `${v} px`;
|
|||||||
export const BrushSize = memo(() => {
|
export const BrushSize = memo(() => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const brushSize = useAppSelector((s) => s.controlLayers.present.brushSize);
|
const brushSize = useAppSelector((s) => s.canvasV2.brushSize);
|
||||||
const onChange = useCallback(
|
const onChange = useCallback(
|
||||||
(v: number) => {
|
(v: number) => {
|
||||||
dispatch(brushSizeChanged(Math.round(v)));
|
dispatch(brushSizeChanged(Math.round(v)));
|
||||||
|
@ -19,7 +19,7 @@ type Props = {
|
|||||||
export const CALayer = memo(({ layerId }: Props) => {
|
export const CALayer = memo(({ layerId }: Props) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const isSelected = useAppSelector(
|
const isSelected = useAppSelector(
|
||||||
(s) => selectLayerOrThrow(s.controlLayers.present, layerId, isControlAdapterLayer).isSelected
|
(s) => selectLayerOrThrow(s.canvasV2, layerId, isControlAdapterLayer).isSelected
|
||||||
);
|
);
|
||||||
const onClick = useCallback(() => {
|
const onClick = useCallback(() => {
|
||||||
dispatch(layerSelected(layerId));
|
dispatch(layerSelected(layerId));
|
||||||
|
@ -28,7 +28,7 @@ type Props = {
|
|||||||
export const CALayerControlAdapterWrapper = memo(({ layerId }: Props) => {
|
export const CALayerControlAdapterWrapper = memo(({ layerId }: Props) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const controlAdapter = useAppSelector(
|
const controlAdapter = useAppSelector(
|
||||||
(s) => selectLayerOrThrow(s.controlLayers.present, layerId, isControlAdapterLayer).controlAdapter
|
(s) => selectLayerOrThrow(s.canvasV2, layerId, isControlAdapterLayer).controlAdapter
|
||||||
);
|
);
|
||||||
|
|
||||||
const onChangeBeginEndStepPct = useCallback(
|
const onChangeBeginEndStepPct = useCallback(
|
||||||
|
@ -19,7 +19,7 @@ import { memo } from 'react';
|
|||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
|
||||||
const selectLayerIdTypePairs = createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
const selectLayerIdTypePairs = createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const [renderableLayers, ipAdapterLayers] = partition(controlLayers.present.layers, isRenderableLayer);
|
const [renderableLayers, ipAdapterLayers] = partition(canvasV2.layers, isRenderableLayer);
|
||||||
return [...ipAdapterLayers, ...renderableLayers].map((l) => ({ id: l.id, type: l.type })).reverse();
|
return [...ipAdapterLayers, ...renderableLayers].map((l) => ({ id: l.id, type: l.type })).reverse();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@ import { PiTrashSimpleBold } from 'react-icons/pi';
|
|||||||
export const DeleteAllLayersButton = memo(() => {
|
export const DeleteAllLayersButton = memo(() => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const isDisabled = useAppSelector((s) => s.controlLayers.present.layers.length === 0);
|
const isDisabled = useAppSelector((s) => s.canvasV2.layers.length === 0);
|
||||||
const onClick = useCallback(() => {
|
const onClick = useCallback(() => {
|
||||||
dispatch(allLayersDeleted());
|
dispatch(allLayersDeleted());
|
||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
@ -14,7 +14,7 @@ export const GlobalMaskLayerOpacity = memo(() => {
|
|||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const globalMaskLayerOpacity = useAppSelector((s) =>
|
const globalMaskLayerOpacity = useAppSelector((s) =>
|
||||||
Math.round(s.controlLayers.present.globalMaskLayerOpacity * 100)
|
Math.round(s.canvasV2.globalMaskLayerOpacity * 100)
|
||||||
);
|
);
|
||||||
const onChange = useCallback(
|
const onChange = useCallback(
|
||||||
(v: number) => {
|
(v: number) => {
|
||||||
|
@ -7,8 +7,8 @@ import { memo } from 'react';
|
|||||||
|
|
||||||
export const HeadsUpDisplay = memo(() => {
|
export const HeadsUpDisplay = memo(() => {
|
||||||
const stageAttrs = useStore($stageAttrs);
|
const stageAttrs = useStore($stageAttrs);
|
||||||
const layerCount = useAppSelector((s) => s.controlLayers.present.layers.length);
|
const layerCount = useAppSelector((s) => s.canvasV2.layers.length);
|
||||||
const bbox = useAppSelector((s) => s.controlLayers.present.bbox);
|
const bbox = useAppSelector((s) => s.canvasV2.bbox);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex flexDir="column" bg="blackAlpha.400" borderBottomEndRadius="base" p={2} minW={64} gap={2}>
|
<Flex flexDir="column" bg="blackAlpha.400" borderBottomEndRadius="base" p={2} minW={64} gap={2}>
|
||||||
|
@ -25,7 +25,7 @@ type Props = {
|
|||||||
|
|
||||||
export const IILayer = memo(({ layerId }: Props) => {
|
export const IILayer = memo(({ layerId }: Props) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const layer = useAppSelector((s) => selectLayerOrThrow(s.controlLayers.present, layerId, isInitialImageLayer));
|
const layer = useAppSelector((s) => selectLayerOrThrow(s.canvasV2, layerId, isInitialImageLayer));
|
||||||
const onClick = useCallback(() => {
|
const onClick = useCallback(() => {
|
||||||
dispatch(layerSelected(layerId));
|
dispatch(layerSelected(layerId));
|
||||||
}, [dispatch, layerId]);
|
}, [dispatch, layerId]);
|
||||||
|
@ -16,7 +16,7 @@ type Props = {
|
|||||||
export const IPALayer = memo(({ layerId }: Props) => {
|
export const IPALayer = memo(({ layerId }: Props) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const isSelected = useAppSelector(
|
const isSelected = useAppSelector(
|
||||||
(s) => selectLayerOrThrow(s.controlLayers.present, layerId, isIPAdapterLayer).isSelected
|
(s) => selectLayerOrThrow(s.canvasV2, layerId, isIPAdapterLayer).isSelected
|
||||||
);
|
);
|
||||||
const { isOpen, onToggle } = useDisclosure({ defaultIsOpen: true });
|
const { isOpen, onToggle } = useDisclosure({ defaultIsOpen: true });
|
||||||
const onClick = useCallback(() => {
|
const onClick = useCallback(() => {
|
||||||
|
@ -22,7 +22,7 @@ type Props = {
|
|||||||
export const IPALayerIPAdapterWrapper = memo(({ layerId }: Props) => {
|
export const IPALayerIPAdapterWrapper = memo(({ layerId }: Props) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const ipAdapter = useAppSelector(
|
const ipAdapter = useAppSelector(
|
||||||
(s) => selectLayerOrThrow(s.controlLayers.present, layerId, isIPAdapterLayer).ipAdapter
|
(s) => selectLayerOrThrow(s.canvasV2, layerId, isIPAdapterLayer).ipAdapter
|
||||||
);
|
);
|
||||||
|
|
||||||
const onChangeBeginEndStepPct = useCallback(
|
const onChangeBeginEndStepPct = useCallback(
|
||||||
|
@ -22,10 +22,10 @@ export const LayerMenuArrangeActions = memo(({ layerId }: Props) => {
|
|||||||
const selectValidActions = useMemo(
|
const selectValidActions = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRenderableLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRenderableLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
const layerIndex = controlLayers.present.layers.findIndex((l) => l.id === layerId);
|
const layerIndex = canvasV2.layers.findIndex((l) => l.id === layerId);
|
||||||
const layerCount = controlLayers.present.layers.length;
|
const layerCount = canvasV2.layers.length;
|
||||||
return {
|
return {
|
||||||
canMoveForward: layerIndex < layerCount - 1,
|
canMoveForward: layerIndex < layerCount - 1,
|
||||||
canMoveBackward: layerIndex > 0,
|
canMoveBackward: layerIndex > 0,
|
||||||
|
@ -22,7 +22,7 @@ export const LayerMenuRGActions = memo(({ layerId }: Props) => {
|
|||||||
const selectValidActions = useMemo(
|
const selectValidActions = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
return {
|
return {
|
||||||
canAddPositivePrompt: layer.positivePrompt === null,
|
canAddPositivePrompt: layer.positivePrompt === null,
|
||||||
|
@ -37,7 +37,7 @@ export const LayerOpacity = memo(({ layerId }: Props) => {
|
|||||||
const selectOpacity = useMemo(
|
const selectOpacity = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = selectLayerOrThrow(controlLayers.present, layerId, isLayerWithOpacity);
|
const layer = selectLayerOrThrow(canvasV2, layerId, isLayerWithOpacity);
|
||||||
return Math.round(layer.opacity * 100);
|
return Math.round(layer.opacity * 100);
|
||||||
}),
|
}),
|
||||||
[layerId]
|
[layerId]
|
||||||
|
@ -30,14 +30,14 @@ export const RGLayer = memo(({ layerId }: Props) => {
|
|||||||
const selector = useMemo(
|
const selector = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
return {
|
return {
|
||||||
color: rgbColorToString(layer.previewColor),
|
color: rgbColorToString(layer.previewColor),
|
||||||
hasPositivePrompt: layer.positivePrompt !== null,
|
hasPositivePrompt: layer.positivePrompt !== null,
|
||||||
hasNegativePrompt: layer.negativePrompt !== null,
|
hasNegativePrompt: layer.negativePrompt !== null,
|
||||||
hasIPAdapters: layer.ipAdapters.length > 0,
|
hasIPAdapters: layer.ipAdapters.length > 0,
|
||||||
isSelected: layerId === controlLayers.present.selectedLayerId,
|
isSelected: layerId === canvasV2.selectedLayerId,
|
||||||
autoNegative: layer.autoNegative,
|
autoNegative: layer.autoNegative,
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
|
@ -16,7 +16,7 @@ const useAutoNegative = (layerId: string) => {
|
|||||||
const selectAutoNegative = useMemo(
|
const selectAutoNegative = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
return layer.autoNegative;
|
return layer.autoNegative;
|
||||||
}),
|
}),
|
||||||
|
@ -20,7 +20,7 @@ export const RGLayerColorPicker = memo(({ layerId }: Props) => {
|
|||||||
const selectColor = useMemo(
|
const selectColor = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an vector mask layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an vector mask layer`);
|
||||||
return layer.previewColor;
|
return layer.previewColor;
|
||||||
}),
|
}),
|
||||||
|
@ -15,7 +15,7 @@ export const RGLayerIPAdapterList = memo(({ layerId }: Props) => {
|
|||||||
const selectIPAdapterIds = useMemo(
|
const selectIPAdapterIds = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.filter(isRegionalGuidanceLayer).find((l) => l.id === layerId);
|
const layer = canvasV2.layers.filter(isRegionalGuidanceLayer).find((l) => l.id === layerId);
|
||||||
assert(layer, `Layer ${layerId} not found`);
|
assert(layer, `Layer ${layerId} not found`);
|
||||||
return layer.ipAdapters;
|
return layer.ipAdapters;
|
||||||
}),
|
}),
|
||||||
|
@ -28,7 +28,7 @@ export const RGLayerIPAdapterWrapper = memo(({ layerId, ipAdapterId, ipAdapterNu
|
|||||||
const onDeleteIPAdapter = useCallback(() => {
|
const onDeleteIPAdapter = useCallback(() => {
|
||||||
dispatch(regionalGuidanceIPAdapterDeleted({ layerId, ipAdapterId }));
|
dispatch(regionalGuidanceIPAdapterDeleted({ layerId, ipAdapterId }));
|
||||||
}, [dispatch, ipAdapterId, layerId]);
|
}, [dispatch, ipAdapterId, layerId]);
|
||||||
const ipAdapter = useAppSelector((s) => selectRGLayerIPAdapterOrThrow(s.controlLayers.present, layerId, ipAdapterId));
|
const ipAdapter = useAppSelector((s) => selectRGLayerIPAdapterOrThrow(s.canvasV2, layerId, ipAdapterId));
|
||||||
|
|
||||||
const onChangeBeginEndStepPct = useCallback(
|
const onChangeBeginEndStepPct = useCallback(
|
||||||
(beginEndStepPct: [number, number]) => {
|
(beginEndStepPct: [number, number]) => {
|
||||||
|
@ -19,7 +19,7 @@ type Props = {
|
|||||||
export const RasterLayer = memo(({ layerId }: Props) => {
|
export const RasterLayer = memo(({ layerId }: Props) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const isSelected = useAppSelector(
|
const isSelected = useAppSelector(
|
||||||
(s) => selectLayerOrThrow(s.controlLayers.present, layerId, isRasterLayer).isSelected
|
(s) => selectLayerOrThrow(s.canvasV2, layerId, isRasterLayer).isSelected
|
||||||
);
|
);
|
||||||
const onClick = useCallback(() => {
|
const onClick = useCallback(() => {
|
||||||
dispatch(layerSelected(layerId));
|
dispatch(layerSelected(layerId));
|
||||||
|
@ -5,51 +5,59 @@ import { logger } from 'app/logging/logger';
|
|||||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
import { rgbaColorToString } from 'features/canvas/util/colorToString';
|
import { rgbaColorToString } from 'features/canvas/util/colorToString';
|
||||||
import { HeadsUpDisplay } from 'features/controlLayers/components/HeadsUpDisplay';
|
import { HeadsUpDisplay } from 'features/controlLayers/components/HeadsUpDisplay';
|
||||||
import {
|
import { TRANSPARENCY_CHECKER_PATTERN } from 'features/controlLayers/konva/constants';
|
||||||
BRUSH_SPACING_PCT,
|
|
||||||
MAX_BRUSH_SPACING_PX,
|
|
||||||
MIN_BRUSH_SPACING_PX,
|
|
||||||
TRANSPARENCY_CHECKER_PATTERN,
|
|
||||||
} from 'features/controlLayers/konva/constants';
|
|
||||||
import { setStageEventHandlers } from 'features/controlLayers/konva/events';
|
import { setStageEventHandlers } from 'features/controlLayers/konva/events';
|
||||||
import { debouncedRenderers, renderers as normalRenderers } from 'features/controlLayers/konva/renderers/layers';
|
import { debouncedRenderers, renderers as normalRenderers } from 'features/controlLayers/konva/renderers/layers';
|
||||||
|
import { caBboxChanged, caTranslated } from 'features/controlLayers/store/controlAdaptersSlice';
|
||||||
import {
|
import {
|
||||||
$bbox,
|
$bbox,
|
||||||
$brushSpacingPx,
|
$currentFill,
|
||||||
$brushWidth,
|
|
||||||
$fill,
|
|
||||||
$invertScroll,
|
|
||||||
$isDrawing,
|
$isDrawing,
|
||||||
$isMouseDown,
|
$isMouseDown,
|
||||||
$lastAddedPoint,
|
$lastAddedPoint,
|
||||||
$lastCursorPos,
|
$lastCursorPos,
|
||||||
$lastMouseDownPos,
|
$lastMouseDownPos,
|
||||||
$selectedLayer,
|
$selectedEntity,
|
||||||
$spaceKey,
|
$spaceKey,
|
||||||
$stageAttrs,
|
$stageAttrs,
|
||||||
$tool,
|
$toolState,
|
||||||
$toolBuffer,
|
|
||||||
bboxChanged,
|
bboxChanged,
|
||||||
brushLineAdded,
|
brushWidthChanged,
|
||||||
brushSizeChanged,
|
eraserWidthChanged,
|
||||||
eraserLineAdded,
|
|
||||||
layerBboxChanged,
|
|
||||||
layerTranslated,
|
|
||||||
linePointsAdded,
|
|
||||||
rectAdded,
|
|
||||||
selectCanvasV2Slice,
|
selectCanvasV2Slice,
|
||||||
|
toolBufferChanged,
|
||||||
|
toolChanged,
|
||||||
} from 'features/controlLayers/store/controlLayersSlice';
|
} from 'features/controlLayers/store/controlLayersSlice';
|
||||||
import { selectLayersSlice } from 'features/controlLayers/store/layersSlice';
|
import {
|
||||||
import { selectRegionalGuidanceSlice } from 'features/controlLayers/store/regionalGuidanceSlice';
|
layerBboxChanged,
|
||||||
|
layerBrushLineAdded,
|
||||||
|
layerEraserLineAdded,
|
||||||
|
layerLinePointAdded,
|
||||||
|
layerRectAdded,
|
||||||
|
layerTranslated,
|
||||||
|
selectLayersSlice,
|
||||||
|
} from 'features/controlLayers/store/layersSlice';
|
||||||
|
import {
|
||||||
|
rgBboxChanged,
|
||||||
|
rgBrushLineAdded,
|
||||||
|
rgEraserLineAdded,
|
||||||
|
rgLinePointAdded,
|
||||||
|
rgRectAdded,
|
||||||
|
rgTranslated,
|
||||||
|
selectRegionalGuidanceSlice,
|
||||||
|
} from 'features/controlLayers/store/regionalGuidanceSlice';
|
||||||
import type {
|
import type {
|
||||||
AddBrushLineArg,
|
BboxChangedArg,
|
||||||
AddEraserLineArg,
|
BrushLineAddedArg,
|
||||||
AddPointToLineArg,
|
CanvasEntity,
|
||||||
AddRectShapeArg,
|
EraserLineAddedArg,
|
||||||
|
PointAddedToLineArg,
|
||||||
|
PosChangedArg,
|
||||||
|
RectShapeAddedArg,
|
||||||
|
Tool,
|
||||||
} from 'features/controlLayers/store/types';
|
} from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
import type { IRect } from 'konva/lib/types';
|
import type { IRect } from 'konva/lib/types';
|
||||||
import { clamp } from 'lodash-es';
|
|
||||||
import { memo, useCallback, useEffect, useLayoutEffect, useMemo, useState } from 'react';
|
import { memo, useCallback, useEffect, useLayoutEffect, useMemo, useState } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { getImageDTO } from 'services/api/endpoints/images';
|
import { getImageDTO } from 'services/api/endpoints/images';
|
||||||
@ -61,12 +69,12 @@ Konva.showWarnings = false;
|
|||||||
|
|
||||||
const log = logger('controlLayers');
|
const log = logger('controlLayers');
|
||||||
|
|
||||||
const selectBrushColor = createSelector(
|
const selectBrushFill = createSelector(
|
||||||
selectCanvasV2Slice,
|
selectCanvasV2Slice,
|
||||||
selectLayersSlice,
|
selectLayersSlice,
|
||||||
selectRegionalGuidanceSlice,
|
selectRegionalGuidanceSlice,
|
||||||
(canvas, layers, regionalGuidance) => {
|
(canvas, layers, regionalGuidance) => {
|
||||||
const rg = regionalGuidance.regions.find((i) => i.id === canvas.lastSelectedItem?.id);
|
const rg = regionalGuidance.regions.find((i) => i.id === canvas.selectedEntityIdentifier?.id);
|
||||||
|
|
||||||
if (rg) {
|
if (rg) {
|
||||||
return rgbaColorToString({ ...rg.fill, a: regionalGuidance.opacity });
|
return rgbaColorToString({ ...rg.fill, a: regionalGuidance.opacity });
|
||||||
@ -76,89 +84,121 @@ const selectBrushColor = createSelector(
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const selectSelectedLayer = createSelector(selectCanvasV2Slice, (controlLayers) => {
|
|
||||||
return controlLayers.present.layers.find((l) => l.id === controlLayers.present.selectedLayerId) ?? null;
|
|
||||||
});
|
|
||||||
|
|
||||||
const selectLayerCount = createSelector(selectCanvasV2Slice, (controlLayers) => controlLayers.present.layers.length);
|
|
||||||
|
|
||||||
const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null, asPreview: boolean) => {
|
const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null, asPreview: boolean) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const state = useAppSelector((s) => s.controlLayers.present);
|
const canvasV2State = useAppSelector(selectCanvasV2Slice);
|
||||||
const tool = useStore($tool);
|
const layersState = useAppSelector((s) => s.layers);
|
||||||
|
const controlAdaptersState = useAppSelector((s) => s.controlAdaptersV2);
|
||||||
|
const ipAdaptersState = useAppSelector((s) => s.ipAdapters);
|
||||||
|
const regionalGuidanceState = useAppSelector((s) => s.regionalGuidance);
|
||||||
const lastCursorPos = useStore($lastCursorPos);
|
const lastCursorPos = useStore($lastCursorPos);
|
||||||
const lastMouseDownPos = useStore($lastMouseDownPos);
|
const lastMouseDownPos = useStore($lastMouseDownPos);
|
||||||
const isMouseDown = useStore($isMouseDown);
|
const isMouseDown = useStore($isMouseDown);
|
||||||
const isDrawing = useStore($isDrawing);
|
const isDrawing = useStore($isDrawing);
|
||||||
const brushColor = useAppSelector(selectBrushColor);
|
const brushColor = useAppSelector(selectBrushFill);
|
||||||
const selectedLayer = useAppSelector(selectSelectedLayer);
|
const selectedEntity = useMemo(() => {
|
||||||
const renderers = useMemo(() => (asPreview ? debouncedRenderers : normalRenderers), [asPreview]);
|
const identifier = canvasV2State.selectedEntityIdentifier;
|
||||||
const dpr = useDevicePixelRatio({ round: false });
|
if (!identifier) {
|
||||||
const shouldInvertBrushSizeScrollDirection = useAppSelector((s) => s.canvas.shouldInvertBrushSizeScrollDirection);
|
return null;
|
||||||
const brushSpacingPx = useMemo(
|
} else if (identifier.type === 'layer') {
|
||||||
() => clamp(state.brushSize / BRUSH_SPACING_PCT, MIN_BRUSH_SPACING_PX, MAX_BRUSH_SPACING_PX),
|
return layersState.layers.find((i) => i.id === identifier.id) ?? null;
|
||||||
[state.brushSize]
|
} else if (identifier.type === 'control_adapter') {
|
||||||
);
|
return controlAdaptersState.controlAdapters.find((i) => i.id === identifier.id) ?? null;
|
||||||
|
} else if (identifier.type === 'ip_adapter') {
|
||||||
useLayoutEffect(() => {
|
return ipAdaptersState.ipAdapters.find((i) => i.id === identifier.id) ?? null;
|
||||||
$fill.set(brushColor);
|
} else if (identifier.type === 'regional_guidance') {
|
||||||
$brushWidth.set(state.brushSize);
|
return regionalGuidanceState.regions.find((i) => i.id === identifier.id) ?? null;
|
||||||
$brushSpacingPx.set(brushSpacingPx);
|
} else {
|
||||||
$selectedLayer.set(selectedLayer);
|
return null;
|
||||||
$invertScroll.set(shouldInvertBrushSizeScrollDirection);
|
}
|
||||||
$bbox.set(state.bbox);
|
|
||||||
}, [
|
}, [
|
||||||
brushSpacingPx,
|
canvasV2State.selectedEntityIdentifier,
|
||||||
brushColor,
|
controlAdaptersState.controlAdapters,
|
||||||
selectedLayer,
|
ipAdaptersState.ipAdapters,
|
||||||
shouldInvertBrushSizeScrollDirection,
|
layersState.layers,
|
||||||
state.brushSize,
|
regionalGuidanceState.regions,
|
||||||
state.selectedLayerId,
|
|
||||||
state.brushColor,
|
|
||||||
state.bbox,
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const onLayerPosChanged = useCallback(
|
const currentFill = useMemo(() => {
|
||||||
(layerId: string, x: number, y: number) => {
|
if (selectedEntity && selectedEntity.type === 'regional_guidance') {
|
||||||
dispatch(layerTranslated({ layerId, x, y }));
|
return { ...selectedEntity.fill, a: regionalGuidanceState.opacity };
|
||||||
|
}
|
||||||
|
return canvasV2State.tool.fill;
|
||||||
|
}, [canvasV2State.tool.fill, regionalGuidanceState.opacity, selectedEntity]);
|
||||||
|
|
||||||
|
const renderers = useMemo(() => (asPreview ? debouncedRenderers : normalRenderers), [asPreview]);
|
||||||
|
const dpr = useDevicePixelRatio({ round: false });
|
||||||
|
|
||||||
|
useLayoutEffect(() => {
|
||||||
|
$toolState.set(canvasV2State.tool);
|
||||||
|
$selectedEntity.set(selectedEntity);
|
||||||
|
$bbox.set(canvasV2State.bbox);
|
||||||
|
$currentFill.set(currentFill);
|
||||||
|
}, [selectedEntity, canvasV2State.tool, canvasV2State.bbox, currentFill]);
|
||||||
|
|
||||||
|
const onPosChanged = useCallback(
|
||||||
|
(arg: PosChangedArg, entityType: CanvasEntity['type']) => {
|
||||||
|
if (entityType === 'layer') {
|
||||||
|
dispatch(layerTranslated(arg));
|
||||||
|
} else if (entityType === 'control_adapter') {
|
||||||
|
dispatch(caTranslated(arg));
|
||||||
|
} else if (entityType === 'regional_guidance') {
|
||||||
|
dispatch(rgTranslated(arg));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
|
|
||||||
const onBboxChanged = useCallback(
|
const onBboxChanged = useCallback(
|
||||||
(layerId: string, bbox: IRect | null) => {
|
(arg: BboxChangedArg, entityType: CanvasEntity['type']) => {
|
||||||
dispatch(layerBboxChanged({ layerId, bbox }));
|
if (entityType === 'layer') {
|
||||||
|
dispatch(layerBboxChanged(arg));
|
||||||
|
} else if (entityType === 'control_adapter') {
|
||||||
|
dispatch(caBboxChanged(arg));
|
||||||
|
} else if (entityType === 'regional_guidance') {
|
||||||
|
dispatch(rgBboxChanged(arg));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
|
|
||||||
const onBrushLineAdded = useCallback(
|
const onBrushLineAdded = useCallback(
|
||||||
(arg: AddBrushLineArg) => {
|
(arg: BrushLineAddedArg, entityType: CanvasEntity['type']) => {
|
||||||
dispatch(brushLineAdded(arg));
|
if (entityType === 'layer') {
|
||||||
|
dispatch(layerBrushLineAdded(arg));
|
||||||
|
} else if (entityType === 'regional_guidance') {
|
||||||
|
dispatch(rgBrushLineAdded(arg));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
const onEraserLineAdded = useCallback(
|
const onEraserLineAdded = useCallback(
|
||||||
(arg: AddEraserLineArg) => {
|
(arg: EraserLineAddedArg, entityType: CanvasEntity['type']) => {
|
||||||
dispatch(eraserLineAdded(arg));
|
if (entityType === 'layer') {
|
||||||
|
dispatch(layerEraserLineAdded(arg));
|
||||||
|
} else if (entityType === 'regional_guidance') {
|
||||||
|
dispatch(rgEraserLineAdded(arg));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
const onPointAddedToLine = useCallback(
|
const onPointAddedToLine = useCallback(
|
||||||
(arg: AddPointToLineArg) => {
|
(arg: PointAddedToLineArg, entityType: CanvasEntity['type']) => {
|
||||||
dispatch(linePointsAdded(arg));
|
if (entityType === 'layer') {
|
||||||
|
dispatch(layerLinePointAdded(arg));
|
||||||
|
} else if (entityType === 'regional_guidance') {
|
||||||
|
dispatch(rgLinePointAdded(arg));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
const onRectShapeAdded = useCallback(
|
const onRectShapeAdded = useCallback(
|
||||||
(arg: AddRectShapeArg) => {
|
(arg: RectShapeAddedArg, entityType: CanvasEntity['type']) => {
|
||||||
dispatch(rectAdded(arg));
|
if (entityType === 'layer') {
|
||||||
},
|
dispatch(layerRectAdded(arg));
|
||||||
[dispatch]
|
} else if (entityType === 'regional_guidance') {
|
||||||
);
|
dispatch(rgRectAdded(arg));
|
||||||
const onBrushSizeChanged = useCallback(
|
}
|
||||||
(size: number) => {
|
|
||||||
dispatch(brushSizeChanged(size));
|
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
@ -168,6 +208,30 @@ const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null,
|
|||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
|
const onBrushWidthChanged = useCallback(
|
||||||
|
(width: number) => {
|
||||||
|
dispatch(brushWidthChanged(width));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
const onEraserWidthChanged = useCallback(
|
||||||
|
(width: number) => {
|
||||||
|
dispatch(eraserWidthChanged(width));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
const setTool = useCallback(
|
||||||
|
(tool: Tool) => {
|
||||||
|
dispatch(toolChanged(tool));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
const setToolBuffer = useCallback(
|
||||||
|
(toolBuffer: Tool | null) => {
|
||||||
|
dispatch(toolBufferChanged(toolBuffer));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
log.trace('Initializing stage');
|
log.trace('Initializing stage');
|
||||||
@ -189,32 +253,29 @@ const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null,
|
|||||||
|
|
||||||
const cleanup = setStageEventHandlers({
|
const cleanup = setStageEventHandlers({
|
||||||
stage,
|
stage,
|
||||||
getTool: $tool.get,
|
getToolState: $toolState.get,
|
||||||
setTool: $tool.set,
|
setTool,
|
||||||
getToolBuffer: $toolBuffer.get,
|
setToolBuffer,
|
||||||
setToolBuffer: $toolBuffer.set,
|
|
||||||
getIsDrawing: $isDrawing.get,
|
getIsDrawing: $isDrawing.get,
|
||||||
setIsDrawing: $isDrawing.set,
|
setIsDrawing: $isDrawing.set,
|
||||||
getIsMouseDown: $isMouseDown.get,
|
getIsMouseDown: $isMouseDown.get,
|
||||||
setIsMouseDown: $isMouseDown.set,
|
setIsMouseDown: $isMouseDown.set,
|
||||||
getBrushColor: $fill.get,
|
getSelectedEntity: $selectedEntity.get,
|
||||||
getBrushSize: $brushWidth.get,
|
|
||||||
getBrushSpacingPx: $brushSpacingPx.get,
|
|
||||||
getSelectedLayer: $selectedLayer.get,
|
|
||||||
getLastAddedPoint: $lastAddedPoint.get,
|
getLastAddedPoint: $lastAddedPoint.get,
|
||||||
setLastAddedPoint: $lastAddedPoint.set,
|
setLastAddedPoint: $lastAddedPoint.set,
|
||||||
getLastCursorPos: $lastCursorPos.get,
|
getLastCursorPos: $lastCursorPos.get,
|
||||||
setLastCursorPos: $lastCursorPos.set,
|
setLastCursorPos: $lastCursorPos.set,
|
||||||
getLastMouseDownPos: $lastMouseDownPos.get,
|
getLastMouseDownPos: $lastMouseDownPos.get,
|
||||||
setLastMouseDownPos: $lastMouseDownPos.set,
|
setLastMouseDownPos: $lastMouseDownPos.set,
|
||||||
getShouldInvert: $invertScroll.get,
|
|
||||||
getSpaceKey: $spaceKey.get,
|
getSpaceKey: $spaceKey.get,
|
||||||
setStageAttrs: $stageAttrs.set,
|
setStageAttrs: $stageAttrs.set,
|
||||||
onBrushSizeChanged,
|
|
||||||
onBrushLineAdded,
|
onBrushLineAdded,
|
||||||
onEraserLineAdded,
|
onEraserLineAdded,
|
||||||
onPointAddedToLine,
|
onPointAddedToLine,
|
||||||
onRectShapeAdded,
|
onRectShapeAdded,
|
||||||
|
onBrushWidthChanged,
|
||||||
|
onEraserWidthChanged,
|
||||||
|
getCurrentFill: $currentFill.get,
|
||||||
});
|
});
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
@ -224,12 +285,15 @@ const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null,
|
|||||||
}, [
|
}, [
|
||||||
asPreview,
|
asPreview,
|
||||||
onBrushLineAdded,
|
onBrushLineAdded,
|
||||||
onBrushSizeChanged,
|
onBrushWidthChanged,
|
||||||
onEraserLineAdded,
|
onEraserLineAdded,
|
||||||
onPointAddedToLine,
|
onPointAddedToLine,
|
||||||
onRectShapeAdded,
|
onRectShapeAdded,
|
||||||
stage,
|
stage,
|
||||||
container,
|
container,
|
||||||
|
onEraserWidthChanged,
|
||||||
|
setTool,
|
||||||
|
setToolBuffer,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
@ -267,29 +331,26 @@ const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null,
|
|||||||
log.trace('Rendering tool preview');
|
log.trace('Rendering tool preview');
|
||||||
renderers.renderToolPreview(
|
renderers.renderToolPreview(
|
||||||
stage,
|
stage,
|
||||||
tool,
|
canvasV2State.tool,
|
||||||
brushColor,
|
currentFill,
|
||||||
selectedLayer?.type ?? null,
|
selectedEntity,
|
||||||
state.globalMaskLayerOpacity,
|
|
||||||
lastCursorPos,
|
lastCursorPos,
|
||||||
lastMouseDownPos,
|
lastMouseDownPos,
|
||||||
state.brushSize,
|
|
||||||
isDrawing,
|
isDrawing,
|
||||||
isMouseDown
|
isMouseDown
|
||||||
);
|
);
|
||||||
}, [
|
}, [
|
||||||
asPreview,
|
asPreview,
|
||||||
brushColor,
|
brushColor,
|
||||||
|
canvasV2State.tool,
|
||||||
|
currentFill,
|
||||||
isDrawing,
|
isDrawing,
|
||||||
isMouseDown,
|
isMouseDown,
|
||||||
lastCursorPos,
|
lastCursorPos,
|
||||||
lastMouseDownPos,
|
lastMouseDownPos,
|
||||||
renderers,
|
renderers,
|
||||||
selectedLayer?.type,
|
selectedEntity,
|
||||||
stage,
|
stage,
|
||||||
state.brushSize,
|
|
||||||
state.globalMaskLayerOpacity,
|
|
||||||
tool,
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
@ -300,8 +361,8 @@ const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null,
|
|||||||
log.trace('Rendering bbox preview');
|
log.trace('Rendering bbox preview');
|
||||||
renderers.renderBboxPreview(
|
renderers.renderBboxPreview(
|
||||||
stage,
|
stage,
|
||||||
state.bbox,
|
canvasV2State.bbox,
|
||||||
tool,
|
canvasV2State.tool.selected,
|
||||||
$bbox.get,
|
$bbox.get,
|
||||||
onBboxTransformed,
|
onBboxTransformed,
|
||||||
$shift.get,
|
$shift.get,
|
||||||
@ -309,21 +370,41 @@ const useStageRenderer = (stage: Konva.Stage, container: HTMLDivElement | null,
|
|||||||
$meta.get,
|
$meta.get,
|
||||||
$alt.get
|
$alt.get
|
||||||
);
|
);
|
||||||
}, [asPreview, onBboxTransformed, renderers, stage, state.bbox, tool]);
|
}, [asPreview, canvasV2State.bbox, canvasV2State.tool.selected, onBboxTransformed, renderers, stage]);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
log.trace('Rendering layers');
|
log.trace('Rendering layers');
|
||||||
renderers.renderLayers(stage, state.layers, state.globalMaskLayerOpacity, tool, getImageDTO, onLayerPosChanged);
|
renderers.renderLayers(
|
||||||
}, [stage, state.layers, state.globalMaskLayerOpacity, tool, onLayerPosChanged, renderers]);
|
stage,
|
||||||
|
layersState.layers,
|
||||||
|
controlAdaptersState.controlAdapters,
|
||||||
|
regionalGuidanceState.regions,
|
||||||
|
regionalGuidanceState.opacity,
|
||||||
|
canvasV2State.tool.selected,
|
||||||
|
selectedEntity,
|
||||||
|
getImageDTO,
|
||||||
|
onPosChanged
|
||||||
|
);
|
||||||
|
}, [
|
||||||
|
stage,
|
||||||
|
renderers,
|
||||||
|
layersState.layers,
|
||||||
|
controlAdaptersState.controlAdapters,
|
||||||
|
regionalGuidanceState.regions,
|
||||||
|
regionalGuidanceState.opacity,
|
||||||
|
onPosChanged,
|
||||||
|
canvasV2State.tool.selected,
|
||||||
|
selectedEntity,
|
||||||
|
]);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
// useLayoutEffect(() => {
|
||||||
if (asPreview) {
|
// if (asPreview) {
|
||||||
// Preview should not check for transparency
|
// // Preview should not check for transparency
|
||||||
return;
|
// return;
|
||||||
}
|
// }
|
||||||
log.trace('Updating bboxes');
|
// log.trace('Updating bboxes');
|
||||||
debouncedRenderers.updateBboxes(stage, state.layers, onBboxChanged);
|
// debouncedRenderers.updateBboxes(stage, state.layers, onBboxChanged);
|
||||||
}, [stage, asPreview, state.layers, onBboxChanged]);
|
// }, [stage, asPreview, state.layers, onBboxChanged]);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
Konva.pixelRatio = dpr;
|
Konva.pixelRatio = dpr;
|
||||||
@ -395,7 +476,7 @@ StageComponent.displayName = 'StageComponent';
|
|||||||
|
|
||||||
const NoLayersFallback = () => {
|
const NoLayersFallback = () => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const layerCount = useAppSelector(selectLayerCount);
|
const layerCount = useAppSelector((s) => s.layers.layers.length);
|
||||||
if (layerCount) {
|
if (layerCount) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,7 @@ import {
|
|||||||
} from 'react-icons/pi';
|
} from 'react-icons/pi';
|
||||||
|
|
||||||
const selectIsDisabled = createSelector(selectCanvasV2Slice, (controlLayers) => {
|
const selectIsDisabled = createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const selectedLayer = controlLayers.present.layers.find((l) => l.id === controlLayers.present.selectedLayerId);
|
const selectedLayer = canvasV2.layers.find((l) => l.id === canvasV2.selectedLayerId);
|
||||||
return selectedLayer?.type !== 'regional_guidance_layer' && selectedLayer?.type !== 'raster_layer';
|
return selectedLayer?.type !== 'regional_guidance_layer' && selectedLayer?.type !== 'raster_layer';
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -29,7 +29,7 @@ export const ToolChooser: React.FC = () => {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const isDisabled = useAppSelector(selectIsDisabled);
|
const isDisabled = useAppSelector(selectIsDisabled);
|
||||||
const selectedLayerId = useAppSelector((s) => s.controlLayers.present.selectedLayerId);
|
const selectedLayerId = useAppSelector((s) => s.canvasV2.selectedLayerId);
|
||||||
const tool = useStore($tool);
|
const tool = useStore($tool);
|
||||||
|
|
||||||
const setToolToBrush = useCallback(() => {
|
const setToolToBrush = useCallback(() => {
|
||||||
|
@ -102,7 +102,7 @@ export const useAddIPAdapterToIPALayer = (layerId: string) => {
|
|||||||
|
|
||||||
export const useAddIILayer = () => {
|
export const useAddIILayer = () => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const isDisabled = useAppSelector((s) => Boolean(s.controlLayers.present.layers.find(isInitialImageLayer)));
|
const isDisabled = useAppSelector((s) => Boolean(s.canvasV2.layers.find(isInitialImageLayer)));
|
||||||
const addIILayer = useCallback(() => {
|
const addIILayer = useCallback(() => {
|
||||||
dispatch(iiLayerAdded(null));
|
dispatch(iiLayerAdded(null));
|
||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
@ -10,7 +10,7 @@ export const useLayerPositivePrompt = (layerId: string) => {
|
|||||||
const selectLayer = useMemo(
|
const selectLayer = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
assert(layer.positivePrompt !== null, `Layer ${layerId} does not have a positive prompt`);
|
assert(layer.positivePrompt !== null, `Layer ${layerId} does not have a positive prompt`);
|
||||||
return layer.positivePrompt;
|
return layer.positivePrompt;
|
||||||
@ -25,7 +25,7 @@ export const useLayerNegativePrompt = (layerId: string) => {
|
|||||||
const selectLayer = useMemo(
|
const selectLayer = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
assert(isRegionalGuidanceLayer(layer), `Layer ${layerId} not found or not an RP layer`);
|
||||||
assert(layer.negativePrompt !== null, `Layer ${layerId} does not have a negative prompt`);
|
assert(layer.negativePrompt !== null, `Layer ${layerId} does not have a negative prompt`);
|
||||||
return layer.negativePrompt;
|
return layer.negativePrompt;
|
||||||
@ -40,7 +40,7 @@ export const useLayerIsEnabled = (layerId: string) => {
|
|||||||
const selectLayer = useMemo(
|
const selectLayer = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(layer, `Layer ${layerId} not found`);
|
assert(layer, `Layer ${layerId} not found`);
|
||||||
return layer.isEnabled;
|
return layer.isEnabled;
|
||||||
}),
|
}),
|
||||||
@ -54,7 +54,7 @@ export const useLayerType = (layerId: string) => {
|
|||||||
const selectLayer = useMemo(
|
const selectLayer = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
createSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.find((l) => l.id === layerId);
|
const layer = canvasV2.layers.find((l) => l.id === layerId);
|
||||||
assert(layer, `Layer ${layerId} not found`);
|
assert(layer, `Layer ${layerId} not found`);
|
||||||
return layer.type;
|
return layer.type;
|
||||||
}),
|
}),
|
||||||
@ -68,7 +68,7 @@ export const useCALayerOpacity = (layerId: string) => {
|
|||||||
const selectLayer = useMemo(
|
const selectLayer = useMemo(
|
||||||
() =>
|
() =>
|
||||||
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
createMemoizedSelector(selectCanvasV2Slice, (controlLayers) => {
|
||||||
const layer = controlLayers.present.layers.filter(isControlAdapterLayer).find((l) => l.id === layerId);
|
const layer = canvasV2.layers.filter(isControlAdapterLayer).find((l) => l.id === layerId);
|
||||||
assert(layer, `Layer ${layerId} not found`);
|
assert(layer, `Layer ${layerId} not found`);
|
||||||
return { opacity: Math.round(layer.opacity * 100), isFilterEnabled: layer.isFilterEnabled };
|
return { opacity: Math.round(layer.opacity * 100), isFilterEnabled: layer.isFilterEnabled };
|
||||||
}),
|
}),
|
||||||
|
@ -2,27 +2,27 @@ import { calculateNewBrushSize } from 'features/canvas/hooks/useCanvasZoom';
|
|||||||
import { CANVAS_SCALE_BY, MAX_CANVAS_SCALE, MIN_CANVAS_SCALE } from 'features/canvas/util/constants';
|
import { CANVAS_SCALE_BY, MAX_CANVAS_SCALE, MIN_CANVAS_SCALE } from 'features/canvas/util/constants';
|
||||||
import { getScaledFlooredCursorPosition } from 'features/controlLayers/konva/util';
|
import { getScaledFlooredCursorPosition } from 'features/controlLayers/konva/util';
|
||||||
import type {
|
import type {
|
||||||
AddBrushLineArg,
|
BrushLineAddedArg,
|
||||||
AddEraserLineArg,
|
CanvasEntity,
|
||||||
AddPointToLineArg,
|
CanvasV2State,
|
||||||
AddRectShapeArg,
|
EraserLineAddedArg,
|
||||||
LayerData,
|
PointAddedToLineArg,
|
||||||
|
RectShapeAddedArg,
|
||||||
|
RgbaColor,
|
||||||
StageAttrs,
|
StageAttrs,
|
||||||
Tool,
|
Tool,
|
||||||
} from 'features/controlLayers/store/types';
|
} from 'features/controlLayers/store/types';
|
||||||
import { DEFAULT_RGBA_COLOR } from 'features/controlLayers/store/types';
|
|
||||||
import type Konva from 'konva';
|
import type Konva from 'konva';
|
||||||
import type { Vector2d } from 'konva/lib/types';
|
import type { Vector2d } from 'konva/lib/types';
|
||||||
import { clamp } from 'lodash-es';
|
import { clamp } from 'lodash-es';
|
||||||
import type { RgbaColor } from 'react-colorful';
|
|
||||||
|
|
||||||
import { PREVIEW_TOOL_GROUP_ID } from './naming';
|
import { PREVIEW_TOOL_GROUP_ID } from './naming';
|
||||||
|
|
||||||
type Arg = {
|
type Arg = {
|
||||||
stage: Konva.Stage;
|
stage: Konva.Stage;
|
||||||
getTool: () => Tool;
|
getToolState: () => CanvasV2State['tool'];
|
||||||
|
getCurrentFill: () => RgbaColor;
|
||||||
setTool: (tool: Tool) => void;
|
setTool: (tool: Tool) => void;
|
||||||
getToolBuffer: () => Tool | null;
|
|
||||||
setToolBuffer: (tool: Tool | null) => void;
|
setToolBuffer: (tool: Tool | null) => void;
|
||||||
getIsDrawing: () => boolean;
|
getIsDrawing: () => boolean;
|
||||||
setIsDrawing: (isDrawing: boolean) => void;
|
setIsDrawing: (isDrawing: boolean) => void;
|
||||||
@ -35,17 +35,14 @@ type Arg = {
|
|||||||
getLastAddedPoint: () => Vector2d | null;
|
getLastAddedPoint: () => Vector2d | null;
|
||||||
setLastAddedPoint: (pos: Vector2d | null) => void;
|
setLastAddedPoint: (pos: Vector2d | null) => void;
|
||||||
setStageAttrs: (attrs: StageAttrs) => void;
|
setStageAttrs: (attrs: StageAttrs) => void;
|
||||||
getBrushColor: () => RgbaColor;
|
getSelectedEntity: () => CanvasEntity | null;
|
||||||
getBrushSize: () => number;
|
|
||||||
getBrushSpacingPx: () => number;
|
|
||||||
getSelectedLayer: () => LayerData | null;
|
|
||||||
getShouldInvert: () => boolean;
|
|
||||||
getSpaceKey: () => boolean;
|
getSpaceKey: () => boolean;
|
||||||
onBrushLineAdded: (arg: AddBrushLineArg) => void;
|
onBrushLineAdded: (arg: BrushLineAddedArg, entityType: CanvasEntity['type']) => void;
|
||||||
onEraserLineAdded: (arg: AddEraserLineArg) => void;
|
onEraserLineAdded: (arg: EraserLineAddedArg, entityType: CanvasEntity['type']) => void;
|
||||||
onPointAddedToLine: (arg: AddPointToLineArg) => void;
|
onPointAddedToLine: (arg: PointAddedToLineArg, entityType: CanvasEntity['type']) => void;
|
||||||
onRectShapeAdded: (arg: AddRectShapeArg) => void;
|
onRectShapeAdded: (arg: RectShapeAddedArg, entityType: CanvasEntity['type']) => void;
|
||||||
onBrushSizeChanged: (size: number) => void;
|
onBrushWidthChanged: (size: number) => void;
|
||||||
|
onEraserWidthChanged: (size: number) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -72,30 +69,41 @@ const updateLastCursorPos = (stage: Konva.Stage, setLastCursorPos: Arg['setLastC
|
|||||||
* @param onPointAddedToLine The callback to add a point to a line
|
* @param onPointAddedToLine The callback to add a point to a line
|
||||||
*/
|
*/
|
||||||
const maybeAddNextPoint = (
|
const maybeAddNextPoint = (
|
||||||
selectedLayer: LayerData,
|
selectedEntity: CanvasEntity,
|
||||||
currentPos: Vector2d,
|
currentPos: Vector2d,
|
||||||
|
getToolState: Arg['getToolState'],
|
||||||
getLastAddedPoint: Arg['getLastAddedPoint'],
|
getLastAddedPoint: Arg['getLastAddedPoint'],
|
||||||
setLastAddedPoint: Arg['setLastAddedPoint'],
|
setLastAddedPoint: Arg['setLastAddedPoint'],
|
||||||
getBrushSpacingPx: Arg['getBrushSpacingPx'],
|
|
||||||
onPointAddedToLine: Arg['onPointAddedToLine']
|
onPointAddedToLine: Arg['onPointAddedToLine']
|
||||||
) => {
|
) => {
|
||||||
|
if (selectedEntity.type !== 'layer' && selectedEntity.type !== 'regional_guidance') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
// Continue the last line
|
// Continue the last line
|
||||||
const lastAddedPoint = getLastAddedPoint();
|
const lastAddedPoint = getLastAddedPoint();
|
||||||
|
const toolState = getToolState();
|
||||||
|
const minSpacingPx = toolState.selected === 'brush' ? toolState.brush.width * 0.05 : toolState.eraser.width * 0.05;
|
||||||
if (lastAddedPoint) {
|
if (lastAddedPoint) {
|
||||||
// Dispatching redux events impacts perf substantially - using brush spacing keeps dispatches to a reasonable number
|
// Dispatching redux events impacts perf substantially - using brush spacing keeps dispatches to a reasonable number
|
||||||
if (Math.hypot(lastAddedPoint.x - currentPos.x, lastAddedPoint.y - currentPos.y) < getBrushSpacingPx()) {
|
if (Math.hypot(lastAddedPoint.x - currentPos.x, lastAddedPoint.y - currentPos.y) < minSpacingPx) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
setLastAddedPoint(currentPos);
|
setLastAddedPoint(currentPos);
|
||||||
onPointAddedToLine({ layerId, point: [currentPos.x - selectedLayer.x, currentPos.y - selectedLayer.y] });
|
onPointAddedToLine(
|
||||||
|
{
|
||||||
|
id: selectedEntity.id,
|
||||||
|
point: [currentPos.x - selectedEntity.x, currentPos.y - selectedEntity.y],
|
||||||
|
},
|
||||||
|
selectedEntity.type
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const setStageEventHandlers = ({
|
export const setStageEventHandlers = ({
|
||||||
stage,
|
stage,
|
||||||
getTool,
|
getToolState,
|
||||||
|
getCurrentFill,
|
||||||
setTool,
|
setTool,
|
||||||
getToolBuffer,
|
|
||||||
setToolBuffer,
|
setToolBuffer,
|
||||||
getIsDrawing,
|
getIsDrawing,
|
||||||
setIsDrawing,
|
setIsDrawing,
|
||||||
@ -108,17 +116,14 @@ export const setStageEventHandlers = ({
|
|||||||
getLastAddedPoint,
|
getLastAddedPoint,
|
||||||
setLastAddedPoint,
|
setLastAddedPoint,
|
||||||
setStageAttrs,
|
setStageAttrs,
|
||||||
getBrushColor,
|
getSelectedEntity,
|
||||||
getBrushSize,
|
|
||||||
getBrushSpacingPx,
|
|
||||||
getSelectedLayer,
|
|
||||||
getShouldInvert,
|
|
||||||
getSpaceKey,
|
getSpaceKey,
|
||||||
onBrushLineAdded,
|
onBrushLineAdded,
|
||||||
onEraserLineAdded,
|
onEraserLineAdded,
|
||||||
onPointAddedToLine,
|
onPointAddedToLine,
|
||||||
onRectShapeAdded,
|
onRectShapeAdded,
|
||||||
onBrushSizeChanged,
|
onBrushWidthChanged: onBrushSizeChanged,
|
||||||
|
onEraserWidthChanged: onEraserSizeChanged,
|
||||||
}: Arg): (() => void) => {
|
}: Arg): (() => void) => {
|
||||||
//#region mouseenter
|
//#region mouseenter
|
||||||
stage.on('mouseenter', (e) => {
|
stage.on('mouseenter', (e) => {
|
||||||
@ -126,7 +131,7 @@ export const setStageEventHandlers = ({
|
|||||||
if (!stage) {
|
if (!stage) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const tool = getTool();
|
const tool = getToolState().selected;
|
||||||
stage.findOne<Konva.Layer>(`#${PREVIEW_TOOL_GROUP_ID}`)?.visible(tool === 'brush' || tool === 'eraser');
|
stage.findOne<Konva.Layer>(`#${PREVIEW_TOOL_GROUP_ID}`)?.visible(tool === 'brush' || tool === 'eraser');
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -137,13 +142,13 @@ export const setStageEventHandlers = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
setIsMouseDown(true);
|
setIsMouseDown(true);
|
||||||
const tool = getTool();
|
const toolState = getToolState();
|
||||||
const pos = updateLastCursorPos(stage, setLastCursorPos);
|
const pos = updateLastCursorPos(stage, setLastCursorPos);
|
||||||
const selectedLayer = getSelectedLayer();
|
const selectedEntity = getSelectedEntity();
|
||||||
if (!pos || !selectedLayer) {
|
if (!pos || !selectedEntity) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (selectedLayer.type !== 'regional_guidance_layer' && selectedLayer.type !== 'raster_layer') {
|
if (selectedEntity.type !== 'regional_guidance' && selectedEntity.type !== 'layer') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -155,23 +160,37 @@ export const setStageEventHandlers = ({
|
|||||||
setIsDrawing(true);
|
setIsDrawing(true);
|
||||||
setLastMouseDownPos(pos);
|
setLastMouseDownPos(pos);
|
||||||
|
|
||||||
if (tool === 'brush') {
|
if (toolState.selected === 'brush') {
|
||||||
onBrushLineAdded({
|
onBrushLineAdded(
|
||||||
layerId: selectedLayer.id,
|
{
|
||||||
points: [pos.x - selectedLayer.x, pos.y - selectedLayer.y, pos.x - selectedLayer.x, pos.y - selectedLayer.y],
|
id: selectedEntity.id,
|
||||||
color: selectedLayer.type === 'raster_layer' ? getBrushColor() : DEFAULT_RGBA_COLOR,
|
points: [
|
||||||
});
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
|
],
|
||||||
|
color: getCurrentFill(),
|
||||||
|
width: toolState.brush.width,
|
||||||
|
},
|
||||||
|
selectedEntity.type
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tool === 'eraser') {
|
if (toolState.selected === 'eraser') {
|
||||||
onEraserLineAdded({
|
onEraserLineAdded(
|
||||||
layerId: selectedLayer.id,
|
{
|
||||||
points: [pos.x - selectedLayer.x, pos.y - selectedLayer.y, pos.x - selectedLayer.x, pos.y - selectedLayer.y],
|
id: selectedEntity.id,
|
||||||
});
|
points: [
|
||||||
}
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
if (tool === 'rect') {
|
pos.x - selectedEntity.x,
|
||||||
// Setting the last mouse down pos starts a rect
|
pos.y - selectedEntity.y,
|
||||||
|
],
|
||||||
|
width: toolState.eraser.width,
|
||||||
|
},
|
||||||
|
selectedEntity.type
|
||||||
|
);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -183,12 +202,12 @@ export const setStageEventHandlers = ({
|
|||||||
}
|
}
|
||||||
setIsMouseDown(false);
|
setIsMouseDown(false);
|
||||||
const pos = getLastCursorPos();
|
const pos = getLastCursorPos();
|
||||||
const selectedLayer = getSelectedLayer();
|
const selectedEntity = getSelectedEntity();
|
||||||
|
|
||||||
if (!pos || !selectedLayer) {
|
if (!pos || !selectedEntity) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (selectedLayer.type !== 'regional_guidance_layer' && selectedLayer.type !== 'raster_layer') {
|
if (selectedEntity.type !== 'regional_guidance' && selectedEntity.type !== 'layer') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -197,21 +216,24 @@ export const setStageEventHandlers = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const tool = getTool();
|
const toolState = getToolState();
|
||||||
|
|
||||||
if (tool === 'rect') {
|
if (toolState.selected === 'rect') {
|
||||||
const lastMouseDownPos = getLastMouseDownPos();
|
const lastMouseDownPos = getLastMouseDownPos();
|
||||||
if (lastMouseDownPos) {
|
if (lastMouseDownPos) {
|
||||||
onRectShapeAdded({
|
onRectShapeAdded(
|
||||||
layerId: selectedLayer.id,
|
{
|
||||||
|
id: selectedEntity.id,
|
||||||
rect: {
|
rect: {
|
||||||
x: Math.min(pos.x, lastMouseDownPos.x),
|
x: Math.min(pos.x, lastMouseDownPos.x),
|
||||||
y: Math.min(pos.y, lastMouseDownPos.y),
|
y: Math.min(pos.y, lastMouseDownPos.y),
|
||||||
width: Math.abs(pos.x - lastMouseDownPos.x),
|
width: Math.abs(pos.x - lastMouseDownPos.x),
|
||||||
height: Math.abs(pos.y - lastMouseDownPos.y),
|
height: Math.abs(pos.y - lastMouseDownPos.y),
|
||||||
},
|
},
|
||||||
color: selectedLayer.type === 'raster_layer' ? getBrushColor() : DEFAULT_RGBA_COLOR,
|
color: getCurrentFill(),
|
||||||
});
|
},
|
||||||
|
selectedEntity.type
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -225,16 +247,18 @@ export const setStageEventHandlers = ({
|
|||||||
if (!stage) {
|
if (!stage) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const tool = getTool();
|
const toolState = getToolState();
|
||||||
const pos = updateLastCursorPos(stage, setLastCursorPos);
|
const pos = updateLastCursorPos(stage, setLastCursorPos);
|
||||||
const selectedLayer = getSelectedLayer();
|
const selectedEntity = getSelectedEntity();
|
||||||
|
|
||||||
stage.findOne<Konva.Layer>(`#${PREVIEW_TOOL_GROUP_ID}`)?.visible(tool === 'brush' || tool === 'eraser');
|
stage
|
||||||
|
.findOne<Konva.Layer>(`#${PREVIEW_TOOL_GROUP_ID}`)
|
||||||
|
?.visible(toolState.selected === 'brush' || toolState.selected === 'eraser');
|
||||||
|
|
||||||
if (!pos || !selectedLayer) {
|
if (!pos || !selectedEntity) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (selectedLayer.type !== 'regional_guidance_layer' && selectedLayer.type !== 'raster_layer') {
|
if (selectedEntity.type !== 'regional_guidance' && selectedEntity.type !== 'layer') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -247,45 +271,49 @@ export const setStageEventHandlers = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tool === 'brush') {
|
if (toolState.selected === 'brush') {
|
||||||
if (getIsDrawing()) {
|
if (getIsDrawing()) {
|
||||||
// Continue the last line
|
// Continue the last line
|
||||||
maybeAddNextPoint(
|
maybeAddNextPoint(selectedEntity, pos, getToolState, getLastAddedPoint, setLastAddedPoint, onPointAddedToLine);
|
||||||
selectedLayer.id,
|
|
||||||
pos,
|
|
||||||
getLastAddedPoint,
|
|
||||||
setLastAddedPoint,
|
|
||||||
getBrushSpacingPx,
|
|
||||||
onPointAddedToLine
|
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
// Start a new line
|
// Start a new line
|
||||||
onBrushLineAdded({
|
onBrushLineAdded(
|
||||||
layerId: selectedLayer.id,
|
{
|
||||||
points: [pos.x - selectedLayer.x, pos.y - selectedLayer.y, pos.x - selectedLayer.x, pos.y - selectedLayer.y],
|
id: selectedEntity.id,
|
||||||
color: selectedLayer.type === 'raster_layer' ? getBrushColor() : DEFAULT_RGBA_COLOR,
|
points: [
|
||||||
});
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
|
],
|
||||||
|
width: toolState.brush.width,
|
||||||
|
color: getCurrentFill(),
|
||||||
|
},
|
||||||
|
selectedEntity.type
|
||||||
|
);
|
||||||
setIsDrawing(true);
|
setIsDrawing(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tool === 'eraser') {
|
if (toolState.selected === 'eraser') {
|
||||||
if (getIsDrawing()) {
|
if (getIsDrawing()) {
|
||||||
// Continue the last line
|
// Continue the last line
|
||||||
maybeAddNextPoint(
|
maybeAddNextPoint(selectedEntity, pos, getToolState, getLastAddedPoint, setLastAddedPoint, onPointAddedToLine);
|
||||||
selectedLayer.id,
|
|
||||||
pos,
|
|
||||||
getLastAddedPoint,
|
|
||||||
setLastAddedPoint,
|
|
||||||
getBrushSpacingPx,
|
|
||||||
onPointAddedToLine
|
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
// Start a new line
|
// Start a new line
|
||||||
onEraserLineAdded({
|
onEraserLineAdded(
|
||||||
layerId: selectedLayer.id,
|
{
|
||||||
points: [pos.x - selectedLayer.x, pos.y - selectedLayer.y, pos.x - selectedLayer.x, pos.y - selectedLayer.y],
|
id: selectedEntity.id,
|
||||||
});
|
points: [
|
||||||
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
|
pos.x - selectedEntity.x,
|
||||||
|
pos.y - selectedEntity.y,
|
||||||
|
],
|
||||||
|
width: toolState.eraser.width,
|
||||||
|
},
|
||||||
|
selectedEntity.type
|
||||||
|
);
|
||||||
setIsDrawing(true);
|
setIsDrawing(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -301,15 +329,15 @@ export const setStageEventHandlers = ({
|
|||||||
setIsDrawing(false);
|
setIsDrawing(false);
|
||||||
setLastCursorPos(null);
|
setLastCursorPos(null);
|
||||||
setLastMouseDownPos(null);
|
setLastMouseDownPos(null);
|
||||||
const selectedLayer = getSelectedLayer();
|
const selectedEntity = getSelectedEntity();
|
||||||
const tool = getTool();
|
const toolState = getToolState();
|
||||||
|
|
||||||
stage.findOne<Konva.Layer>(`#${PREVIEW_TOOL_GROUP_ID}`)?.visible(false);
|
stage.findOne<Konva.Layer>(`#${PREVIEW_TOOL_GROUP_ID}`)?.visible(false);
|
||||||
|
|
||||||
if (!pos || !selectedLayer) {
|
if (!pos || !selectedEntity) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (selectedLayer.type !== 'regional_guidance_layer' && selectedLayer.type !== 'raster_layer') {
|
if (selectedEntity.type !== 'regional_guidance' && selectedEntity.type !== 'layer') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (getSpaceKey()) {
|
if (getSpaceKey()) {
|
||||||
@ -317,12 +345,11 @@ export const setStageEventHandlers = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (getIsMouseDown()) {
|
if (getIsMouseDown()) {
|
||||||
if (tool === 'brush') {
|
if (toolState.selected === 'brush') {
|
||||||
onPointAddedToLine({ layerId: selectedLayer.id, point: [pos.x, pos.y] });
|
onPointAddedToLine({ id: selectedEntity.id, point: [pos.x, pos.y] }, selectedEntity.type);
|
||||||
}
|
}
|
||||||
|
if (toolState.selected === 'eraser') {
|
||||||
if (tool === 'eraser') {
|
onPointAddedToLine({ id: selectedEntity.id, point: [pos.x, pos.y] }, selectedEntity.type);
|
||||||
onPointAddedToLine({ layerId: selectedLayer.id, point: [pos.x, pos.y] });
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -331,12 +358,17 @@ export const setStageEventHandlers = ({
|
|||||||
e.evt.preventDefault();
|
e.evt.preventDefault();
|
||||||
|
|
||||||
if (e.evt.ctrlKey || e.evt.metaKey) {
|
if (e.evt.ctrlKey || e.evt.metaKey) {
|
||||||
|
const toolState = getToolState();
|
||||||
let delta = e.evt.deltaY;
|
let delta = e.evt.deltaY;
|
||||||
if (getShouldInvert()) {
|
if (toolState.invertScroll) {
|
||||||
delta = -delta;
|
delta = -delta;
|
||||||
}
|
}
|
||||||
// Holding ctrl or meta while scrolling changes the brush size
|
// Holding ctrl or meta while scrolling changes the brush size
|
||||||
onBrushSizeChanged(calculateNewBrushSize(getBrushSize(), delta));
|
if (toolState.selected === 'brush') {
|
||||||
|
onBrushSizeChanged(calculateNewBrushSize(toolState.brush.width, delta));
|
||||||
|
} else if (toolState.selected === 'eraser') {
|
||||||
|
onEraserSizeChanged(calculateNewBrushSize(toolState.eraser.width, delta));
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// We need the absolute cursor position - not the scaled position
|
// We need the absolute cursor position - not the scaled position
|
||||||
const cursorPos = stage.getPointerPosition();
|
const cursorPos = stage.getPointerPosition();
|
||||||
@ -396,7 +428,7 @@ export const setStageEventHandlers = ({
|
|||||||
setIsDrawing(false);
|
setIsDrawing(false);
|
||||||
setLastMouseDownPos(null);
|
setLastMouseDownPos(null);
|
||||||
} else if (e.key === ' ') {
|
} else if (e.key === ' ') {
|
||||||
setToolBuffer(getTool());
|
setToolBuffer(getToolState().selected);
|
||||||
setTool('view');
|
setTool('view');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -408,7 +440,7 @@ export const setStageEventHandlers = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (e.key === ' ') {
|
if (e.key === ' ') {
|
||||||
const toolBuffer = getToolBuffer();
|
const toolBuffer = getToolState().selectedBuffer;
|
||||||
setTool(toolBuffer ?? 'move');
|
setTool(toolBuffer ?? 'move');
|
||||||
setToolBuffer(null);
|
setToolBuffer(null);
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { LightnessToAlphaFilter } from 'features/controlLayers/konva/filters';
|
import { LightnessToAlphaFilter } from 'features/controlLayers/konva/filters';
|
||||||
import { CA_LAYER_IMAGE_NAME, CA_LAYER_NAME, getCALayerImageId } from 'features/controlLayers/konva/naming';
|
import { CA_LAYER_IMAGE_NAME, CA_LAYER_NAME, getCALayerImageId } from 'features/controlLayers/konva/naming';
|
||||||
import type { ControlAdapterLayer } from 'features/controlLayers/store/types';
|
import type { ControlAdapterData } from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
import type { ImageDTO } from 'services/api/types';
|
import type { ImageDTO } from 'services/api/types';
|
||||||
|
|
||||||
@ -12,11 +12,11 @@ import type { ImageDTO } from 'services/api/types';
|
|||||||
/**
|
/**
|
||||||
* Creates a control adapter layer.
|
* Creates a control adapter layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerState The control adapter layer state
|
* @param ca The control adapter layer state
|
||||||
*/
|
*/
|
||||||
const createCALayer = (stage: Konva.Stage, layerState: ControlAdapterLayer): Konva.Layer => {
|
const createCALayer = (stage: Konva.Stage, ca: ControlAdapterData): Konva.Layer => {
|
||||||
const konvaLayer = new Konva.Layer({
|
const konvaLayer = new Konva.Layer({
|
||||||
id: layerState.id,
|
id: ca.id,
|
||||||
name: CA_LAYER_NAME,
|
name: CA_LAYER_NAME,
|
||||||
imageSmoothingEnabled: false,
|
imageSmoothingEnabled: false,
|
||||||
listening: false,
|
listening: false,
|
||||||
@ -44,16 +44,16 @@ const createCALayerImage = (konvaLayer: Konva.Layer, imageEl: HTMLImageElement):
|
|||||||
* the konva image.
|
* the konva image.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param konvaLayer The konva layer
|
* @param konvaLayer The konva layer
|
||||||
* @param layerState The control adapter layer state
|
* @param ca The control adapter layer state
|
||||||
* @param getImageDTO A function to retrieve an image DTO from the server, used to update the image source
|
* @param getImageDTO A function to retrieve an image DTO from the server, used to update the image source
|
||||||
*/
|
*/
|
||||||
const updateCALayerImageSource = async (
|
const updateCALayerImageSource = async (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
konvaLayer: Konva.Layer,
|
konvaLayer: Konva.Layer,
|
||||||
layerState: ControlAdapterLayer,
|
ca: ControlAdapterData,
|
||||||
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
||||||
): Promise<void> => {
|
): Promise<void> => {
|
||||||
const image = layerState.controlAdapter.processedImage ?? layerState.controlAdapter.image;
|
const image = ca.processedImage ?? ca.image;
|
||||||
if (image) {
|
if (image) {
|
||||||
const imageName = image.name;
|
const imageName = image.name;
|
||||||
const imageDTO = await getImageDTO(imageName);
|
const imageDTO = await getImageDTO(imageName);
|
||||||
@ -61,7 +61,7 @@ const updateCALayerImageSource = async (
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const imageEl = new Image();
|
const imageEl = new Image();
|
||||||
const imageId = getCALayerImageId(layerState.id, imageName);
|
const imageId = getCALayerImageId(ca.id, imageName);
|
||||||
imageEl.onload = () => {
|
imageEl.onload = () => {
|
||||||
// Find the existing image or create a new one - must find using the name, bc the id may have just changed
|
// Find the existing image or create a new one - must find using the name, bc the id may have just changed
|
||||||
const konvaImage =
|
const konvaImage =
|
||||||
@ -72,7 +72,7 @@ const updateCALayerImageSource = async (
|
|||||||
id: imageId,
|
id: imageId,
|
||||||
image: imageEl,
|
image: imageEl,
|
||||||
});
|
});
|
||||||
updateCALayerImageAttrs(stage, konvaImage, layerState);
|
updateCALayerImageAttrs(stage, konvaImage, ca);
|
||||||
// Must cache after this to apply the filters
|
// Must cache after this to apply the filters
|
||||||
konvaImage.cache();
|
konvaImage.cache();
|
||||||
imageEl.id = imageId;
|
imageEl.id = imageId;
|
||||||
@ -87,36 +87,33 @@ const updateCALayerImageSource = async (
|
|||||||
* Updates the image attributes for a control adapter layer's image (width, height, visibility, opacity, filters).
|
* Updates the image attributes for a control adapter layer's image (width, height, visibility, opacity, filters).
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param konvaImage The konva image
|
* @param konvaImage The konva image
|
||||||
* @param layerState The control adapter layer state
|
* @param ca The control adapter layer state
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const updateCALayerImageAttrs = (
|
const updateCALayerImageAttrs = (stage: Konva.Stage, konvaImage: Konva.Image, ca: ControlAdapterData): void => {
|
||||||
stage: Konva.Stage,
|
|
||||||
konvaImage: Konva.Image,
|
|
||||||
layerState: ControlAdapterLayer
|
|
||||||
): void => {
|
|
||||||
let needsCache = false;
|
let needsCache = false;
|
||||||
// Konva erroneously reports NaN for width and height when the stage is hidden. This causes errors when caching,
|
// Konva erroneously reports NaN for width and height when the stage is hidden. This causes errors when caching,
|
||||||
// but it doesn't seem to break anything.
|
// but it doesn't seem to break anything.
|
||||||
// TODO(psyche): Investigate and report upstream.
|
// TODO(psyche): Investigate and report upstream.
|
||||||
const hasFilter = konvaImage.filters() !== null && konvaImage.filters().length > 0;
|
const filter = konvaImage.filters()[0] ?? null;
|
||||||
|
const filterNeedsUpdate = (filter === null && ca.filter !== 'none') || (filter && filter.name !== ca.filter);
|
||||||
if (
|
if (
|
||||||
konvaImage.x() !== layerState.x ||
|
konvaImage.x() !== ca.x ||
|
||||||
konvaImage.y() !== layerState.y ||
|
konvaImage.y() !== ca.y ||
|
||||||
konvaImage.visible() !== layerState.isEnabled ||
|
konvaImage.visible() !== ca.isEnabled ||
|
||||||
hasFilter !== layerState.isFilterEnabled
|
filterNeedsUpdate
|
||||||
) {
|
) {
|
||||||
konvaImage.setAttrs({
|
konvaImage.setAttrs({
|
||||||
opacity: layerState.opacity,
|
opacity: ca.opacity,
|
||||||
scaleX: 1,
|
scaleX: 1,
|
||||||
scaleY: 1,
|
scaleY: 1,
|
||||||
visible: layerState.isEnabled,
|
visible: ca.isEnabled,
|
||||||
filters: layerState.isFilterEnabled ? [LightnessToAlphaFilter] : [],
|
filters: ca.filter === LightnessToAlphaFilter.name ? [LightnessToAlphaFilter] : [],
|
||||||
});
|
});
|
||||||
needsCache = true;
|
needsCache = true;
|
||||||
}
|
}
|
||||||
if (konvaImage.opacity() !== layerState.opacity) {
|
if (konvaImage.opacity() !== ca.opacity) {
|
||||||
konvaImage.opacity(layerState.opacity);
|
konvaImage.opacity(ca.opacity);
|
||||||
}
|
}
|
||||||
if (needsCache) {
|
if (needsCache) {
|
||||||
konvaImage.cache();
|
konvaImage.cache();
|
||||||
@ -127,16 +124,16 @@ const updateCALayerImageAttrs = (
|
|||||||
* Renders a control adapter layer. If the layer doesn't already exist, it is created. Otherwise, the layer is updated
|
* Renders a control adapter layer. If the layer doesn't already exist, it is created. Otherwise, the layer is updated
|
||||||
* with the current image source and attributes.
|
* with the current image source and attributes.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerState The control adapter layer state
|
* @param ca The control adapter layer state
|
||||||
* @param getImageDTO A function to retrieve an image DTO from the server, used to update the image source
|
* @param getImageDTO A function to retrieve an image DTO from the server, used to update the image source
|
||||||
*/
|
*/
|
||||||
export const renderCALayer = (
|
export const renderCALayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
layerState: ControlAdapterLayer,
|
ca: ControlAdapterData,
|
||||||
zIndex: number,
|
zIndex: number,
|
||||||
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
||||||
): void => {
|
): void => {
|
||||||
const konvaLayer = stage.findOne<Konva.Layer>(`#${layerState.id}`) ?? createCALayer(stage, layerState);
|
const konvaLayer = stage.findOne<Konva.Layer>(`#${ca.id}`) ?? createCALayer(stage, ca);
|
||||||
|
|
||||||
konvaLayer.zIndex(zIndex);
|
konvaLayer.zIndex(zIndex);
|
||||||
|
|
||||||
@ -146,8 +143,8 @@ export const renderCALayer = (
|
|||||||
let imageSourceNeedsUpdate = false;
|
let imageSourceNeedsUpdate = false;
|
||||||
|
|
||||||
if (canvasImageSource instanceof HTMLImageElement) {
|
if (canvasImageSource instanceof HTMLImageElement) {
|
||||||
const image = layerState.controlAdapter.processedImage ?? layerState.controlAdapter.image;
|
const image = ca.processedImage ?? ca.image;
|
||||||
if (image && canvasImageSource.id !== getCALayerImageId(layerState.id, image.name)) {
|
if (image && canvasImageSource.id !== getCALayerImageId(ca.id, image.name)) {
|
||||||
imageSourceNeedsUpdate = true;
|
imageSourceNeedsUpdate = true;
|
||||||
} else if (!image) {
|
} else if (!image) {
|
||||||
imageSourceNeedsUpdate = true;
|
imageSourceNeedsUpdate = true;
|
||||||
@ -157,8 +154,8 @@ export const renderCALayer = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (imageSourceNeedsUpdate) {
|
if (imageSourceNeedsUpdate) {
|
||||||
updateCALayerImageSource(stage, konvaLayer, layerState, getImageDTO);
|
updateCALayerImageSource(stage, konvaLayer, ca, getImageDTO);
|
||||||
} else if (konvaImage) {
|
} else if (konvaImage) {
|
||||||
updateCALayerImageAttrs(stage, konvaImage, layerState);
|
updateCALayerImageAttrs(stage, konvaImage, ca);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -2,19 +2,17 @@ import { DEBOUNCE_MS } from 'features/controlLayers/konva/constants';
|
|||||||
import { PREVIEW_LAYER_ID } from 'features/controlLayers/konva/naming';
|
import { PREVIEW_LAYER_ID } from 'features/controlLayers/konva/naming';
|
||||||
import { updateBboxes } from 'features/controlLayers/konva/renderers/bbox';
|
import { updateBboxes } from 'features/controlLayers/konva/renderers/bbox';
|
||||||
import { renderCALayer } from 'features/controlLayers/konva/renderers/caLayer';
|
import { renderCALayer } from 'features/controlLayers/konva/renderers/caLayer';
|
||||||
import { renderIILayer } from 'features/controlLayers/konva/renderers/iiLayer';
|
|
||||||
import { renderBboxPreview, renderToolPreview } from 'features/controlLayers/konva/renderers/previewLayer';
|
import { renderBboxPreview, renderToolPreview } from 'features/controlLayers/konva/renderers/previewLayer';
|
||||||
import { renderRasterLayer } from 'features/controlLayers/konva/renderers/rasterLayer';
|
import { renderRasterLayer } from 'features/controlLayers/konva/renderers/rasterLayer';
|
||||||
import { renderRGLayer } from 'features/controlLayers/konva/renderers/rgLayer';
|
import { renderRGLayer } from 'features/controlLayers/konva/renderers/rgLayer';
|
||||||
import { mapId, selectRenderableLayers } from 'features/controlLayers/konva/util';
|
import { mapId, selectRenderableLayers } from 'features/controlLayers/konva/util';
|
||||||
import type { LayerData, Tool } from 'features/controlLayers/store/types';
|
import type {
|
||||||
import {
|
CanvasEntity,
|
||||||
isControlAdapterLayer,
|
ControlAdapterData,
|
||||||
isInitialImageLayer,
|
LayerData,
|
||||||
isInpaintMaskLayer,
|
PosChangedArg,
|
||||||
isRasterLayer,
|
RegionalGuidanceData,
|
||||||
isRegionalGuidanceLayer,
|
Tool,
|
||||||
isRenderableLayer,
|
|
||||||
} from 'features/controlLayers/store/types';
|
} from 'features/controlLayers/store/types';
|
||||||
import type Konva from 'konva';
|
import type Konva from 'konva';
|
||||||
import { debounce } from 'lodash-es';
|
import { debounce } from 'lodash-es';
|
||||||
@ -27,43 +25,42 @@ import type { ImageDTO } from 'services/api/types';
|
|||||||
/**
|
/**
|
||||||
* Renders the layers on the stage.
|
* Renders the layers on the stage.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerStates Array of all layer states
|
* @param layers Array of all layer states
|
||||||
* @param globalMaskLayerOpacity The global mask layer opacity
|
* @param rgGlobalOpacity The global mask layer opacity
|
||||||
* @param tool The current tool
|
* @param tool The current tool
|
||||||
* @param getImageDTO A function to retrieve an image DTO from the server, used to update the image source
|
* @param getImageDTO A function to retrieve an image DTO from the server, used to update the image source
|
||||||
* @param onLayerPosChanged Callback for when the layer's position changes
|
* @param onPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
const renderLayers = (
|
const renderLayers = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
layerStates: LayerData[],
|
layers: LayerData[],
|
||||||
globalMaskLayerOpacity: number,
|
controlAdapters: ControlAdapterData[],
|
||||||
|
regions: RegionalGuidanceData[],
|
||||||
|
rgGlobalOpacity: number,
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
|
selectedEntity: CanvasEntity | null,
|
||||||
getImageDTO: (imageName: string) => Promise<ImageDTO | null>,
|
getImageDTO: (imageName: string) => Promise<ImageDTO | null>,
|
||||||
onLayerPosChanged?: (layerId: string, x: number, y: number) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): void => {
|
): void => {
|
||||||
const layerIds = layerStates.filter(isRenderableLayer).map(mapId);
|
const renderableIds = [...layers.map(mapId), ...controlAdapters.map(mapId), ...regions.map(mapId)];
|
||||||
// Remove un-rendered layers
|
// Remove un-rendered layers
|
||||||
for (const konvaLayer of stage.find<Konva.Layer>(selectRenderableLayers)) {
|
for (const konvaLayer of stage.find<Konva.Layer>(selectRenderableLayers)) {
|
||||||
if (!layerIds.includes(konvaLayer.id())) {
|
if (!renderableIds.includes(konvaLayer.id())) {
|
||||||
konvaLayer.destroy();
|
konvaLayer.destroy();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// We'll need to ensure the tool preview layer is on top of the rest of the layers
|
// We'll need to ensure the tool preview layer is on top of the rest of the layers
|
||||||
let zIndex = 0;
|
let zIndex = 0;
|
||||||
for (const layer of layerStates) {
|
for (const layer of layers) {
|
||||||
if (isRegionalGuidanceLayer(layer)) {
|
renderRasterLayer(stage, layer, tool, zIndex, onPosChanged);
|
||||||
renderRGLayer(stage, layer, globalMaskLayerOpacity, tool, zIndex, onLayerPosChanged);
|
zIndex++;
|
||||||
} else if (isControlAdapterLayer(layer)) {
|
|
||||||
renderCALayer(stage, layer, zIndex, getImageDTO);
|
|
||||||
} else if (isInitialImageLayer(layer)) {
|
|
||||||
renderIILayer(stage, layer, zIndex, getImageDTO);
|
|
||||||
} else if (isRasterLayer(layer)) {
|
|
||||||
renderRasterLayer(stage, layer, tool, zIndex, onLayerPosChanged);
|
|
||||||
} else if (isInpaintMaskLayer(layer)) {
|
|
||||||
//
|
|
||||||
}
|
}
|
||||||
// IP Adapter layers are not rendered
|
for (const ca of controlAdapters) {
|
||||||
// Increment the z-index for the tool layer
|
renderCALayer(stage, ca, zIndex, getImageDTO);
|
||||||
|
zIndex++;
|
||||||
|
}
|
||||||
|
for (const rg of regions) {
|
||||||
|
renderRGLayer(stage, rg, rgGlobalOpacity, tool, zIndex, selectedEntity, onPosChanged);
|
||||||
zIndex++;
|
zIndex++;
|
||||||
}
|
}
|
||||||
// Arrange the tool preview layer
|
// Arrange the tool preview layer
|
||||||
|
@ -5,7 +5,13 @@ import {
|
|||||||
LAYER_BBOX_NAME,
|
LAYER_BBOX_NAME,
|
||||||
PREVIEW_GENERATION_BBOX_DUMMY_RECT,
|
PREVIEW_GENERATION_BBOX_DUMMY_RECT,
|
||||||
} from 'features/controlLayers/konva/naming';
|
} from 'features/controlLayers/konva/naming';
|
||||||
import type { BrushLine, EraserLine, ImageObject, LayerData, RectShape } from 'features/controlLayers/store/types';
|
import type {
|
||||||
|
BrushLine,
|
||||||
|
CanvasEntity,
|
||||||
|
EraserLine,
|
||||||
|
ImageObject,
|
||||||
|
RectShape,
|
||||||
|
} from 'features/controlLayers/store/types';
|
||||||
import { DEFAULT_RGBA_COLOR } from 'features/controlLayers/store/types';
|
import { DEFAULT_RGBA_COLOR } from 'features/controlLayers/store/types';
|
||||||
import { t } from 'i18next';
|
import { t } from 'i18next';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
@ -174,12 +180,12 @@ export const createImageObjectGroup = async (
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a bounding box rect for a layer.
|
* Creates a bounding box rect for a layer.
|
||||||
* @param layerState The layer state for the layer to create the bounding box for
|
* @param entity The layer state for the layer to create the bounding box for
|
||||||
* @param konvaLayer The konva layer to attach the bounding box to
|
* @param konvaLayer The konva layer to attach the bounding box to
|
||||||
*/
|
*/
|
||||||
export const createBboxRect = (layerState: LayerData, konvaLayer: Konva.Layer): Konva.Rect => {
|
export const createBboxRect = (entity: CanvasEntity, konvaLayer: Konva.Layer): Konva.Rect => {
|
||||||
const rect = new Konva.Rect({
|
const rect = new Konva.Rect({
|
||||||
id: getLayerBboxId(layerState.id),
|
id: getLayerBboxId(entity.id),
|
||||||
name: LAYER_BBOX_NAME,
|
name: LAYER_BBOX_NAME,
|
||||||
strokeWidth: 1,
|
strokeWidth: 1,
|
||||||
visible: false,
|
visible: false,
|
||||||
|
@ -18,7 +18,7 @@ import {
|
|||||||
PREVIEW_TOOL_GROUP_ID,
|
PREVIEW_TOOL_GROUP_ID,
|
||||||
} from 'features/controlLayers/konva/naming';
|
} from 'features/controlLayers/konva/naming';
|
||||||
import { selectRenderableLayers } from 'features/controlLayers/konva/util';
|
import { selectRenderableLayers } from 'features/controlLayers/konva/util';
|
||||||
import type { LayerData, RgbaColor, Tool } from 'features/controlLayers/store/types';
|
import type { CanvasEntity, CanvasV2State, RgbaColor, Tool } from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
import type { IRect, Vector2d } from 'konva/lib/types';
|
import type { IRect, Vector2d } from 'konva/lib/types';
|
||||||
import { atom } from 'nanostores';
|
import { atom } from 'nanostores';
|
||||||
@ -327,8 +327,8 @@ export const getToolPreviewGroup = (stage: Konva.Stage): Konva.Group => {
|
|||||||
* Renders the preview layer.
|
* Renders the preview layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param tool The selected tool
|
* @param tool The selected tool
|
||||||
* @param color The selected layer's color
|
* @param currentFill The selected layer's color
|
||||||
* @param selectedLayerType The selected layer's type
|
* @param selectedEntity The selected layer's type
|
||||||
* @param globalMaskLayerOpacity The global mask layer opacity
|
* @param globalMaskLayerOpacity The global mask layer opacity
|
||||||
* @param cursorPos The cursor position
|
* @param cursorPos The cursor position
|
||||||
* @param lastMouseDownPos The position of the last mouse down event - used for the rect tool
|
* @param lastMouseDownPos The position of the last mouse down event - used for the rect tool
|
||||||
@ -336,17 +336,16 @@ export const getToolPreviewGroup = (stage: Konva.Stage): Konva.Group => {
|
|||||||
*/
|
*/
|
||||||
export const renderToolPreview = (
|
export const renderToolPreview = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
tool: Tool,
|
toolState: CanvasV2State['tool'],
|
||||||
brushColor: RgbaColor,
|
currentFill: RgbaColor,
|
||||||
selectedLayerType: LayerData['type'] | null,
|
selectedEntity: CanvasEntity | null,
|
||||||
globalMaskLayerOpacity: number,
|
|
||||||
cursorPos: Vector2d | null,
|
cursorPos: Vector2d | null,
|
||||||
lastMouseDownPos: Vector2d | null,
|
lastMouseDownPos: Vector2d | null,
|
||||||
brushSize: number,
|
|
||||||
isDrawing: boolean,
|
isDrawing: boolean,
|
||||||
isMouseDown: boolean
|
isMouseDown: boolean
|
||||||
): void => {
|
): void => {
|
||||||
const layerCount = stage.find(selectRenderableLayers).length;
|
const layerCount = stage.find(selectRenderableLayers).length;
|
||||||
|
const tool = toolState.selected;
|
||||||
// Update the stage's pointer style
|
// Update the stage's pointer style
|
||||||
if (tool === 'view') {
|
if (tool === 'view') {
|
||||||
// View gets a hand
|
// View gets a hand
|
||||||
@ -354,7 +353,7 @@ export const renderToolPreview = (
|
|||||||
} else if (layerCount === 0) {
|
} else if (layerCount === 0) {
|
||||||
// We have no layers, so we should not render any tool
|
// We have no layers, so we should not render any tool
|
||||||
stage.container().style.cursor = 'default';
|
stage.container().style.cursor = 'default';
|
||||||
} else if (selectedLayerType !== 'regional_guidance_layer' && selectedLayerType !== 'raster_layer') {
|
} else if (selectedEntity?.type !== 'regional_guidance' && selectedEntity?.type !== 'layer') {
|
||||||
// Non-mask-guidance layers don't have tools
|
// Non-mask-guidance layers don't have tools
|
||||||
stage.container().style.cursor = 'not-allowed';
|
stage.container().style.cursor = 'not-allowed';
|
||||||
} else if (tool === 'move') {
|
} else if (tool === 'move') {
|
||||||
@ -377,7 +376,7 @@ export const renderToolPreview = (
|
|||||||
if (
|
if (
|
||||||
!cursorPos ||
|
!cursorPos ||
|
||||||
layerCount === 0 ||
|
layerCount === 0 ||
|
||||||
(selectedLayerType !== 'regional_guidance_layer' && selectedLayerType !== 'raster_layer')
|
(selectedEntity?.type !== 'regional_guidance' && selectedEntity?.type !== 'layer')
|
||||||
) {
|
) {
|
||||||
// We can bail early if the mouse isn't over the stage or there are no layers
|
// We can bail early if the mouse isn't over the stage or there are no layers
|
||||||
toolPreviewGroup.visible(false);
|
toolPreviewGroup.visible(false);
|
||||||
@ -394,24 +393,25 @@ export const renderToolPreview = (
|
|||||||
if (cursorPos && (tool === 'brush' || tool === 'eraser')) {
|
if (cursorPos && (tool === 'brush' || tool === 'eraser')) {
|
||||||
// Update the fill circle
|
// Update the fill circle
|
||||||
const brushPreviewFill = brushPreviewGroup.findOne<Konva.Circle>(`#${PREVIEW_BRUSH_FILL_ID}`);
|
const brushPreviewFill = brushPreviewGroup.findOne<Konva.Circle>(`#${PREVIEW_BRUSH_FILL_ID}`);
|
||||||
|
const radius = (tool === 'brush' ? toolState.brush.width : toolState.eraser.width) / 2;
|
||||||
brushPreviewFill?.setAttrs({
|
brushPreviewFill?.setAttrs({
|
||||||
x: cursorPos.x,
|
x: cursorPos.x,
|
||||||
y: cursorPos.y,
|
y: cursorPos.y,
|
||||||
radius: brushSize / 2,
|
radius,
|
||||||
fill: isDrawing ? '' : rgbaColorToString(brushColor),
|
fill: isDrawing ? '' : rgbaColorToString(currentFill),
|
||||||
globalCompositeOperation: tool === 'brush' ? 'source-over' : 'destination-out',
|
globalCompositeOperation: tool === 'brush' ? 'source-over' : 'destination-out',
|
||||||
});
|
});
|
||||||
|
|
||||||
// Update the inner border of the brush preview
|
// Update the inner border of the brush preview
|
||||||
const brushPreviewInner = brushPreviewGroup.findOne<Konva.Circle>(`#${PREVIEW_BRUSH_BORDER_INNER_ID}`);
|
const brushPreviewInner = brushPreviewGroup.findOne<Konva.Circle>(`#${PREVIEW_BRUSH_BORDER_INNER_ID}`);
|
||||||
brushPreviewInner?.setAttrs({ x: cursorPos.x, y: cursorPos.y, radius: brushSize / 2 });
|
brushPreviewInner?.setAttrs({ x: cursorPos.x, y: cursorPos.y, radius });
|
||||||
|
|
||||||
// Update the outer border of the brush preview
|
// Update the outer border of the brush preview
|
||||||
const brushPreviewOuter = brushPreviewGroup.findOne<Konva.Circle>(`#${PREVIEW_BRUSH_BORDER_OUTER_ID}`);
|
const brushPreviewOuter = brushPreviewGroup.findOne<Konva.Circle>(`#${PREVIEW_BRUSH_BORDER_OUTER_ID}`);
|
||||||
brushPreviewOuter?.setAttrs({
|
brushPreviewOuter?.setAttrs({
|
||||||
x: cursorPos.x,
|
x: cursorPos.x,
|
||||||
y: cursorPos.y,
|
y: cursorPos.y,
|
||||||
radius: brushSize / 2 + 1,
|
radius: radius + 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
brushPreviewGroup.visible(true);
|
brushPreviewGroup.visible(true);
|
||||||
@ -426,7 +426,7 @@ export const renderToolPreview = (
|
|||||||
y: Math.min(cursorPos.y, lastMouseDownPos.y),
|
y: Math.min(cursorPos.y, lastMouseDownPos.y),
|
||||||
width: Math.abs(cursorPos.x - lastMouseDownPos.x),
|
width: Math.abs(cursorPos.x - lastMouseDownPos.x),
|
||||||
height: Math.abs(cursorPos.y - lastMouseDownPos.y),
|
height: Math.abs(cursorPos.y - lastMouseDownPos.y),
|
||||||
fill: rgbaColorToString(brushColor),
|
fill: rgbaColorToString(currentFill),
|
||||||
});
|
});
|
||||||
rectPreview?.visible(true);
|
rectPreview?.visible(true);
|
||||||
} else {
|
} else {
|
||||||
|
@ -1,6 +1,4 @@
|
|||||||
import { BBOX_SELECTED_STROKE } from 'features/controlLayers/konva/constants';
|
|
||||||
import {
|
import {
|
||||||
LAYER_BBOX_NAME,
|
|
||||||
RASTER_LAYER_BRUSH_LINE_NAME,
|
RASTER_LAYER_BRUSH_LINE_NAME,
|
||||||
RASTER_LAYER_ERASER_LINE_NAME,
|
RASTER_LAYER_ERASER_LINE_NAME,
|
||||||
RASTER_LAYER_IMAGE_NAME,
|
RASTER_LAYER_IMAGE_NAME,
|
||||||
@ -9,7 +7,6 @@ import {
|
|||||||
RASTER_LAYER_RECT_SHAPE_NAME,
|
RASTER_LAYER_RECT_SHAPE_NAME,
|
||||||
} from 'features/controlLayers/konva/naming';
|
} from 'features/controlLayers/konva/naming';
|
||||||
import {
|
import {
|
||||||
createBboxRect,
|
|
||||||
createBrushLine,
|
createBrushLine,
|
||||||
createEraserLine,
|
createEraserLine,
|
||||||
createImageObjectGroup,
|
createImageObjectGroup,
|
||||||
@ -17,7 +14,7 @@ import {
|
|||||||
createRectShape,
|
createRectShape,
|
||||||
} from 'features/controlLayers/konva/renderers/objects';
|
} from 'features/controlLayers/konva/renderers/objects';
|
||||||
import { mapId, selectRasterObjects } from 'features/controlLayers/konva/util';
|
import { mapId, selectRasterObjects } from 'features/controlLayers/konva/util';
|
||||||
import type { RasterLayer, Tool } from 'features/controlLayers/store/types';
|
import type { CanvasEntity, LayerData, PosChangedArg, Tool } from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -28,12 +25,12 @@ import Konva from 'konva';
|
|||||||
* Creates a raster layer.
|
* Creates a raster layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerState The raster layer state
|
* @param layerState The raster layer state
|
||||||
* @param onLayerPosChanged Callback for when the layer's position changes
|
* @param onPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
const createRasterLayer = (
|
const createRasterLayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
layerState: RasterLayer,
|
layerState: LayerData,
|
||||||
onLayerPosChanged?: (layerId: string, x: number, y: number) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): Konva.Layer => {
|
): Konva.Layer => {
|
||||||
// This layer hasn't been added to the konva state yet
|
// This layer hasn't been added to the konva state yet
|
||||||
const konvaLayer = new Konva.Layer({
|
const konvaLayer = new Konva.Layer({
|
||||||
@ -45,9 +42,9 @@ const createRasterLayer = (
|
|||||||
|
|
||||||
// When a drag on the layer finishes, update the layer's position in state. During the drag, konva handles changing
|
// When a drag on the layer finishes, update the layer's position in state. During the drag, konva handles changing
|
||||||
// the position - we do not need to call this on the `dragmove` event.
|
// the position - we do not need to call this on the `dragmove` event.
|
||||||
if (onLayerPosChanged) {
|
if (onPosChanged) {
|
||||||
konvaLayer.on('dragend', function (e) {
|
konvaLayer.on('dragend', function (e) {
|
||||||
onLayerPosChanged(layerState.id, Math.floor(e.target.x()), Math.floor(e.target.y()));
|
onPosChanged({ id: layerState.id, x: Math.floor(e.target.x()), y: Math.floor(e.target.y()) }, 'layer');
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -61,17 +58,17 @@ const createRasterLayer = (
|
|||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerState The regional guidance layer state
|
* @param layerState The regional guidance layer state
|
||||||
* @param tool The current tool
|
* @param tool The current tool
|
||||||
* @param onLayerPosChanged Callback for when the layer's position changes
|
* @param onPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
export const renderRasterLayer = async (
|
export const renderRasterLayer = async (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
layerState: RasterLayer,
|
layerState: LayerData,
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
zIndex: number,
|
zIndex: number,
|
||||||
onLayerPosChanged?: (layerId: string, x: number, y: number) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
) => {
|
) => {
|
||||||
const konvaLayer =
|
const konvaLayer =
|
||||||
stage.findOne<Konva.Layer>(`#${layerState.id}`) ?? createRasterLayer(stage, layerState, onLayerPosChanged);
|
stage.findOne<Konva.Layer>(`#${layerState.id}`) ?? createRasterLayer(stage, layerState, onPosChanged);
|
||||||
|
|
||||||
// Update the layer's position and listening state
|
// Update the layer's position and listening state
|
||||||
konvaLayer.setAttrs({
|
konvaLayer.setAttrs({
|
||||||
@ -128,23 +125,23 @@ export const renderRasterLayer = async (
|
|||||||
konvaLayer.visible(layerState.isEnabled);
|
konvaLayer.visible(layerState.isEnabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(layerState, konvaLayer);
|
// const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(layerState, konvaLayer);
|
||||||
|
|
||||||
if (layerState.bbox) {
|
// if (layerState.bbox) {
|
||||||
const active = !layerState.bboxNeedsUpdate && layerState.isSelected && tool === 'move';
|
// const active = !layerState.bboxNeedsUpdate && layerState.isSelected && tool === 'move';
|
||||||
bboxRect.setAttrs({
|
// bboxRect.setAttrs({
|
||||||
visible: active,
|
// visible: active,
|
||||||
listening: active,
|
// listening: active,
|
||||||
x: layerState.bbox.x,
|
// x: layerState.bbox.x,
|
||||||
y: layerState.bbox.y,
|
// y: layerState.bbox.y,
|
||||||
width: layerState.bbox.width,
|
// width: layerState.bbox.width,
|
||||||
height: layerState.bbox.height,
|
// height: layerState.bbox.height,
|
||||||
stroke: layerState.isSelected ? BBOX_SELECTED_STROKE : '',
|
// stroke: layerState.isSelected ? BBOX_SELECTED_STROKE : '',
|
||||||
strokeWidth: 1 / stage.scaleX(),
|
// strokeWidth: 1 / stage.scaleX(),
|
||||||
});
|
// });
|
||||||
} else {
|
// } else {
|
||||||
bboxRect.visible(false);
|
// bboxRect.visible(false);
|
||||||
}
|
// }
|
||||||
|
|
||||||
konvaObjectGroup.opacity(layerState.opacity);
|
konvaObjectGroup.opacity(layerState.opacity);
|
||||||
};
|
};
|
||||||
|
@ -18,7 +18,7 @@ import {
|
|||||||
createRectShape,
|
createRectShape,
|
||||||
} from 'features/controlLayers/konva/renderers/objects';
|
} from 'features/controlLayers/konva/renderers/objects';
|
||||||
import { mapId, selectVectorMaskObjects } from 'features/controlLayers/konva/util';
|
import { mapId, selectVectorMaskObjects } from 'features/controlLayers/konva/util';
|
||||||
import type { RegionalGuidanceLayer, Tool } from 'features/controlLayers/store/types';
|
import type { CanvasEntity, PosChangedArg, RegionalGuidanceData, Tool } from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -41,17 +41,17 @@ const createCompositingRect = (konvaLayer: Konva.Layer): Konva.Rect => {
|
|||||||
/**
|
/**
|
||||||
* Creates a regional guidance layer.
|
* Creates a regional guidance layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerState The regional guidance layer state
|
* @param rg The regional guidance layer state
|
||||||
* @param onLayerPosChanged Callback for when the layer's position changes
|
* @param onLayerPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
const createRGLayer = (
|
const createRGLayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
layerState: RegionalGuidanceLayer,
|
rg: RegionalGuidanceData,
|
||||||
onLayerPosChanged?: (layerId: string, x: number, y: number) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): Konva.Layer => {
|
): Konva.Layer => {
|
||||||
// This layer hasn't been added to the konva state yet
|
// This layer hasn't been added to the konva state yet
|
||||||
const konvaLayer = new Konva.Layer({
|
const konvaLayer = new Konva.Layer({
|
||||||
id: layerState.id,
|
id: rg.id,
|
||||||
name: RG_LAYER_NAME,
|
name: RG_LAYER_NAME,
|
||||||
draggable: true,
|
draggable: true,
|
||||||
dragDistance: 0,
|
dragDistance: 0,
|
||||||
@ -59,9 +59,9 @@ const createRGLayer = (
|
|||||||
|
|
||||||
// When a drag on the layer finishes, update the layer's position in state. During the drag, konva handles changing
|
// When a drag on the layer finishes, update the layer's position in state. During the drag, konva handles changing
|
||||||
// the position - we do not need to call this on the `dragmove` event.
|
// the position - we do not need to call this on the `dragmove` event.
|
||||||
if (onLayerPosChanged) {
|
if (onPosChanged) {
|
||||||
konvaLayer.on('dragend', function (e) {
|
konvaLayer.on('dragend', function (e) {
|
||||||
onLayerPosChanged(layerState.id, Math.floor(e.target.x()), Math.floor(e.target.y()));
|
onPosChanged({ id: rg.id, x: Math.floor(e.target.x()), y: Math.floor(e.target.y()) }, 'regional_guidance');
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -73,32 +73,32 @@ const createRGLayer = (
|
|||||||
/**
|
/**
|
||||||
* Renders a raster layer.
|
* Renders a raster layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param layerState The regional guidance layer state
|
* @param rg The regional guidance layer state
|
||||||
* @param globalMaskLayerOpacity The global mask layer opacity
|
* @param globalMaskLayerOpacity The global mask layer opacity
|
||||||
* @param tool The current tool
|
* @param tool The current tool
|
||||||
* @param onLayerPosChanged Callback for when the layer's position changes
|
* @param onPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
export const renderRGLayer = (
|
export const renderRGLayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
layerState: RegionalGuidanceLayer,
|
rg: RegionalGuidanceData,
|
||||||
globalMaskLayerOpacity: number,
|
globalMaskLayerOpacity: number,
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
zIndex: number,
|
zIndex: number,
|
||||||
onLayerPosChanged?: (layerId: string, x: number, y: number) => void
|
selectedEntity: CanvasEntity | null,
|
||||||
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): void => {
|
): void => {
|
||||||
const konvaLayer =
|
const konvaLayer = stage.findOne<Konva.Layer>(`#${rg.id}`) ?? createRGLayer(stage, rg, onPosChanged);
|
||||||
stage.findOne<Konva.Layer>(`#${layerState.id}`) ?? createRGLayer(stage, layerState, onLayerPosChanged);
|
|
||||||
|
|
||||||
// Update the layer's position and listening state
|
// Update the layer's position and listening state
|
||||||
konvaLayer.setAttrs({
|
konvaLayer.setAttrs({
|
||||||
listening: tool === 'move', // The layer only listens when using the move tool - otherwise the stage is handling mouse events
|
listening: tool === 'move', // The layer only listens when using the move tool - otherwise the stage is handling mouse events
|
||||||
x: Math.floor(layerState.x),
|
x: Math.floor(rg.x),
|
||||||
y: Math.floor(layerState.y),
|
y: Math.floor(rg.y),
|
||||||
zIndex,
|
zIndex,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Convert the color to a string, stripping the alpha - the object group will handle opacity.
|
// Convert the color to a string, stripping the alpha - the object group will handle opacity.
|
||||||
const rgbColor = rgbColorToString(layerState.previewColor);
|
const rgbColor = rgbColorToString(rg.fill);
|
||||||
|
|
||||||
const konvaObjectGroup =
|
const konvaObjectGroup =
|
||||||
konvaLayer.findOne<Konva.Group>(`.${RG_LAYER_OBJECT_GROUP_NAME}`) ??
|
konvaLayer.findOne<Konva.Group>(`.${RG_LAYER_OBJECT_GROUP_NAME}`) ??
|
||||||
@ -107,7 +107,7 @@ export const renderRGLayer = (
|
|||||||
// We use caching to handle "global" layer opacity, but caching is expensive and we should only do it when required.
|
// We use caching to handle "global" layer opacity, but caching is expensive and we should only do it when required.
|
||||||
let groupNeedsCache = false;
|
let groupNeedsCache = false;
|
||||||
|
|
||||||
const objectIds = layerState.objects.map(mapId);
|
const objectIds = rg.objects.map(mapId);
|
||||||
// Destroy any objects that are no longer in the redux state
|
// Destroy any objects that are no longer in the redux state
|
||||||
for (const objectNode of konvaObjectGroup.find(selectVectorMaskObjects)) {
|
for (const objectNode of konvaObjectGroup.find(selectVectorMaskObjects)) {
|
||||||
if (!objectIds.includes(objectNode.id())) {
|
if (!objectIds.includes(objectNode.id())) {
|
||||||
@ -116,7 +116,7 @@ export const renderRGLayer = (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const obj of layerState.objects) {
|
for (const obj of rg.objects) {
|
||||||
if (obj.type === 'brush_line') {
|
if (obj.type === 'brush_line') {
|
||||||
const konvaBrushLine =
|
const konvaBrushLine =
|
||||||
stage.findOne<Konva.Line>(`#${obj.id}`) ?? createBrushLine(obj, konvaObjectGroup, RG_LAYER_BRUSH_LINE_NAME);
|
stage.findOne<Konva.Line>(`#${obj.id}`) ?? createBrushLine(obj, konvaObjectGroup, RG_LAYER_BRUSH_LINE_NAME);
|
||||||
@ -160,8 +160,8 @@ export const renderRGLayer = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Only update layer visibility if it has changed.
|
// Only update layer visibility if it has changed.
|
||||||
if (konvaLayer.visible() !== layerState.isEnabled) {
|
if (konvaLayer.visible() !== rg.isEnabled) {
|
||||||
konvaLayer.visible(layerState.isEnabled);
|
konvaLayer.visible(rg.isEnabled);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -173,6 +173,7 @@ export const renderRGLayer = (
|
|||||||
|
|
||||||
const compositingRect =
|
const compositingRect =
|
||||||
konvaLayer.findOne<Konva.Rect>(`.${COMPOSITING_RECT_NAME}`) ?? createCompositingRect(konvaLayer);
|
konvaLayer.findOne<Konva.Rect>(`.${COMPOSITING_RECT_NAME}`) ?? createCompositingRect(konvaLayer);
|
||||||
|
const isSelected = selectedEntity?.id === rg.id;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
* When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
||||||
@ -185,7 +186,7 @@ export const renderRGLayer = (
|
|||||||
* Instead, with the special handling, the effect is as if you drew all the shapes at 100% opacity, flattened them to
|
* Instead, with the special handling, the effect is as if you drew all the shapes at 100% opacity, flattened them to
|
||||||
* a single raster image, and _then_ applied the 50% opacity.
|
* a single raster image, and _then_ applied the 50% opacity.
|
||||||
*/
|
*/
|
||||||
if (layerState.isSelected && tool !== 'move') {
|
if (isSelected && tool !== 'move') {
|
||||||
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||||
if (konvaObjectGroup.isCached()) {
|
if (konvaObjectGroup.isCached()) {
|
||||||
konvaObjectGroup.clearCache();
|
konvaObjectGroup.clearCache();
|
||||||
@ -195,7 +196,7 @@ export const renderRGLayer = (
|
|||||||
|
|
||||||
compositingRect.setAttrs({
|
compositingRect.setAttrs({
|
||||||
// The rect should be the size of the layer - use the fast method if we don't have a pixel-perfect bbox already
|
// The rect should be the size of the layer - use the fast method if we don't have a pixel-perfect bbox already
|
||||||
...(!layerState.bboxNeedsUpdate && layerState.bbox ? layerState.bbox : getLayerBboxFast(konvaLayer)),
|
...(!rg.bboxNeedsUpdate && rg.bbox ? rg.bbox : getLayerBboxFast(konvaLayer)),
|
||||||
fill: rgbColor,
|
fill: rgbColor,
|
||||||
opacity: globalMaskLayerOpacity,
|
opacity: globalMaskLayerOpacity,
|
||||||
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||||
@ -215,18 +216,18 @@ export const renderRGLayer = (
|
|||||||
konvaObjectGroup.opacity(globalMaskLayerOpacity);
|
konvaObjectGroup.opacity(globalMaskLayerOpacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(layerState, konvaLayer);
|
const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(rg, konvaLayer);
|
||||||
|
|
||||||
if (layerState.bbox) {
|
if (rg.bbox) {
|
||||||
const active = !layerState.bboxNeedsUpdate && layerState.isSelected && tool === 'move';
|
const active = !rg.bboxNeedsUpdate && isSelected && tool === 'move';
|
||||||
bboxRect.setAttrs({
|
bboxRect.setAttrs({
|
||||||
visible: active,
|
visible: active,
|
||||||
listening: active,
|
listening: active,
|
||||||
x: layerState.bbox.x,
|
x: rg.bbox.x,
|
||||||
y: layerState.bbox.y,
|
y: rg.bbox.y,
|
||||||
width: layerState.bbox.width,
|
width: rg.bbox.width,
|
||||||
height: layerState.bbox.height,
|
height: rg.bbox.height,
|
||||||
stroke: layerState.isSelected ? BBOX_SELECTED_STROKE : '',
|
stroke: isSelected ? BBOX_SELECTED_STROKE : '',
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
bboxRect.visible(false);
|
bboxRect.visible(false);
|
||||||
|
@ -11,21 +11,12 @@ import { getIsSizeOptimal, getOptimalDimension } from 'features/parameters/util/
|
|||||||
import type { IRect, Vector2d } from 'konva/lib/types';
|
import type { IRect, Vector2d } from 'konva/lib/types';
|
||||||
import { atom } from 'nanostores';
|
import { atom } from 'nanostores';
|
||||||
|
|
||||||
import type {
|
import type { CanvasEntity, CanvasV2State, RgbaColor, StageAttrs, Tool } from './types';
|
||||||
CanvasV2State,
|
|
||||||
ControlAdapterData,
|
|
||||||
IPAdapterData,
|
|
||||||
LayerData,
|
|
||||||
RegionalGuidanceData,
|
|
||||||
RgbaColor,
|
|
||||||
StageAttrs,
|
|
||||||
Tool,
|
|
||||||
} from './types';
|
|
||||||
import { DEFAULT_RGBA_COLOR } from './types';
|
import { DEFAULT_RGBA_COLOR } from './types';
|
||||||
|
|
||||||
const initialState: CanvasV2State = {
|
const initialState: CanvasV2State = {
|
||||||
_version: 3,
|
_version: 3,
|
||||||
lastSelectedItem: null,
|
selectedEntityIdentifier: null,
|
||||||
prompts: {
|
prompts: {
|
||||||
positivePrompt: '',
|
positivePrompt: '',
|
||||||
negativePrompt: '',
|
negativePrompt: '',
|
||||||
@ -113,6 +104,12 @@ export const canvasV2Slice = createSlice({
|
|||||||
invertScrollChanged: (state, action: PayloadAction<boolean>) => {
|
invertScrollChanged: (state, action: PayloadAction<boolean>) => {
|
||||||
state.tool.invertScroll = action.payload;
|
state.tool.invertScroll = action.payload;
|
||||||
},
|
},
|
||||||
|
toolChanged: (state, action: PayloadAction<Tool>) => {
|
||||||
|
state.tool.selected = action.payload;
|
||||||
|
},
|
||||||
|
toolBufferChanged: (state, action: PayloadAction<Tool | null>) => {
|
||||||
|
state.tool.selectedBuffer = action.payload;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
extraReducers(builder) {
|
extraReducers(builder) {
|
||||||
builder.addCase(modelChanged, (state, action) => {
|
builder.addCase(modelChanged, (state, action) => {
|
||||||
@ -146,6 +143,8 @@ export const {
|
|||||||
eraserWidthChanged,
|
eraserWidthChanged,
|
||||||
fillChanged,
|
fillChanged,
|
||||||
invertScrollChanged,
|
invertScrollChanged,
|
||||||
|
toolChanged,
|
||||||
|
toolBufferChanged,
|
||||||
} = canvasV2Slice.actions;
|
} = canvasV2Slice.actions;
|
||||||
|
|
||||||
export const selectCanvasV2Slice = (state: RootState) => state.canvasV2;
|
export const selectCanvasV2Slice = (state: RootState) => state.canvasV2;
|
||||||
@ -173,18 +172,9 @@ export const $stageAttrs = atom<StageAttrs>({
|
|||||||
|
|
||||||
// Some nanostores that are manually synced to redux state to provide imperative access
|
// Some nanostores that are manually synced to redux state to provide imperative access
|
||||||
// TODO(psyche):
|
// TODO(psyche):
|
||||||
export const $tool = atom<Tool>('brush');
|
export const $toolState = atom<CanvasV2State['tool']>(deepClone(initialState.tool));
|
||||||
export const $toolBuffer = atom<Tool | null>(null);
|
export const $currentFill = atom<RgbaColor>(DEFAULT_RGBA_COLOR);
|
||||||
export const $brushWidth = atom<number>(0);
|
export const $selectedEntity = atom<CanvasEntity | null>(null);
|
||||||
export const $brushSpacingPx = atom<number>(0);
|
|
||||||
export const $eraserWidth = atom<number>(0);
|
|
||||||
export const $eraserSpacingPx = atom<number>(0);
|
|
||||||
export const $fill = atom<RgbaColor>(DEFAULT_RGBA_COLOR);
|
|
||||||
export const $selectedLayer = atom<LayerData | null>(null);
|
|
||||||
export const $selectedRG = atom<RegionalGuidanceData | null>(null);
|
|
||||||
export const $selectedCA = atom<ControlAdapterData | null>(null);
|
|
||||||
export const $selectedIPA = atom<IPAdapterData | null>(null);
|
|
||||||
export const $invertScroll = atom(false);
|
|
||||||
export const $bbox = atom<IRect>({ x: 0, y: 0, width: 0, height: 0 });
|
export const $bbox = atom<IRect>({ x: 0, y: 0, width: 0, height: 0 });
|
||||||
|
|
||||||
export const canvasV2PersistConfig: PersistConfig<CanvasV2State> = {
|
export const canvasV2PersistConfig: PersistConfig<CanvasV2State> = {
|
||||||
|
@ -7,12 +7,12 @@ import type { IRect } from 'konva/lib/types';
|
|||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
AddBrushLineArg,
|
BrushLineAddedArg,
|
||||||
AddEraserLineArg,
|
EraserLineAddedArg,
|
||||||
AddImageObjectArg,
|
ImageObjectAddedArg,
|
||||||
AddPointToLineArg,
|
|
||||||
AddRectShapeArg,
|
|
||||||
LayerData,
|
LayerData,
|
||||||
|
PointAddedToLineArg,
|
||||||
|
RectShapeAddedArg,
|
||||||
} from './types';
|
} from './types';
|
||||||
import { isLine } from './types';
|
import { isLine } from './types';
|
||||||
|
|
||||||
@ -133,7 +133,7 @@ export const layersSlice = createSlice({
|
|||||||
moveToStart(state.layers, layer);
|
moveToStart(state.layers, layer);
|
||||||
},
|
},
|
||||||
layerBrushLineAdded: {
|
layerBrushLineAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddBrushLineArg & { lineId: string }>) => {
|
reducer: (state, action: PayloadAction<BrushLineAddedArg & { lineId: string }>) => {
|
||||||
const { id, points, lineId, color, width } = action.payload;
|
const { id, points, lineId, color, width } = action.payload;
|
||||||
const layer = selectLayer(state, id);
|
const layer = selectLayer(state, id);
|
||||||
if (!layer) {
|
if (!layer) {
|
||||||
@ -149,12 +149,12 @@ export const layersSlice = createSlice({
|
|||||||
});
|
});
|
||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddBrushLineArg) => ({
|
prepare: (payload: BrushLineAddedArg) => ({
|
||||||
payload: { ...payload, lineId: uuidv4() },
|
payload: { ...payload, lineId: uuidv4() },
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
layerEraserLineAdded: {
|
layerEraserLineAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddEraserLineArg & { lineId: string }>) => {
|
reducer: (state, action: PayloadAction<EraserLineAddedArg & { lineId: string }>) => {
|
||||||
const { id, points, lineId, width } = action.payload;
|
const { id, points, lineId, width } = action.payload;
|
||||||
const layer = selectLayer(state, id);
|
const layer = selectLayer(state, id);
|
||||||
if (!layer) {
|
if (!layer) {
|
||||||
@ -169,11 +169,11 @@ export const layersSlice = createSlice({
|
|||||||
});
|
});
|
||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddEraserLineArg) => ({
|
prepare: (payload: EraserLineAddedArg) => ({
|
||||||
payload: { ...payload, lineId: uuidv4() },
|
payload: { ...payload, lineId: uuidv4() },
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
layerLinePointAdded: (state, action: PayloadAction<AddPointToLineArg>) => {
|
layerLinePointAdded: (state, action: PayloadAction<PointAddedToLineArg>) => {
|
||||||
const { id, point } = action.payload;
|
const { id, point } = action.payload;
|
||||||
const layer = selectLayer(state, id);
|
const layer = selectLayer(state, id);
|
||||||
if (!layer) {
|
if (!layer) {
|
||||||
@ -187,7 +187,7 @@ export const layersSlice = createSlice({
|
|||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
},
|
},
|
||||||
layerRectAdded: {
|
layerRectAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddRectShapeArg & { rectId: string }>) => {
|
reducer: (state, action: PayloadAction<RectShapeAddedArg & { rectId: string }>) => {
|
||||||
const { id, rect, rectId, color } = action.payload;
|
const { id, rect, rectId, color } = action.payload;
|
||||||
if (rect.height === 0 || rect.width === 0) {
|
if (rect.height === 0 || rect.width === 0) {
|
||||||
// Ignore zero-area rectangles
|
// Ignore zero-area rectangles
|
||||||
@ -200,18 +200,15 @@ export const layersSlice = createSlice({
|
|||||||
layer.objects.push({
|
layer.objects.push({
|
||||||
type: 'rect_shape',
|
type: 'rect_shape',
|
||||||
id: getRectShapeId(id, rectId),
|
id: getRectShapeId(id, rectId),
|
||||||
x: rect.x - layer.x,
|
...rect,
|
||||||
y: rect.y - layer.y,
|
|
||||||
width: rect.width,
|
|
||||||
height: rect.height,
|
|
||||||
color,
|
color,
|
||||||
});
|
});
|
||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddRectShapeArg) => ({ payload: { ...payload, rectId: uuidv4() } }),
|
prepare: (payload: RectShapeAddedArg) => ({ payload: { ...payload, rectId: uuidv4() } }),
|
||||||
},
|
},
|
||||||
layerImageAdded: {
|
layerImageAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddImageObjectArg & { imageId: string }>) => {
|
reducer: (state, action: PayloadAction<ImageObjectAddedArg & { imageId: string }>) => {
|
||||||
const { id, imageId, imageDTO } = action.payload;
|
const { id, imageId, imageDTO } = action.payload;
|
||||||
const layer = selectLayer(state, id);
|
const layer = selectLayer(state, id);
|
||||||
if (!layer) {
|
if (!layer) {
|
||||||
@ -229,7 +226,7 @@ export const layersSlice = createSlice({
|
|||||||
});
|
});
|
||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddImageObjectArg) => ({ payload: { ...payload, imageId: uuidv4() } }),
|
prepare: (payload: ImageObjectAddedArg) => ({ payload: { ...payload, imageId: uuidv4() } }),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -14,11 +14,11 @@ import { assert } from 'tsafe';
|
|||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
AddBrushLineArg,
|
BrushLineAddedArg,
|
||||||
AddEraserLineArg,
|
EraserLineAddedArg,
|
||||||
AddPointToLineArg,
|
|
||||||
AddRectShapeArg,
|
|
||||||
IPAdapterData,
|
IPAdapterData,
|
||||||
|
PointAddedToLineArg,
|
||||||
|
RectShapeAddedArg,
|
||||||
RegionalGuidanceData,
|
RegionalGuidanceData,
|
||||||
RgbColor,
|
RgbColor,
|
||||||
} from './types';
|
} from './types';
|
||||||
@ -306,7 +306,7 @@ export const regionalGuidanceSlice = createSlice({
|
|||||||
ipa.clipVisionModel = clipVisionModel;
|
ipa.clipVisionModel = clipVisionModel;
|
||||||
},
|
},
|
||||||
rgBrushLineAdded: {
|
rgBrushLineAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddBrushLineArg & { lineId: string }>) => {
|
reducer: (state, action: PayloadAction<BrushLineAddedArg & { lineId: string }>) => {
|
||||||
const { id, points, lineId, color, width } = action.payload;
|
const { id, points, lineId, color, width } = action.payload;
|
||||||
const rg = selectRg(state, id);
|
const rg = selectRg(state, id);
|
||||||
if (!rg) {
|
if (!rg) {
|
||||||
@ -315,21 +315,19 @@ export const regionalGuidanceSlice = createSlice({
|
|||||||
rg.objects.push({
|
rg.objects.push({
|
||||||
id: getBrushLineId(id, lineId),
|
id: getBrushLineId(id, lineId),
|
||||||
type: 'brush_line',
|
type: 'brush_line',
|
||||||
// Points must be offset by the layer's x and y coordinates
|
points,
|
||||||
// TODO: Handle this in the event listener?
|
|
||||||
points: [points[0] - rg.x, points[1] - rg.y, points[2] - rg.x, points[3] - rg.y],
|
|
||||||
strokeWidth: width,
|
strokeWidth: width,
|
||||||
color,
|
color,
|
||||||
});
|
});
|
||||||
rg.bboxNeedsUpdate = true;
|
rg.bboxNeedsUpdate = true;
|
||||||
rg.imageCache = null;
|
rg.imageCache = null;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddBrushLineArg) => ({
|
prepare: (payload: BrushLineAddedArg) => ({
|
||||||
payload: { ...payload, lineId: uuidv4() },
|
payload: { ...payload, lineId: uuidv4() },
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
rgEraserLineAdded: {
|
rgEraserLineAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddEraserLineArg & { lineId: string }>) => {
|
reducer: (state, action: PayloadAction<EraserLineAddedArg & { lineId: string }>) => {
|
||||||
const { id, points, lineId, width } = action.payload;
|
const { id, points, lineId, width } = action.payload;
|
||||||
const rg = selectRg(state, id);
|
const rg = selectRg(state, id);
|
||||||
if (!rg) {
|
if (!rg) {
|
||||||
@ -338,19 +336,17 @@ export const regionalGuidanceSlice = createSlice({
|
|||||||
rg.objects.push({
|
rg.objects.push({
|
||||||
id: getEraserLineId(id, lineId),
|
id: getEraserLineId(id, lineId),
|
||||||
type: 'eraser_line',
|
type: 'eraser_line',
|
||||||
// Points must be offset by the layer's x and y coordinates
|
points,
|
||||||
// TODO: Handle this in the event listener?
|
|
||||||
points: [points[0] - rg.x, points[1] - rg.y, points[2] - rg.x, points[3] - rg.y],
|
|
||||||
strokeWidth: width,
|
strokeWidth: width,
|
||||||
});
|
});
|
||||||
rg.bboxNeedsUpdate = true;
|
rg.bboxNeedsUpdate = true;
|
||||||
rg.imageCache = null;
|
rg.imageCache = null;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddEraserLineArg) => ({
|
prepare: (payload: EraserLineAddedArg) => ({
|
||||||
payload: { ...payload, lineId: uuidv4() },
|
payload: { ...payload, lineId: uuidv4() },
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
rgLinePointAdded: (state, action: PayloadAction<AddPointToLineArg>) => {
|
rgLinePointAdded: (state, action: PayloadAction<PointAddedToLineArg>) => {
|
||||||
const { id, point } = action.payload;
|
const { id, point } = action.payload;
|
||||||
const rg = selectRg(state, id);
|
const rg = selectRg(state, id);
|
||||||
if (!rg) {
|
if (!rg) {
|
||||||
@ -360,14 +356,12 @@ export const regionalGuidanceSlice = createSlice({
|
|||||||
if (!lastObject || !isLine(lastObject)) {
|
if (!lastObject || !isLine(lastObject)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// Points must be offset by the layer's x and y coordinates
|
lastObject.points.push(...point);
|
||||||
// TODO: Handle this in the event listener
|
|
||||||
lastObject.points.push(point[0] - rg.x, point[1] - rg.y);
|
|
||||||
rg.bboxNeedsUpdate = true;
|
rg.bboxNeedsUpdate = true;
|
||||||
rg.imageCache = null;
|
rg.imageCache = null;
|
||||||
},
|
},
|
||||||
rgRectAdded: {
|
rgRectAdded: {
|
||||||
reducer: (state, action: PayloadAction<AddRectShapeArg & { rectId: string }>) => {
|
reducer: (state, action: PayloadAction<RectShapeAddedArg & { rectId: string }>) => {
|
||||||
const { id, rect, rectId, color } = action.payload;
|
const { id, rect, rectId, color } = action.payload;
|
||||||
if (rect.height === 0 || rect.width === 0) {
|
if (rect.height === 0 || rect.width === 0) {
|
||||||
// Ignore zero-area rectangles
|
// Ignore zero-area rectangles
|
||||||
@ -380,16 +374,13 @@ export const regionalGuidanceSlice = createSlice({
|
|||||||
rg.objects.push({
|
rg.objects.push({
|
||||||
type: 'rect_shape',
|
type: 'rect_shape',
|
||||||
id: getRectShapeId(id, rectId),
|
id: getRectShapeId(id, rectId),
|
||||||
x: rect.x - rg.x,
|
...rect,
|
||||||
y: rect.y - rg.y,
|
|
||||||
width: rect.width,
|
|
||||||
height: rect.height,
|
|
||||||
color,
|
color,
|
||||||
});
|
});
|
||||||
rg.bboxNeedsUpdate = true;
|
rg.bboxNeedsUpdate = true;
|
||||||
rg.imageCache = null;
|
rg.imageCache = null;
|
||||||
},
|
},
|
||||||
prepare: (payload: AddRectShapeArg) => ({ payload: { ...payload, rectId: uuidv4() } }),
|
prepare: (payload: RectShapeAddedArg) => ({ payload: { ...payload, rectId: uuidv4() } }),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import { LightnessToAlphaFilter } from 'features/controlLayers/konva/filters';
|
||||||
import {
|
import {
|
||||||
zBeginEndStepPct,
|
zBeginEndStepPct,
|
||||||
zCLIPVisionModelV2,
|
zCLIPVisionModelV2,
|
||||||
@ -243,7 +244,7 @@ const zInpaintMaskData = z.object({
|
|||||||
});
|
});
|
||||||
export type InpaintMaskData = z.infer<typeof zInpaintMaskData>;
|
export type InpaintMaskData = z.infer<typeof zInpaintMaskData>;
|
||||||
|
|
||||||
const zFilter = z.enum(['none', 'lightness_to_alpha']);
|
const zFilter = z.enum(['none', LightnessToAlphaFilter.name]);
|
||||||
export type Filter = z.infer<typeof zFilter>;
|
export type Filter = z.infer<typeof zFilter>;
|
||||||
|
|
||||||
const zControlAdapterData = z.object({
|
const zControlAdapterData = z.object({
|
||||||
@ -271,21 +272,12 @@ export type ControlAdapterConfig = Pick<
|
|||||||
'weight' | 'image' | 'processedImage' | 'processorConfig' | 'beginEndStepPct' | 'model' | 'controlMode'
|
'weight' | 'image' | 'processedImage' | 'processorConfig' | 'beginEndStepPct' | 'model' | 'controlMode'
|
||||||
>;
|
>;
|
||||||
|
|
||||||
const zCanvasItemIdentifier = z.object({
|
export type CanvasEntity = LayerData | IPAdapterData | ControlAdapterData | RegionalGuidanceData | InpaintMaskData;
|
||||||
type: z.enum([
|
export type CanvasEntityIdentifier = Pick<CanvasEntity, 'id' | 'type'>;
|
||||||
zLayerData.shape.type.value,
|
|
||||||
zIPAdapterData.shape.type.value,
|
|
||||||
zControlAdapterData.shape.type.value,
|
|
||||||
zRegionalGuidanceData.shape.type.value,
|
|
||||||
zInpaintMaskData.shape.type.value,
|
|
||||||
]),
|
|
||||||
id: zId,
|
|
||||||
});
|
|
||||||
type CanvasItemIdentifier = z.infer<typeof zCanvasItemIdentifier>;
|
|
||||||
|
|
||||||
export type CanvasV2State = {
|
export type CanvasV2State = {
|
||||||
_version: 3;
|
_version: 3;
|
||||||
lastSelectedItem: CanvasItemIdentifier | null;
|
selectedEntityIdentifier: CanvasEntityIdentifier | null;
|
||||||
prompts: {
|
prompts: {
|
||||||
positivePrompt: ParameterPositivePrompt;
|
positivePrompt: ParameterPositivePrompt;
|
||||||
negativePrompt: ParameterNegativePrompt;
|
negativePrompt: ParameterNegativePrompt;
|
||||||
@ -314,11 +306,17 @@ export type CanvasV2State = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type StageAttrs = { x: number; y: number; width: number; height: number; scale: number };
|
export type StageAttrs = { x: number; y: number; width: number; height: number; scale: number };
|
||||||
export type AddEraserLineArg = { id: string; points: [number, number, number, number]; width: number };
|
export type PosChangedArg = { id: string; x: number; y: number };
|
||||||
export type AddBrushLineArg = AddEraserLineArg & { color: RgbaColor };
|
export type BboxChangedArg = { id: string; bbox: IRect };
|
||||||
export type AddPointToLineArg = { id: string; point: [number, number] };
|
export type EraserLineAddedArg = {
|
||||||
export type AddRectShapeArg = { id: string; rect: IRect; color: RgbaColor };
|
id: string;
|
||||||
export type AddImageObjectArg = { id: string; imageDTO: ImageDTO };
|
points: [number, number, number, number];
|
||||||
|
width: number;
|
||||||
|
};
|
||||||
|
export type BrushLineAddedArg = EraserLineAddedArg & { color: RgbaColor };
|
||||||
|
export type PointAddedToLineArg = { id: string; point: [number, number] };
|
||||||
|
export type RectShapeAddedArg = { id: string; rect: IRect; color: RgbaColor };
|
||||||
|
export type ImageObjectAddedArg = { id: string; imageDTO: ImageDTO };
|
||||||
|
|
||||||
//#region Type guards
|
//#region Type guards
|
||||||
export const isLine = (obj: LayerObject): obj is BrushLine | EraserLine => {
|
export const isLine = (obj: LayerObject): obj is BrushLine | EraserLine => {
|
||||||
|
@ -34,7 +34,7 @@ const selectImageUsages = createMemoizedSelector(
|
|||||||
const { imagesToDelete } = deleteImageModal;
|
const { imagesToDelete } = deleteImageModal;
|
||||||
|
|
||||||
const allImageUsage = (imagesToDelete ?? []).map(({ image_name }) =>
|
const allImageUsage = (imagesToDelete ?? []).map(({ image_name }) =>
|
||||||
getImageUsage(canvas, nodes, controlAdapters, controlLayers.present, image_name)
|
getImageUsage(canvas, nodes, controlAdapters, canvasV2, image_name)
|
||||||
);
|
);
|
||||||
|
|
||||||
const imageUsageSummary: ImageUsage = {
|
const imageUsageSummary: ImageUsage = {
|
||||||
|
@ -84,7 +84,7 @@ export const selectImageUsage = createMemoizedSelector(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const imagesUsage = imagesToDelete.map((i) =>
|
const imagesUsage = imagesToDelete.map((i) =>
|
||||||
getImageUsage(canvas, nodes, controlAdapters, controlLayers.present, i.image_name)
|
getImageUsage(canvas, nodes, controlAdapters, canvasV2, i.image_name)
|
||||||
);
|
);
|
||||||
|
|
||||||
return imagesUsage;
|
return imagesUsage;
|
||||||
|
@ -45,7 +45,7 @@ const DeleteBoardModal = (props: Props) => {
|
|||||||
[selectCanvasSlice, selectNodesSlice, selectControlAdaptersSlice, selectCanvasV2Slice],
|
[selectCanvasSlice, selectNodesSlice, selectControlAdaptersSlice, selectCanvasV2Slice],
|
||||||
(canvas, nodes, controlAdapters, controlLayers) => {
|
(canvas, nodes, controlAdapters, controlLayers) => {
|
||||||
const allImageUsage = (boardImageNames ?? []).map((imageName) =>
|
const allImageUsage = (boardImageNames ?? []).map((imageName) =>
|
||||||
getImageUsage(canvas, nodes, controlAdapters, controlLayers.present, imageName)
|
getImageUsage(canvas, nodes, controlAdapters, canvasV2, imageName)
|
||||||
);
|
);
|
||||||
|
|
||||||
const imageUsageSummary: ImageUsage = {
|
const imageUsageSummary: ImageUsage = {
|
||||||
|
@ -10,7 +10,7 @@ import { CANVAS_COHERENCE_NOISE, METADATA, NOISE, POSITIVE_CONDITIONING } from '
|
|||||||
|
|
||||||
export const prepareLinearUIBatch = (state: RootState, graph: NonNullableGraph, prepend: boolean): BatchConfig => {
|
export const prepareLinearUIBatch = (state: RootState, graph: NonNullableGraph, prepend: boolean): BatchConfig => {
|
||||||
const { iterations, model, shouldRandomizeSeed, seed } = state.generation;
|
const { iterations, model, shouldRandomizeSeed, seed } = state.generation;
|
||||||
const { shouldConcatPrompts } = state.controlLayers.present;
|
const { shouldConcatPrompts } = state.canvasV2;
|
||||||
const { prompts, seedBehaviour } = state.dynamicPrompts;
|
const { prompts, seedBehaviour } = state.dynamicPrompts;
|
||||||
|
|
||||||
const data: Batch['data'] = [];
|
const data: Batch['data'] = [];
|
||||||
|
@ -73,7 +73,7 @@ export const addControlLayers = async (
|
|||||||
): Promise<LayerData[]> => {
|
): Promise<LayerData[]> => {
|
||||||
const isSDXL = base === 'sdxl';
|
const isSDXL = base === 'sdxl';
|
||||||
|
|
||||||
const validLayers = state.controlLayers.present.layers.filter((l) => isValidLayer(l, base));
|
const validLayers = state.canvasV2.layers.filter((l) => isValidLayer(l, base));
|
||||||
|
|
||||||
const validControlAdapters = validLayers.filter(isControlAdapterLayer).map((l) => l.controlAdapter);
|
const validControlAdapters = validLayers.filter(isControlAdapterLayer).map((l) => l.controlAdapter);
|
||||||
for (const ca of validControlAdapters) {
|
for (const ca of validControlAdapters) {
|
||||||
@ -259,7 +259,7 @@ export const addControlLayers = async (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
g.upsertMetadata({ control_layers: { layers: validLayers, version: state.controlLayers.present._version } });
|
g.upsertMetadata({ control_layers: { layers: validLayers, version: state.canvasV2._version } });
|
||||||
return validLayers;
|
return validLayers;
|
||||||
};
|
};
|
||||||
//#endregion
|
//#endregion
|
||||||
@ -421,7 +421,7 @@ const addInitialImageLayerToGraph = (
|
|||||||
) => {
|
) => {
|
||||||
const { vaePrecision } = state.generation;
|
const { vaePrecision } = state.generation;
|
||||||
const { refinerModel, refinerStart } = state.sdxl;
|
const { refinerModel, refinerStart } = state.sdxl;
|
||||||
const { width, height } = state.controlLayers.present.size;
|
const { width, height } = state.canvasV2.size;
|
||||||
assert(layer.isEnabled, 'Initial image layer is not enabled');
|
assert(layer.isEnabled, 'Initial image layer is not enabled');
|
||||||
assert(layer.image, 'Initial image layer has no image');
|
assert(layer.image, 'Initial image layer has no image');
|
||||||
|
|
||||||
@ -567,8 +567,8 @@ const buildControlImage = (
|
|||||||
*/
|
*/
|
||||||
const getRGLayerBlobs = async (layerIds?: string[], preview: boolean = false): Promise<Record<string, Blob>> => {
|
const getRGLayerBlobs = async (layerIds?: string[], preview: boolean = false): Promise<Record<string, Blob>> => {
|
||||||
const state = getStore().getState();
|
const state = getStore().getState();
|
||||||
const { layers } = state.controlLayers.present;
|
const { layers } = state.canvasV2;
|
||||||
const { width, height } = state.controlLayers.present.size;
|
const { width, height } = state.canvasV2.size;
|
||||||
const reduxLayers = layers.filter(isRegionalGuidanceLayer);
|
const reduxLayers = layers.filter(isRegionalGuidanceLayer);
|
||||||
const container = document.createElement('div');
|
const container = document.createElement('div');
|
||||||
const stage = new Konva.Stage({ container, width, height });
|
const stage = new Konva.Stage({ container, width, height });
|
||||||
|
@ -74,7 +74,7 @@ export const addHRF = (
|
|||||||
vaeSource: Invocation<'vae_loader'> | Invocation<'main_model_loader'> | Invocation<'seamless'>
|
vaeSource: Invocation<'vae_loader'> | Invocation<'main_model_loader'> | Invocation<'seamless'>
|
||||||
): Invocation<'l2i'> => {
|
): Invocation<'l2i'> => {
|
||||||
const { hrfStrength, hrfEnabled, hrfMethod } = state.hrf;
|
const { hrfStrength, hrfEnabled, hrfMethod } = state.hrf;
|
||||||
const { width, height } = state.controlLayers.present.size;
|
const { width, height } = state.canvasV2.size;
|
||||||
const optimalDimension = selectOptimalDimension(state);
|
const optimalDimension = selectOptimalDimension(state);
|
||||||
const { newWidth: hrfWidth, newHeight: hrfHeight } = calculateHrfRes(optimalDimension, width, height);
|
const { newWidth: hrfWidth, newHeight: hrfHeight } = calculateHrfRes(optimalDimension, width, height);
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@ export const getPresetModifiedPrompts = (
|
|||||||
state: RootState
|
state: RootState
|
||||||
): { positivePrompt: string; negativePrompt: string; positiveStylePrompt?: string; negativeStylePrompt?: string } => {
|
): { positivePrompt: string; negativePrompt: string; positiveStylePrompt?: string; negativeStylePrompt?: string } => {
|
||||||
const { positivePrompt, negativePrompt, positivePrompt2, negativePrompt2, shouldConcatPrompts } =
|
const { positivePrompt, negativePrompt, positivePrompt2, negativePrompt2, shouldConcatPrompts } =
|
||||||
state.controlLayers.present;
|
state.canvasV2.prompts;
|
||||||
const { activeStylePresetId } = state.stylePreset;
|
const { activeStylePresetId } = state.stylePreset;
|
||||||
|
|
||||||
if (activeStylePresetId) {
|
if (activeStylePresetId) {
|
||||||
|
@ -13,7 +13,7 @@ import { useListStylePresetsQuery } from 'services/api/endpoints/stylePresets';
|
|||||||
|
|
||||||
export const ParamNegativePrompt = memo(() => {
|
export const ParamNegativePrompt = memo(() => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const prompt = useAppSelector((s) => s.controlLayers.present.negativePrompt);
|
const prompt = useAppSelector((s) => s.canvasV2.prompts.negativePrompt);
|
||||||
const viewMode = useAppSelector((s) => s.stylePreset.viewMode);
|
const viewMode = useAppSelector((s) => s.stylePreset.viewMode);
|
||||||
const activeStylePresetId = useAppSelector((s) => s.stylePreset.activeStylePresetId);
|
const activeStylePresetId = useAppSelector((s) => s.stylePreset.activeStylePresetId);
|
||||||
|
|
||||||
|
@ -17,7 +17,7 @@ import { useListStylePresetsQuery } from 'services/api/endpoints/stylePresets';
|
|||||||
|
|
||||||
export const ParamPositivePrompt = memo(() => {
|
export const ParamPositivePrompt = memo(() => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const prompt = useAppSelector((s) => s.controlLayers.present.positivePrompt);
|
const prompt = useAppSelector((s) => s.canvasV2.positivePrompt);
|
||||||
const baseModel = useAppSelector((s) => s.generation.model)?.base;
|
const baseModel = useAppSelector((s) => s.generation.model)?.base;
|
||||||
const viewMode = useAppSelector((s) => s.stylePreset.viewMode);
|
const viewMode = useAppSelector((s) => s.stylePreset.viewMode);
|
||||||
const activeStylePresetId = useAppSelector((s) => s.stylePreset.activeStylePresetId);
|
const activeStylePresetId = useAppSelector((s) => s.stylePreset.activeStylePresetId);
|
||||||
|
@ -15,7 +15,7 @@ const selectPromptsCount = createSelector(
|
|||||||
selectCanvasV2Slice,
|
selectCanvasV2Slice,
|
||||||
selectDynamicPromptsSlice,
|
selectDynamicPromptsSlice,
|
||||||
(controlLayers, dynamicPrompts) =>
|
(controlLayers, dynamicPrompts) =>
|
||||||
getShouldProcessPrompt(controlLayers.present.positivePrompt) ? dynamicPrompts.prompts.length : 1
|
getShouldProcessPrompt(canvasV2.positivePrompt) ? dynamicPrompts.prompts.length : 1
|
||||||
);
|
);
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
|
@ -12,7 +12,7 @@ import { useTranslation } from 'react-i18next';
|
|||||||
|
|
||||||
export const ParamSDXLNegativeStylePrompt = memo(() => {
|
export const ParamSDXLNegativeStylePrompt = memo(() => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const prompt = useAppSelector((s) => s.controlLayers.present.negativePrompt2);
|
const prompt = useAppSelector((s) => s.canvasV2.negativePrompt2);
|
||||||
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const handleChange = useCallback(
|
const handleChange = useCallback(
|
||||||
|
@ -11,7 +11,7 @@ import { useTranslation } from 'react-i18next';
|
|||||||
|
|
||||||
export const ParamSDXLPositiveStylePrompt = memo(() => {
|
export const ParamSDXLPositiveStylePrompt = memo(() => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const prompt = useAppSelector((s) => s.controlLayers.present.positivePrompt2);
|
const prompt = useAppSelector((s) => s.canvasV2.positivePrompt2);
|
||||||
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const handleChange = useCallback(
|
const handleChange = useCallback(
|
||||||
|
@ -6,7 +6,7 @@ import { useTranslation } from 'react-i18next';
|
|||||||
import { PiLinkSimpleBold, PiLinkSimpleBreakBold } from 'react-icons/pi';
|
import { PiLinkSimpleBold, PiLinkSimpleBreakBold } from 'react-icons/pi';
|
||||||
|
|
||||||
export const SDXLConcatButton = memo(() => {
|
export const SDXLConcatButton = memo(() => {
|
||||||
const shouldConcatPrompts = useAppSelector((s) => s.controlLayers.present.shouldConcatPrompts);
|
const shouldConcatPrompts = useAppSelector((s) => s.canvasV2.shouldConcatPrompts);
|
||||||
|
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
@ -42,7 +42,7 @@ const selector = createMemoizedSelector(
|
|||||||
badges.push('locked');
|
badges.push('locked');
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const { aspectRatio, width, height } = controlLayers.present.size;
|
const { aspectRatio, width, height } = canvasV2.size;
|
||||||
badges.push(`${width}×${height}`);
|
badges.push(`${width}×${height}`);
|
||||||
badges.push(aspectRatio.id);
|
badges.push(aspectRatio.id);
|
||||||
if (aspectRatio.isLocked) {
|
if (aspectRatio.isLocked) {
|
||||||
|
@ -9,9 +9,9 @@ import { memo, useCallback } from 'react';
|
|||||||
|
|
||||||
export const ImageSizeLinear = memo(() => {
|
export const ImageSizeLinear = memo(() => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const width = useAppSelector((s) => s.controlLayers.present.size.width);
|
const width = useAppSelector((s) => s.canvasV2.size.width);
|
||||||
const height = useAppSelector((s) => s.controlLayers.present.size.height);
|
const height = useAppSelector((s) => s.canvasV2.size.height);
|
||||||
const aspectRatioState = useAppSelector((s) => s.controlLayers.present.size.aspectRatio);
|
const aspectRatioState = useAppSelector((s) => s.canvasV2.size.aspectRatio);
|
||||||
|
|
||||||
const onChangeWidth = useCallback(
|
const onChangeWidth = useCallback(
|
||||||
(width: number) => {
|
(width: number) => {
|
||||||
|
@ -10,7 +10,7 @@ export const buildPresetModifiedPrompt = (presetPrompt: string, currentPrompt: s
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const usePresetModifiedPrompts = () => {
|
export const usePresetModifiedPrompts = () => {
|
||||||
const { positivePrompt, negativePrompt } = useAppSelector((s) => s.controlLayers.present);
|
const { positivePrompt, negativePrompt } = useAppSelector((s) => s.canvasV2.prompts);
|
||||||
|
|
||||||
const activeStylePresetId = useAppSelector((s) => s.stylePreset.activeStylePresetId);
|
const activeStylePresetId = useAppSelector((s) => s.stylePreset.activeStylePresetId);
|
||||||
|
|
||||||
|
@ -44,7 +44,7 @@ const ParametersPanelTextToImage = () => {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const activeTabName = useAppSelector(activeTabNameSelector);
|
const activeTabName = useAppSelector(activeTabNameSelector);
|
||||||
const controlLayersCount = useAppSelector((s) => s.controlLayers.present.layers.length);
|
const controlLayersCount = useAppSelector((s) => s.canvasV2.layers.length);
|
||||||
const controlLayersTitle = useMemo(() => {
|
const controlLayersTitle = useMemo(() => {
|
||||||
if (controlLayersCount === 0) {
|
if (controlLayersCount === 0) {
|
||||||
return t('controlLayers.controlLayers');
|
return t('controlLayers.controlLayers');
|
||||||
|
Loading…
Reference in New Issue
Block a user