mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
feat(nodes): restore canvas functionality (non-latents)
This commit is contained in:
parent
29c952dcf6
commit
1fb307abf4
@ -14,14 +14,17 @@ from invokeai.app.models.image import ImageCategory, ImageType
|
|||||||
from invokeai.app.util.misc import SEED_MAX, get_random_seed
|
from invokeai.app.util.misc import SEED_MAX, get_random_seed
|
||||||
from invokeai.backend.generator.inpaint import infill_methods
|
from invokeai.backend.generator.inpaint import infill_methods
|
||||||
from .baseinvocation import BaseInvocation, InvocationContext, InvocationConfig
|
from .baseinvocation import BaseInvocation, InvocationContext, InvocationConfig
|
||||||
from .image import ImageOutput, build_image_output
|
from .image import ImageOutput
|
||||||
from ...backend.generator import Txt2Img, Img2Img, Inpaint, InvokeAIGenerator
|
from ...backend.generator import Txt2Img, Img2Img, Inpaint, InvokeAIGenerator
|
||||||
from ...backend.stable_diffusion import PipelineIntermediateState
|
from ...backend.stable_diffusion import PipelineIntermediateState
|
||||||
from ..util.step_callback import stable_diffusion_step_callback
|
from ..util.step_callback import stable_diffusion_step_callback
|
||||||
|
|
||||||
SAMPLER_NAME_VALUES = Literal[tuple(InvokeAIGenerator.schedulers())]
|
SAMPLER_NAME_VALUES = Literal[tuple(InvokeAIGenerator.schedulers())]
|
||||||
INFILL_METHODS = Literal[tuple(infill_methods())]
|
INFILL_METHODS = Literal[tuple(infill_methods())]
|
||||||
DEFAULT_INFILL_METHOD = 'patchmatch' if 'patchmatch' in get_args(INFILL_METHODS) else 'tile'
|
DEFAULT_INFILL_METHOD = (
|
||||||
|
"patchmatch" if "patchmatch" in get_args(INFILL_METHODS) else "tile"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class SDImageInvocation(BaseModel):
|
class SDImageInvocation(BaseModel):
|
||||||
"""Helper class to provide all Stable Diffusion raster image invocations with additional config"""
|
"""Helper class to provide all Stable Diffusion raster image invocations with additional config"""
|
||||||
@ -92,7 +95,7 @@ class TextToImageInvocation(BaseInvocation, SDImageInvocation):
|
|||||||
# each time it is called. We only need the first one.
|
# each time it is called. We only need the first one.
|
||||||
generate_output = next(outputs)
|
generate_output = next(outputs)
|
||||||
|
|
||||||
image_dto = context.services.images_new.create(
|
image_dto = context.services.images.create(
|
||||||
image=generate_output.image,
|
image=generate_output.image,
|
||||||
image_type=ImageType.RESULT,
|
image_type=ImageType.RESULT,
|
||||||
image_category=ImageCategory.GENERAL,
|
image_category=ImageCategory.GENERAL,
|
||||||
@ -100,35 +103,13 @@ class TextToImageInvocation(BaseInvocation, SDImageInvocation):
|
|||||||
node_id=self.id,
|
node_id=self.id,
|
||||||
)
|
)
|
||||||
|
|
||||||
# Results are image and seed, unwrap for now and ignore the seed
|
return ImageOutput(
|
||||||
# TODO: pre-seed?
|
image=ImageField(
|
||||||
# TODO: can this return multiple results? Should it?
|
|
||||||
# image_type = ImageType.RESULT
|
|
||||||
# image_name = context.services.images.create_name(
|
|
||||||
# context.graph_execution_state_id, self.id
|
|
||||||
# )
|
|
||||||
|
|
||||||
# metadata = context.services.metadata.build_metadata(
|
|
||||||
# session_id=context.graph_execution_state_id, node=self
|
|
||||||
# )
|
|
||||||
|
|
||||||
# context.services.images.save(
|
|
||||||
# image_type, image_name, generate_output.image, metadata
|
|
||||||
# )
|
|
||||||
|
|
||||||
# context.services.images_db.set(
|
|
||||||
# id=image_name,
|
|
||||||
# image_type=ImageType.RESULT,
|
|
||||||
# image_category=ImageCategory.GENERAL,
|
|
||||||
# session_id=context.graph_execution_state_id,
|
|
||||||
# node_id=self.id,
|
|
||||||
# metadata=GeneratedImageOrLatentsMetadata(),
|
|
||||||
# )
|
|
||||||
|
|
||||||
return build_image_output(
|
|
||||||
image_type=image_dto.image_type,
|
|
||||||
image_name=image_dto.image_name,
|
image_name=image_dto.image_name,
|
||||||
image=generate_output.image,
|
image_type=image_dto.image_type,
|
||||||
|
),
|
||||||
|
width=image_dto.width,
|
||||||
|
height=image_dto.height,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -164,7 +145,7 @@ class ImageToImageInvocation(TextToImageInvocation):
|
|||||||
image = (
|
image = (
|
||||||
None
|
None
|
||||||
if self.image is None
|
if self.image is None
|
||||||
else context.services.images.get(
|
else context.services.images.get_pil_image(
|
||||||
self.image.image_type, self.image.image_name
|
self.image.image_type, self.image.image_name
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -194,26 +175,23 @@ class ImageToImageInvocation(TextToImageInvocation):
|
|||||||
# each time it is called. We only need the first one.
|
# each time it is called. We only need the first one.
|
||||||
generator_output = next(outputs)
|
generator_output = next(outputs)
|
||||||
|
|
||||||
result_image = generator_output.image
|
image_dto = context.services.images.create(
|
||||||
|
image=generator_output.image,
|
||||||
# Results are image and seed, unwrap for now and ignore the seed
|
image_type=ImageType.RESULT,
|
||||||
# TODO: pre-seed?
|
image_category=ImageCategory.GENERAL,
|
||||||
# TODO: can this return multiple results? Should it?
|
session_id=context.graph_execution_state_id,
|
||||||
image_type = ImageType.RESULT
|
node_id=self.id,
|
||||||
image_name = context.services.images.create_name(
|
|
||||||
context.graph_execution_state_id, self.id
|
|
||||||
)
|
)
|
||||||
|
|
||||||
metadata = context.services.metadata.build_metadata(
|
return ImageOutput(
|
||||||
session_id=context.graph_execution_state_id, node=self
|
image=ImageField(
|
||||||
|
image_name=image_dto.image_name,
|
||||||
|
image_type=image_dto.image_type,
|
||||||
|
),
|
||||||
|
width=image_dto.width,
|
||||||
|
height=image_dto.height,
|
||||||
)
|
)
|
||||||
|
|
||||||
context.services.images.save(image_type, image_name, result_image, metadata)
|
|
||||||
return build_image_output(
|
|
||||||
image_type=image_type,
|
|
||||||
image_name=image_name,
|
|
||||||
image=result_image,
|
|
||||||
)
|
|
||||||
|
|
||||||
class InpaintInvocation(ImageToImageInvocation):
|
class InpaintInvocation(ImageToImageInvocation):
|
||||||
"""Generates an image using inpaint."""
|
"""Generates an image using inpaint."""
|
||||||
@ -223,16 +201,38 @@ class InpaintInvocation(ImageToImageInvocation):
|
|||||||
# Inputs
|
# Inputs
|
||||||
mask: Union[ImageField, None] = Field(description="The mask")
|
mask: Union[ImageField, None] = Field(description="The mask")
|
||||||
seam_size: int = Field(default=96, ge=1, description="The seam inpaint size (px)")
|
seam_size: int = Field(default=96, ge=1, description="The seam inpaint size (px)")
|
||||||
seam_blur: int = Field(default=16, ge=0, description="The seam inpaint blur radius (px)")
|
seam_blur: int = Field(
|
||||||
|
default=16, ge=0, description="The seam inpaint blur radius (px)"
|
||||||
|
)
|
||||||
seam_strength: float = Field(
|
seam_strength: float = Field(
|
||||||
default=0.75, gt=0, le=1, description="The seam inpaint strength"
|
default=0.75, gt=0, le=1, description="The seam inpaint strength"
|
||||||
)
|
)
|
||||||
seam_steps: int = Field(default=30, ge=1, description="The number of steps to use for seam inpaint")
|
seam_steps: int = Field(
|
||||||
tile_size: int = Field(default=32, ge=1, description="The tile infill method size (px)")
|
default=30, ge=1, description="The number of steps to use for seam inpaint"
|
||||||
infill_method: INFILL_METHODS = Field(default=DEFAULT_INFILL_METHOD, description="The method used to infill empty regions (px)")
|
)
|
||||||
inpaint_width: Optional[int] = Field(default=None, multiple_of=8, gt=0, description="The width of the inpaint region (px)")
|
tile_size: int = Field(
|
||||||
inpaint_height: Optional[int] = Field(default=None, multiple_of=8, gt=0, description="The height of the inpaint region (px)")
|
default=32, ge=1, description="The tile infill method size (px)"
|
||||||
inpaint_fill: Optional[ColorField] = Field(default=ColorField(r=127, g=127, b=127, a=255), description="The solid infill method color")
|
)
|
||||||
|
infill_method: INFILL_METHODS = Field(
|
||||||
|
default=DEFAULT_INFILL_METHOD,
|
||||||
|
description="The method used to infill empty regions (px)",
|
||||||
|
)
|
||||||
|
inpaint_width: Optional[int] = Field(
|
||||||
|
default=None,
|
||||||
|
multiple_of=8,
|
||||||
|
gt=0,
|
||||||
|
description="The width of the inpaint region (px)",
|
||||||
|
)
|
||||||
|
inpaint_height: Optional[int] = Field(
|
||||||
|
default=None,
|
||||||
|
multiple_of=8,
|
||||||
|
gt=0,
|
||||||
|
description="The height of the inpaint region (px)",
|
||||||
|
)
|
||||||
|
inpaint_fill: Optional[ColorField] = Field(
|
||||||
|
default=ColorField(r=127, g=127, b=127, a=255),
|
||||||
|
description="The solid infill method color",
|
||||||
|
)
|
||||||
inpaint_replace: float = Field(
|
inpaint_replace: float = Field(
|
||||||
default=0.0,
|
default=0.0,
|
||||||
ge=0.0,
|
ge=0.0,
|
||||||
@ -257,14 +257,14 @@ class InpaintInvocation(ImageToImageInvocation):
|
|||||||
image = (
|
image = (
|
||||||
None
|
None
|
||||||
if self.image is None
|
if self.image is None
|
||||||
else context.services.images.get(
|
else context.services.images.get_pil_image(
|
||||||
self.image.image_type, self.image.image_name
|
self.image.image_type, self.image.image_name
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
mask = (
|
mask = (
|
||||||
None
|
None
|
||||||
if self.mask is None
|
if self.mask is None
|
||||||
else context.services.images.get(self.mask.image_type, self.mask.image_name)
|
else context.services.images.get_pil_image(self.mask.image_type, self.mask.image_name)
|
||||||
)
|
)
|
||||||
|
|
||||||
# Handle invalid model parameter
|
# Handle invalid model parameter
|
||||||
@ -290,23 +290,19 @@ class InpaintInvocation(ImageToImageInvocation):
|
|||||||
# each time it is called. We only need the first one.
|
# each time it is called. We only need the first one.
|
||||||
generator_output = next(outputs)
|
generator_output = next(outputs)
|
||||||
|
|
||||||
result_image = generator_output.image
|
image_dto = context.services.images.create(
|
||||||
|
image=generator_output.image,
|
||||||
# Results are image and seed, unwrap for now and ignore the seed
|
image_type=ImageType.RESULT,
|
||||||
# TODO: pre-seed?
|
image_category=ImageCategory.GENERAL,
|
||||||
# TODO: can this return multiple results? Should it?
|
session_id=context.graph_execution_state_id,
|
||||||
image_type = ImageType.RESULT
|
node_id=self.id,
|
||||||
image_name = context.services.images.create_name(
|
|
||||||
context.graph_execution_state_id, self.id
|
|
||||||
)
|
)
|
||||||
|
|
||||||
metadata = context.services.metadata.build_metadata(
|
return ImageOutput(
|
||||||
session_id=context.graph_execution_state_id, node=self
|
image=ImageField(
|
||||||
)
|
image_name=image_dto.image_name,
|
||||||
|
image_type=image_dto.image_type,
|
||||||
context.services.images.save(image_type, image_name, result_image, metadata)
|
),
|
||||||
return build_image_output(
|
width=image_dto.width,
|
||||||
image_type=image_type,
|
height=image_dto.height,
|
||||||
image_name=image_name,
|
|
||||||
image=result_image,
|
|
||||||
)
|
)
|
||||||
|
Loading…
Reference in New Issue
Block a user