feat(ui): add canvas background style

This commit is contained in:
psychedelicious 2024-08-16 20:59:32 +10:00
parent 76124ea35b
commit 0670e6b53a
9 changed files with 101 additions and 1 deletions

View File

@ -1721,6 +1721,12 @@
"view": "View", "view": "View",
"transform": "Transform", "transform": "Transform",
"eyeDropper": "Eye Dropper" "eyeDropper": "Eye Dropper"
},
"background": {
"backgroundStyle": "Background Style",
"solid": "Solid",
"checkerboard": "Checkerboard",
"dynamicGrid": "Dynamic Grid"
} }
}, },
"upscaling": { "upscaling": {

View File

@ -0,0 +1,50 @@
import type { ComboboxOnChange, ComboboxOption } from '@invoke-ai/ui-library';
import { Combobox, FormControl, FormLabel } from '@invoke-ai/ui-library';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { canvasBackgroundStyleChanged } from 'features/controlLayers/store/canvasV2Slice';
import { isCanvasBackgroundStyle } from 'features/controlLayers/store/types';
import { memo, useCallback, useMemo } from 'react';
import { useTranslation } from 'react-i18next';
export const CanvasSettingsBackgroundStyle = memo(() => {
const { t } = useTranslation();
const dispatch = useAppDispatch();
const canvasBackgroundStyle = useAppSelector((s) => s.canvasV2.settings.canvasBackgroundStyle);
const onChange = useCallback<ComboboxOnChange>(
(v) => {
if (!isCanvasBackgroundStyle(v?.value)) {
return;
}
dispatch(canvasBackgroundStyleChanged(v.value));
},
[dispatch]
);
const options = useMemo<ComboboxOption[]>(() => {
return [
{
value: 'solid',
label: t('controlLayers.background.solid'),
},
{
value: 'checkerboard',
label: t('controlLayers.background.checkerboard'),
},
{
value: 'dynamicGrid',
label: t('controlLayers.background.dynamicGrid'),
},
];
}, [t]);
const value = useMemo(() => options.find((o) => o.value === canvasBackgroundStyle), [options, canvasBackgroundStyle]);
return (
<FormControl orientation="vertical">
<FormLabel m={0}>{t('controlLayers.background.backgroundStyle')}</FormLabel>
<Combobox value={value} options={options} onChange={onChange} isSearchable={false} />
</FormControl>
);
});
CanvasSettingsBackgroundStyle.displayName = 'CanvasSettingsBackgroundStyle';

View File

@ -12,6 +12,7 @@ import {
} from '@invoke-ai/ui-library'; } from '@invoke-ai/ui-library';
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { CanvasSettingsBackgroundStyle } from 'features/controlLayers/components/CanvasSettingsBackgroundStyle';
import { $canvasManager } from 'features/controlLayers/konva/CanvasManager'; import { $canvasManager } from 'features/controlLayers/konva/CanvasManager';
import { import {
clipToBboxChanged, clipToBboxChanged,
@ -71,6 +72,7 @@ const ControlLayersSettingsPopover = () => {
<FormLabel flexGrow={1}>{t('unifiedCanvas.clipToBbox')}</FormLabel> <FormLabel flexGrow={1}>{t('unifiedCanvas.clipToBbox')}</FormLabel>
<Checkbox isChecked={clipToBbox} onChange={onChangeClipToBbox} /> <Checkbox isChecked={clipToBbox} onChange={onChangeClipToBbox} />
</FormControl> </FormControl>
<CanvasSettingsBackgroundStyle />
<Button onClick={invalidateRasterizationCaches} size="sm"> <Button onClick={invalidateRasterizationCaches} size="sm">
Invalidate Rasterization Caches Invalidate Rasterization Caches
</Button> </Button>

View File

@ -3,8 +3,10 @@ import { useStore } from '@nanostores/react';
import { $socket } from 'app/hooks/useSocketIO'; import { $socket } from 'app/hooks/useSocketIO';
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { useAppStore } from 'app/store/nanostores/store'; import { useAppStore } from 'app/store/nanostores/store';
import { useAppSelector } from 'app/store/storeHooks';
import { HeadsUpDisplay } from 'features/controlLayers/components/HeadsUpDisplay'; import { HeadsUpDisplay } from 'features/controlLayers/components/HeadsUpDisplay';
import { $canvasManager, CanvasManager } from 'features/controlLayers/konva/CanvasManager'; import { $canvasManager, CanvasManager } from 'features/controlLayers/konva/CanvasManager';
import { TRANSPARENCY_CHECKER_PATTERN } from 'features/controlLayers/konva/constants';
import Konva from 'konva'; import Konva from 'konva';
import { memo, useCallback, useEffect, useLayoutEffect, useState } from 'react'; import { memo, useCallback, useEffect, useLayoutEffect, useState } from 'react';
import { useDevicePixelRatio } from 'use-device-pixel-ratio'; import { useDevicePixelRatio } from 'use-device-pixel-ratio';
@ -52,6 +54,8 @@ type Props = {
}; };
export const StageComponent = memo(({ asPreview = false }: Props) => { export const StageComponent = memo(({ asPreview = false }: Props) => {
const canvasBackgroundStyle = useAppSelector((s) => s.canvasV2.settings.canvasBackgroundStyle);
const [stage] = useState( const [stage] = useState(
() => () =>
new Konva.Stage({ new Konva.Stage({
@ -77,6 +81,17 @@ export const StageComponent = memo(({ asPreview = false }: Props) => {
return ( return (
<Flex position="relative" w="full" h="full"> <Flex position="relative" w="full" h="full">
{canvasBackgroundStyle === 'checkerboard' && (
<Flex
position="absolute"
bgImage={TRANSPARENCY_CHECKER_PATTERN}
top={0}
right={0}
bottom={0}
left={0}
opacity={0.1}
/>
)}
<Flex <Flex
position="absolute" position="absolute"
top={0} top={0}

View File

@ -34,6 +34,15 @@ export class CanvasBackground {
} }
render() { render() {
const settings = this.manager.stateApi.getSettings();
if (settings.canvasBackgroundStyle !== 'dynamicGrid') {
this.konva.layer.visible(false);
return;
}
this.konva.layer.visible(true);
this.konva.layer.zIndex(0); this.konva.layer.zIndex(0);
const scale = this.manager.stage.scaleX(); const scale = this.manager.stage.scaleX();
const gridSpacing = CanvasBackground.getGridSpacing(scale); const gridSpacing = CanvasBackground.getGridSpacing(scale);

View File

@ -270,6 +270,13 @@ export class CanvasManager {
return; return;
} }
if (
this._isFirstRender ||
state.settings.canvasBackgroundStyle !== this._prevState.settings.canvasBackgroundStyle
) {
this.background.render();
}
if (this._isFirstRender || state.rasterLayers.entities !== this._prevState.rasterLayers.entities) { if (this._isFirstRender || state.rasterLayers.entities !== this._prevState.rasterLayers.entities) {
this.log.debug('Rendering raster layers'); this.log.debug('Rendering raster layers');

View File

@ -94,6 +94,7 @@ const initialState: CanvasV2State = {
showHUD: true, showHUD: true,
clipToBbox: false, clipToBbox: false,
cropToBboxOnSave: false, cropToBboxOnSave: false,
canvasBackgroundStyle: 'dynamicGrid',
}, },
compositing: { compositing: {
maskBlur: 16, maskBlur: 16,
@ -474,6 +475,7 @@ export const {
clipToBboxChanged, clipToBboxChanged,
canvasReset, canvasReset,
rasterizationCachesInvalidated, rasterizationCachesInvalidated,
canvasBackgroundStyleChanged,
// All entities // All entities
entitySelected, entitySelected,
entityNameChanged, entityNameChanged,

View File

@ -1,8 +1,11 @@
import type { PayloadAction, SliceCaseReducers } from '@reduxjs/toolkit'; import type { PayloadAction, SliceCaseReducers } from '@reduxjs/toolkit';
import type { CanvasV2State } from 'features/controlLayers/store/types'; import type { CanvasBackgroundStyle, CanvasV2State } from 'features/controlLayers/store/types';
export const settingsReducers = { export const settingsReducers = {
clipToBboxChanged: (state, action: PayloadAction<boolean>) => { clipToBboxChanged: (state, action: PayloadAction<boolean>) => {
state.settings.clipToBbox = action.payload; state.settings.clipToBbox = action.payload;
}, },
canvasBackgroundStyleChanged: (state, action: PayloadAction<CanvasBackgroundStyle>) => {
state.settings.canvasBackgroundStyle = action.payload;
},
} satisfies SliceCaseReducers<CanvasV2State>; } satisfies SliceCaseReducers<CanvasV2State>;

View File

@ -839,6 +839,11 @@ export type StagingAreaImage = {
offsetY: number; offsetY: number;
}; };
const zCanvasBackgroundStyle = z.enum(['checkerboard', 'dynamicGrid', 'solid']);
export type CanvasBackgroundStyle = z.infer<typeof zCanvasBackgroundStyle>;
export const isCanvasBackgroundStyle = (v: unknown): v is CanvasBackgroundStyle =>
zCanvasBackgroundStyle.safeParse(v).success;
export type CanvasV2State = { export type CanvasV2State = {
_version: 3; _version: 3;
selectedEntityIdentifier: CanvasEntityIdentifier | null; selectedEntityIdentifier: CanvasEntityIdentifier | null;
@ -863,6 +868,7 @@ export type CanvasV2State = {
preserveMaskedArea: boolean; preserveMaskedArea: boolean;
cropToBboxOnSave: boolean; cropToBboxOnSave: boolean;
clipToBbox: boolean; clipToBbox: boolean;
canvasBackgroundStyle: CanvasBackgroundStyle;
}; };
bbox: { bbox: {
rect: { rect: {