diff --git a/invokeai/app/invocations/metadata.py b/invokeai/app/invocations/metadata.py index 54e385feb3..cecc01096a 100644 --- a/invokeai/app/invocations/metadata.py +++ b/invokeai/app/invocations/metadata.py @@ -156,6 +156,18 @@ class MetadataAccumulatorInvocation(BaseInvocation): description="The VAE used for decoding, if the main model's default was not used", ) + # High resolution fix metadata. + hrf_width: Optional[int] = InputField( + description="The high resolution fix height and width multipler.", + ) + hrf_height: Optional[int] = InputField( + description="The high resolution fix height and width multipler.", + ) + hrf_strength: Optional[float] = InputField( + default=None, + description="The high resolution fix img2img strength used in the upscale pass.", + ) + # SDXL positive_style_prompt: Optional[str] = InputField( default=None, diff --git a/invokeai/frontend/web/public/locales/en.json b/invokeai/frontend/web/public/locales/en.json index e0baefcf9e..dd96193d5d 100644 --- a/invokeai/frontend/web/public/locales/en.json +++ b/invokeai/frontend/web/public/locales/en.json @@ -559,8 +559,10 @@ "negativePrompt": "Negative Prompt", "noImageDetails": "No image details found", "noMetaData": "No metadata found", + "noRecallParameters": "No parameters to recall found", "perlin": "Perlin Noise", "positivePrompt": "Positive Prompt", + "recallParameters": "Recall Parameters", "scheduler": "Scheduler", "seamless": "Seamless", "seed": "Seed", diff --git a/invokeai/frontend/web/src/app/types/invokeai.ts b/invokeai/frontend/web/src/app/types/invokeai.ts index 8331e9c979..39e4ffd27a 100644 --- a/invokeai/frontend/web/src/app/types/invokeai.ts +++ b/invokeai/frontend/web/src/app/types/invokeai.ts @@ -39,7 +39,8 @@ export type SDFeature = | 'hires' | 'lora' | 'embedding' - | 'vae'; + | 'vae' + | 'hrf'; /** * Configuration options for the InvokeAI UI. @@ -110,6 +111,14 @@ export type AppConfig = { fineStep: number; coarseStep: number; }; + hrfStrength: { + initial: number; + min: number; + sliderMax: number; + inputMax: number; + fineStep: number; + coarseStep: number; + }; dynamicPrompts: { maxPrompts: { initial: number; diff --git a/invokeai/frontend/web/src/common/components/IAIInformationalPopover/constants.ts b/invokeai/frontend/web/src/common/components/IAIInformationalPopover/constants.ts index f2398483bf..197f5f4068 100644 --- a/invokeai/frontend/web/src/common/components/IAIInformationalPopover/constants.ts +++ b/invokeai/frontend/web/src/common/components/IAIInformationalPopover/constants.ts @@ -2,6 +2,7 @@ import { PopoverProps } from '@chakra-ui/react'; export type Feature = | 'clipSkip' + | 'hrf' | 'paramNegativeConditioning' | 'paramPositiveConditioning' | 'paramScheduler' diff --git a/invokeai/frontend/web/src/features/gallery/components/ImageMetadataViewer/ImageMetadataViewer.tsx b/invokeai/frontend/web/src/features/gallery/components/ImageMetadataViewer/ImageMetadataViewer.tsx index 8c7e69b2b1..e9cb3ffcaf 100644 --- a/invokeai/frontend/web/src/features/gallery/components/ImageMetadataViewer/ImageMetadataViewer.tsx +++ b/invokeai/frontend/web/src/features/gallery/components/ImageMetadataViewer/ImageMetadataViewer.tsx @@ -18,6 +18,7 @@ import ImageMetadataActions from './ImageMetadataActions'; import { useAppSelector } from '../../../../app/store/storeHooks'; import { configSelector } from '../../../system/store/configSelectors'; import { useTranslation } from 'react-i18next'; +import ScrollableContent from 'features/nodes/components/sidePanel/ScrollableContent'; type ImageMetadataViewerProps = { image: ImageDTO; @@ -65,19 +66,32 @@ const ImageMetadataViewer = ({ image }: ImageMetadataViewerProps) => { - - + {t('metadata.recallParameters')} {t('metadata.metadata')} {t('metadata.imageDetails')} {t('metadata.workflow')} + + {metadata ? ( + + + + ) : ( + + )} + {metadata ? ( diff --git a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addHrfToGraph.ts b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addHrfToGraph.ts new file mode 100644 index 0000000000..4b4a8a8a03 --- /dev/null +++ b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addHrfToGraph.ts @@ -0,0 +1,247 @@ +import { RootState } from 'app/store/store'; +import { NonNullableGraph } from 'features/nodes/types/types'; +import { + DenoiseLatentsInvocation, + ResizeLatentsInvocation, + NoiseInvocation, + LatentsToImageInvocation, + Edge, +} from 'services/api/types'; +import { + LATENTS_TO_IMAGE, + DENOISE_LATENTS, + NOISE, + MAIN_MODEL_LOADER, + METADATA_ACCUMULATOR, + LATENTS_TO_IMAGE_HRF, + DENOISE_LATENTS_HRF, + RESCALE_LATENTS, + NOISE_HRF, + VAE_LOADER, +} from './constants'; +import { logger } from 'app/logging/logger'; + +// Copy certain connections from previous DENOISE_LATENTS to new DENOISE_LATENTS_HRF. +function copyConnectionsToDenoiseLatentsHrf(graph: NonNullableGraph): void { + const destinationFields = [ + 'vae', + 'control', + 'ip_adapter', + 'metadata', + 'unet', + 'positive_conditioning', + 'negative_conditioning', + ]; + const newEdges: Edge[] = []; + + // Loop through the existing edges connected to DENOISE_LATENTS + graph.edges.forEach((edge: Edge) => { + if ( + edge.destination.node_id === DENOISE_LATENTS && + destinationFields.includes(edge.destination.field) + ) { + // Add a similar connection to DENOISE_LATENTS_HRF + newEdges.push({ + source: { + node_id: edge.source.node_id, + field: edge.source.field, + }, + destination: { + node_id: DENOISE_LATENTS_HRF, + field: edge.destination.field, + }, + }); + } + }); + graph.edges = graph.edges.concat(newEdges); +} + +// Adds the high-res fix feature to the given graph. +export const addHrfToGraph = ( + state: RootState, + graph: NonNullableGraph +): void => { + // Double check hrf is enabled. + if ( + !state.generation.hrfEnabled || + state.config.disabledSDFeatures.includes('hrf') || + state.generation.model?.model_type === 'onnx' // TODO: ONNX support + ) { + return; + } + const log = logger('txt2img'); + + const { vae } = state.generation; + const isAutoVae = !vae; + const hrfWidth = state.generation.hrfWidth; + const hrfHeight = state.generation.hrfHeight; + + // Pre-existing (original) graph nodes. + const originalDenoiseLatentsNode = graph.nodes[DENOISE_LATENTS] as + | DenoiseLatentsInvocation + | undefined; + const originalNoiseNode = graph.nodes[NOISE] as NoiseInvocation | undefined; + // Original latents to image should pick this up. + const originalLatentsToImageNode = graph.nodes[LATENTS_TO_IMAGE] as + | LatentsToImageInvocation + | undefined; + // Check if originalDenoiseLatentsNode is undefined and log an error + if (!originalDenoiseLatentsNode) { + log.error('originalDenoiseLatentsNode is undefined'); + return; + } + // Check if originalNoiseNode is undefined and log an error + if (!originalNoiseNode) { + log.error('originalNoiseNode is undefined'); + return; + } + + // Check if originalLatentsToImageNode is undefined and log an error + if (!originalLatentsToImageNode) { + log.error('originalLatentsToImageNode is undefined'); + return; + } + // Change height and width of original noise node to initial resolution. + if (originalNoiseNode) { + originalNoiseNode.width = hrfWidth; + originalNoiseNode.height = hrfHeight; + } + + // Define new nodes. + // Denoise latents node to be run on upscaled latents. + const denoiseLatentsHrfNode: DenoiseLatentsInvocation = { + type: 'denoise_latents', + id: DENOISE_LATENTS_HRF, + is_intermediate: originalDenoiseLatentsNode?.is_intermediate, + cfg_scale: originalDenoiseLatentsNode?.cfg_scale, + scheduler: originalDenoiseLatentsNode?.scheduler, + steps: originalDenoiseLatentsNode?.steps, + denoising_start: 1 - state.generation.hrfStrength, + denoising_end: 1, + }; + + // New base resolution noise node. + const hrfNoiseNode: NoiseInvocation = { + type: 'noise', + id: NOISE_HRF, + seed: originalNoiseNode?.seed, + use_cpu: originalNoiseNode?.use_cpu, + is_intermediate: originalNoiseNode?.is_intermediate, + }; + + const rescaleLatentsNode: ResizeLatentsInvocation = { + id: RESCALE_LATENTS, + type: 'lresize', + width: state.generation.width, + height: state.generation.height, + }; + + // New node to convert latents to image. + const latentsToImageHrfNode: LatentsToImageInvocation | undefined = + originalLatentsToImageNode + ? { + type: 'l2i', + id: LATENTS_TO_IMAGE_HRF, + fp32: originalLatentsToImageNode?.fp32, + is_intermediate: originalLatentsToImageNode?.is_intermediate, + } + : undefined; + + // Add new nodes to graph. + graph.nodes[LATENTS_TO_IMAGE_HRF] = + latentsToImageHrfNode as LatentsToImageInvocation; + graph.nodes[DENOISE_LATENTS_HRF] = + denoiseLatentsHrfNode as DenoiseLatentsInvocation; + graph.nodes[NOISE_HRF] = hrfNoiseNode as NoiseInvocation; + graph.nodes[RESCALE_LATENTS] = rescaleLatentsNode as ResizeLatentsInvocation; + + // Connect nodes. + graph.edges.push( + { + // Set up rescale latents. + source: { + node_id: DENOISE_LATENTS, + field: 'latents', + }, + destination: { + node_id: RESCALE_LATENTS, + field: 'latents', + }, + }, + // Set up new noise node + { + source: { + node_id: RESCALE_LATENTS, + field: 'height', + }, + destination: { + node_id: NOISE_HRF, + field: 'height', + }, + }, + { + source: { + node_id: RESCALE_LATENTS, + field: 'width', + }, + destination: { + node_id: NOISE_HRF, + field: 'width', + }, + }, + // Set up new denoise node. + { + source: { + node_id: RESCALE_LATENTS, + field: 'latents', + }, + destination: { + node_id: DENOISE_LATENTS_HRF, + field: 'latents', + }, + }, + { + source: { + node_id: NOISE_HRF, + field: 'noise', + }, + destination: { + node_id: DENOISE_LATENTS_HRF, + field: 'noise', + }, + }, + // Set up new latents to image node. + { + source: { + node_id: DENOISE_LATENTS_HRF, + field: 'latents', + }, + destination: { + node_id: LATENTS_TO_IMAGE_HRF, + field: 'latents', + }, + }, + { + source: { + node_id: METADATA_ACCUMULATOR, + field: 'metadata', + }, + destination: { + node_id: LATENTS_TO_IMAGE_HRF, + field: 'metadata', + }, + }, + { + source: { + node_id: isAutoVae ? MAIN_MODEL_LOADER : VAE_LOADER, + field: 'vae', + }, + destination: { + node_id: LATENTS_TO_IMAGE_HRF, + field: 'vae', + }, + } + ); + + copyConnectionsToDenoiseLatentsHrf(graph); +}; diff --git a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addSaveImageNode.ts b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addSaveImageNode.ts index 738c69faff..d5a6addf8a 100644 --- a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addSaveImageNode.ts +++ b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/addSaveImageNode.ts @@ -2,6 +2,7 @@ import { NonNullableGraph } from 'features/nodes/types/types'; import { CANVAS_OUTPUT, LATENTS_TO_IMAGE, + LATENTS_TO_IMAGE_HRF, METADATA_ACCUMULATOR, NSFW_CHECKER, SAVE_IMAGE, @@ -82,6 +83,14 @@ export const addSaveImageNode = ( }, destination, }); + } else if (LATENTS_TO_IMAGE_HRF in graph.nodes) { + graph.edges.push({ + source: { + node_id: LATENTS_TO_IMAGE_HRF, + field: 'image', + }, + destination, + }); } else if (LATENTS_TO_IMAGE in graph.nodes) { graph.edges.push({ source: { diff --git a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/buildLinearTextToImageGraph.ts b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/buildLinearTextToImageGraph.ts index c3c7fcdf1b..e692e12fa4 100644 --- a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/buildLinearTextToImageGraph.ts +++ b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/buildLinearTextToImageGraph.ts @@ -6,6 +6,7 @@ import { ONNXTextToLatentsInvocation, } from 'services/api/types'; import { addControlNetToLinearGraph } from './addControlNetToLinearGraph'; +import { addHrfToGraph } from './addHrfToGraph'; import { addIPAdapterToLinearGraph } from './addIPAdapterToLinearGraph'; import { addLoRAsToGraph } from './addLoRAsToGraph'; import { addNSFWCheckerToGraph } from './addNSFWCheckerToGraph'; @@ -47,6 +48,10 @@ export const buildLinearTextToImageGraph = ( seamlessXAxis, seamlessYAxis, seed, + hrfWidth, + hrfHeight, + hrfStrength, + hrfEnabled: hrfEnabled, } = state.generation; const use_cpu = shouldUseCpuNoise; @@ -254,6 +259,9 @@ export const buildLinearTextToImageGraph = ( ipAdapters: [], // populated in addIPAdapterToLinearGraph t2iAdapters: [], // populated in addT2IAdapterToLinearGraph clip_skip: clipSkip, + hrf_width: hrfEnabled ? hrfWidth : undefined, + hrf_height: hrfEnabled ? hrfHeight : undefined, + hrf_strength: hrfEnabled ? hrfStrength : undefined, }; graph.edges.push({ @@ -287,6 +295,12 @@ export const buildLinearTextToImageGraph = ( addT2IAdaptersToLinearGraph(state, graph, DENOISE_LATENTS); + // High resolution fix. + // NOTE: Not supported for onnx models. + if (state.generation.hrfEnabled && !isUsingOnnxModel) { + addHrfToGraph(state, graph); + } + // NSFW & watermark - must be last thing added to graph if (state.system.shouldUseNSFWChecker) { // must add before watermarker! diff --git a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/constants.ts b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/constants.ts index e538094731..7d547d09e6 100644 --- a/invokeai/frontend/web/src/features/nodes/util/graphBuilders/constants.ts +++ b/invokeai/frontend/web/src/features/nodes/util/graphBuilders/constants.ts @@ -2,11 +2,14 @@ export const POSITIVE_CONDITIONING = 'positive_conditioning'; export const NEGATIVE_CONDITIONING = 'negative_conditioning'; export const DENOISE_LATENTS = 'denoise_latents'; +export const DENOISE_LATENTS_HRF = 'denoise_latents_hrf'; export const LATENTS_TO_IMAGE = 'latents_to_image'; +export const LATENTS_TO_IMAGE_HRF = 'latents_to_image_hrf'; export const SAVE_IMAGE = 'save_image'; export const NSFW_CHECKER = 'nsfw_checker'; export const WATERMARKER = 'invisible_watermark'; export const NOISE = 'noise'; +export const NOISE_HRF = 'noise_hrf'; export const RANDOM_INT = 'rand_int'; export const RANGE_OF_SIZE = 'range_of_size'; export const ITERATE = 'iterate'; @@ -18,6 +21,7 @@ export const CLIP_SKIP = 'clip_skip'; export const IMAGE_TO_LATENTS = 'image_to_latents'; export const LATENTS_TO_LATENTS = 'latents_to_latents'; export const RESIZE = 'resize_image'; +export const RESCALE_LATENTS = 'rescale_latents'; export const IMG2IMG_RESIZE = 'img2img_resize'; export const CANVAS_OUTPUT = 'canvas_output'; export const INPAINT_IMAGE = 'inpaint_image'; diff --git a/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfCollapse.tsx b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfCollapse.tsx new file mode 100644 index 0000000000..ef0db1af6d --- /dev/null +++ b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfCollapse.tsx @@ -0,0 +1,65 @@ +import { Flex } from '@chakra-ui/react'; +import { createSelector } from '@reduxjs/toolkit'; +import { RootState, stateSelector } from 'app/store/store'; +import { useAppSelector } from 'app/store/storeHooks'; +import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; +import IAICollapse from 'common/components/IAICollapse'; +import { useMemo } from 'react'; +import ParamHrfStrength from './ParamHrfStrength'; +import ParamHrfToggle from './ParamHrfToggle'; +import ParamHrfWidth from './ParamHrfWidth'; +import ParamHrfHeight from './ParamHrfHeight'; +import { useFeatureStatus } from 'features/system/hooks/useFeatureStatus'; + +const selector = createSelector( + stateSelector, + (state: RootState) => { + const { hrfEnabled } = state.generation; + + return { hrfEnabled }; + }, + defaultSelectorOptions +); + +export default function ParamHrfCollapse() { + const isHRFFeatureEnabled = useFeatureStatus('hrf').isFeatureEnabled; + const { hrfEnabled } = useAppSelector(selector); + const activeLabel = useMemo(() => { + if (hrfEnabled) { + return 'On'; + } else { + return 'Off'; + } + }, [hrfEnabled]); + + if (!isHRFFeatureEnabled) { + return null; + } + + return ( + + + + {hrfEnabled && ( + + + + + )} + {hrfEnabled && } + + + ); +} diff --git a/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfHeight.tsx b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfHeight.tsx new file mode 100644 index 0000000000..053da11a5a --- /dev/null +++ b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfHeight.tsx @@ -0,0 +1,87 @@ +import { createSelector } from '@reduxjs/toolkit'; +import { stateSelector } from 'app/store/store'; +import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; +import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; +import IAISlider, { IAIFullSliderProps } from 'common/components/IAISlider'; +import { roundToMultiple } from 'common/util/roundDownToMultiple'; +import { + setHrfHeight, + setHrfWidth, +} from 'features/parameters/store/generationSlice'; +import { memo, useCallback } from 'react'; + +function findPrevMultipleOfEight(n: number): number { + return Math.floor((n - 1) / 8) * 8; +} + +const selector = createSelector( + [stateSelector], + ({ generation, hotkeys, config }) => { + const { min, fineStep, coarseStep } = config.sd.height; + const { model, height, hrfHeight, aspectRatio, hrfEnabled } = generation; + + const step = hotkeys.shift ? fineStep : coarseStep; + + return { + model, + height, + hrfHeight, + min, + step, + aspectRatio, + hrfEnabled, + }; + }, + defaultSelectorOptions +); + +type ParamHeightProps = Omit< + IAIFullSliderProps, + 'label' | 'value' | 'onChange' +>; + +const ParamHrfHeight = (props: ParamHeightProps) => { + const { height, hrfHeight, min, step, aspectRatio, hrfEnabled } = + useAppSelector(selector); + const dispatch = useAppDispatch(); + const maxHrfHeight = Math.max(findPrevMultipleOfEight(height), min); + + const handleChange = useCallback( + (v: number) => { + dispatch(setHrfHeight(v)); + if (aspectRatio) { + const newWidth = roundToMultiple(v * aspectRatio, 8); + dispatch(setHrfWidth(newWidth)); + } + }, + [dispatch, aspectRatio] + ); + + const handleReset = useCallback(() => { + dispatch(setHrfHeight(maxHrfHeight)); + if (aspectRatio) { + const newWidth = roundToMultiple(maxHrfHeight * aspectRatio, 8); + dispatch(setHrfWidth(newWidth)); + } + }, [dispatch, maxHrfHeight, aspectRatio]); + + return ( + + ); +}; + +export default memo(ParamHrfHeight); diff --git a/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfStrength.tsx b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfStrength.tsx new file mode 100644 index 0000000000..ba17bc2f36 --- /dev/null +++ b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfStrength.tsx @@ -0,0 +1,64 @@ +import { createSelector } from '@reduxjs/toolkit'; +import { useAppSelector, useAppDispatch } from 'app/store/storeHooks'; +import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; +import { memo, useCallback } from 'react'; +import { stateSelector } from 'app/store/store'; +import { setHrfStrength } from 'features/parameters/store/generationSlice'; +import IAISlider from 'common/components/IAISlider'; + +const selector = createSelector( + [stateSelector], + ({ generation, hotkeys, config }) => { + const { initial, min, sliderMax, inputMax, fineStep, coarseStep } = + config.sd.hrfStrength; + const { hrfStrength, hrfEnabled } = generation; + const step = hotkeys.shift ? fineStep : coarseStep; + + return { + hrfStrength, + initial, + min, + sliderMax, + inputMax, + step, + hrfEnabled, + }; + }, + defaultSelectorOptions +); + +const ParamHrfStrength = () => { + const { hrfStrength, initial, min, sliderMax, step, hrfEnabled } = + useAppSelector(selector); + const dispatch = useAppDispatch(); + + const handleHrfStrengthReset = useCallback(() => { + dispatch(setHrfStrength(initial)); + }, [dispatch, initial]); + + const handleHrfStrengthChange = useCallback( + (v: number) => { + dispatch(setHrfStrength(v)); + }, + [dispatch] + ); + + return ( + + ); +}; + +export default memo(ParamHrfStrength); diff --git a/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfToggle.tsx b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfToggle.tsx new file mode 100644 index 0000000000..8a68ead652 --- /dev/null +++ b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfToggle.tsx @@ -0,0 +1,27 @@ +import { RootState } from 'app/store/store'; +import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; +import IAISwitch from 'common/components/IAISwitch'; +import { setHrfEnabled } from 'features/parameters/store/generationSlice'; +import { ChangeEvent, useCallback } from 'react'; + +export default function ParamHrfToggle() { + const dispatch = useAppDispatch(); + + const hrfEnabled = useAppSelector( + (state: RootState) => state.generation.hrfEnabled + ); + + const handleHrfEnabled = useCallback( + (e: ChangeEvent) => + dispatch(setHrfEnabled(e.target.checked)), + [dispatch] + ); + + return ( + + ); +} diff --git a/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfWidth.tsx b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfWidth.tsx new file mode 100644 index 0000000000..09615d5154 --- /dev/null +++ b/invokeai/frontend/web/src/features/parameters/components/Parameters/HighResFix/ParamHrfWidth.tsx @@ -0,0 +1,84 @@ +import { createSelector } from '@reduxjs/toolkit'; +import { stateSelector } from 'app/store/store'; +import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; +import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions'; +import IAISlider, { IAIFullSliderProps } from 'common/components/IAISlider'; +import { roundToMultiple } from 'common/util/roundDownToMultiple'; +import { + setHrfHeight, + setHrfWidth, +} from 'features/parameters/store/generationSlice'; +import { memo, useCallback } from 'react'; + +function findPrevMultipleOfEight(n: number): number { + return Math.floor((n - 1) / 8) * 8; +} + +const selector = createSelector( + [stateSelector], + ({ generation, hotkeys, config }) => { + const { min, fineStep, coarseStep } = config.sd.width; + const { model, width, hrfWidth, aspectRatio, hrfEnabled } = generation; + + const step = hotkeys.shift ? fineStep : coarseStep; + + return { + model, + width, + hrfWidth, + min, + step, + aspectRatio, + hrfEnabled, + }; + }, + defaultSelectorOptions +); + +type ParamWidthProps = Omit; + +const ParamHrfWidth = (props: ParamWidthProps) => { + const { width, hrfWidth, min, step, aspectRatio, hrfEnabled } = + useAppSelector(selector); + const dispatch = useAppDispatch(); + const maxHrfWidth = Math.max(findPrevMultipleOfEight(width), min); + + const handleChange = useCallback( + (v: number) => { + dispatch(setHrfWidth(v)); + if (aspectRatio) { + const newHeight = roundToMultiple(v / aspectRatio, 8); + dispatch(setHrfHeight(newHeight)); + } + }, + [dispatch, aspectRatio] + ); + + const handleReset = useCallback(() => { + dispatch(setHrfWidth(maxHrfWidth)); + if (aspectRatio) { + const newHeight = roundToMultiple(maxHrfWidth / aspectRatio, 8); + dispatch(setHrfHeight(newHeight)); + } + }, [dispatch, maxHrfWidth, aspectRatio]); + + return ( + + ); +}; + +export default memo(ParamHrfWidth); diff --git a/invokeai/frontend/web/src/features/parameters/store/generationSlice.ts b/invokeai/frontend/web/src/features/parameters/store/generationSlice.ts index 84d26b6476..63d545662a 100644 --- a/invokeai/frontend/web/src/features/parameters/store/generationSlice.ts +++ b/invokeai/frontend/web/src/features/parameters/store/generationSlice.ts @@ -27,6 +27,10 @@ import { } from '../types/parameterSchemas'; export interface GenerationState { + hrfHeight: HeightParam; + hrfWidth: WidthParam; + hrfEnabled: boolean; + hrfStrength: StrengthParam; cfgScale: CfgScaleParam; height: HeightParam; img2imgStrength: StrengthParam; @@ -69,6 +73,10 @@ export interface GenerationState { } export const initialGenerationState: GenerationState = { + hrfHeight: 64, + hrfWidth: 64, + hrfStrength: 0.75, + hrfEnabled: false, cfgScale: 7.5, height: 512, img2imgStrength: 0.75, @@ -271,6 +279,18 @@ export const generationSlice = createSlice({ setClipSkip: (state, action: PayloadAction) => { state.clipSkip = action.payload; }, + setHrfHeight: (state, action: PayloadAction) => { + state.hrfHeight = action.payload; + }, + setHrfWidth: (state, action: PayloadAction) => { + state.hrfWidth = action.payload; + }, + setHrfStrength: (state, action: PayloadAction) => { + state.hrfStrength = action.payload; + }, + setHrfEnabled: (state, action: PayloadAction) => { + state.hrfEnabled = action.payload; + }, shouldUseCpuNoiseChanged: (state, action: PayloadAction) => { state.shouldUseCpuNoise = action.payload; }, @@ -355,6 +375,10 @@ export const { setSeamlessXAxis, setSeamlessYAxis, setClipSkip, + setHrfHeight, + setHrfWidth, + setHrfStrength, + setHrfEnabled, shouldUseCpuNoiseChanged, setAspectRatio, setShouldLockAspectRatio, diff --git a/invokeai/frontend/web/src/features/parameters/types/parameterSchemas.ts b/invokeai/frontend/web/src/features/parameters/types/parameterSchemas.ts index e548b9aeaf..7a6b6b8a5c 100644 --- a/invokeai/frontend/web/src/features/parameters/types/parameterSchemas.ts +++ b/invokeai/frontend/web/src/features/parameters/types/parameterSchemas.ts @@ -210,6 +210,15 @@ export type HeightParam = z.infer; export const isValidHeight = (val: unknown): val is HeightParam => zHeight.safeParse(val).success; +/** + * Zod schema for resolution parameter + */ +export const zResolution = z.tuple([zWidth, zHeight]); +/** + * Type alias for resolution parameter, inferred from its zod schema + */ +export type ResolutionParam = z.infer; + export const zBaseModel = z.enum([ 'any', 'sd-1', diff --git a/invokeai/frontend/web/src/features/system/store/configSlice.ts b/invokeai/frontend/web/src/features/system/store/configSlice.ts index 30eebab5d5..78d540203b 100644 --- a/invokeai/frontend/web/src/features/system/store/configSlice.ts +++ b/invokeai/frontend/web/src/features/system/store/configSlice.ts @@ -69,6 +69,14 @@ export const initialConfigState: AppConfig = { fineStep: 0.01, coarseStep: 0.05, }, + hrfStrength: { + initial: 0.7, + min: 0, + sliderMax: 1, + inputMax: 1, + fineStep: 0.01, + coarseStep: 0.05, + }, dynamicPrompts: { maxPrompts: { initial: 100, diff --git a/invokeai/frontend/web/src/features/ui/components/tabs/TextToImage/TextToImageTabParameters.tsx b/invokeai/frontend/web/src/features/ui/components/tabs/TextToImage/TextToImageTabParameters.tsx index 0c153fd7eb..8ec8c9caed 100644 --- a/invokeai/frontend/web/src/features/ui/components/tabs/TextToImage/TextToImageTabParameters.tsx +++ b/invokeai/frontend/web/src/features/ui/components/tabs/TextToImage/TextToImageTabParameters.tsx @@ -3,6 +3,7 @@ import ParamLoraCollapse from 'features/lora/components/ParamLoraCollapse'; import ParamAdvancedCollapse from 'features/parameters/components/Parameters/Advanced/ParamAdvancedCollapse'; import ControlAdaptersCollapse from 'features/controlAdapters/components/ControlAdaptersCollapse'; import ParamSymmetryCollapse from 'features/parameters/components/Parameters/Symmetry/ParamSymmetryCollapse'; +import ParamHrfCollapse from 'features/parameters/components/Parameters/HighResFix/ParamHrfCollapse'; import { memo } from 'react'; import ParamPromptArea from '../../../../parameters/components/Parameters/Prompt/ParamPromptArea'; import TextToImageTabCoreParameters from './TextToImageTabCoreParameters'; @@ -16,6 +17,7 @@ const TextToImageTabParameters = () => { + ); diff --git a/invokeai/frontend/web/src/services/api/schema.d.ts b/invokeai/frontend/web/src/services/api/schema.d.ts index 6da1fa5154..d71d01e6af 100644 --- a/invokeai/frontend/web/src/services/api/schema.d.ts +++ b/invokeai/frontend/web/src/services/api/schema.d.ts @@ -6190,6 +6190,21 @@ export type components = { * @description The VAE used for decoding, if the main model's default was not used */ vae?: components["schemas"]["VAEModelField"]; + /** + * High Resolution Fix Width + * @description The high resolution fix height and width multiplier. + */ + hrf_width?: number; + /** + * High Resolution Fix Height + * @description The high resolution fix height and width multiplier. + */ + hrf_height?: number; + /** + * High Resolution Strength + * @description The high resolution fix img2img strength used in the upscale pass. + */ + hrf_strength?: number; /** * Positive Style Prompt * @description The positive style prompt parameter diff --git a/invokeai/frontend/web/src/services/api/types.ts b/invokeai/frontend/web/src/services/api/types.ts index da56d2782f..6fda849b89 100644 --- a/invokeai/frontend/web/src/services/api/types.ts +++ b/invokeai/frontend/web/src/services/api/types.ts @@ -137,6 +137,7 @@ export type CompelInvocation = s['CompelInvocation']; export type DynamicPromptInvocation = s['DynamicPromptInvocation']; export type NoiseInvocation = s['NoiseInvocation']; export type DenoiseLatentsInvocation = s['DenoiseLatentsInvocation']; +export type ResizeLatentsInvocation = s['ResizeLatentsInvocation']; export type ONNXTextToLatentsInvocation = s['ONNXTextToLatentsInvocation']; export type SDXLLoraLoaderInvocation = s['SDXLLoraLoaderInvocation']; export type ImageToLatentsInvocation = s['ImageToLatentsInvocation'];