mirror of
https://github.com/comfyanonymous/ComfyUI.git
synced 2025-06-08 15:17:14 +00:00
* Add Ideogram generate node. * Add staging api. * Add API_NODE and common error for missing auth token (#5) * Add Minimax Video Generation + Async Task queue polling example (#6) * [Minimax] Show video preview and embed workflow in ouput (#7) * Remove uv.lock * Remove polling operations. * Revert "Remove polling operations." * Update stubs. * Added Ideogram and Minimax back in. * Added initial BFL Flux 1.1 [pro] Ultra node (#11) * Add --comfy-api-base launch arg (#13) * Add instructions for staging development. (#14) * remove validation to make it easier to run against LAN copies of the API * Manually add BFL polling status response schema (#15) * Add function for uploading files. (#18) * Add Luma nodes (#16) * Refactor util functions (#20) * Add VIDEO type (#21) * Add rest of Luma node functionality (#19) * Fix image_luma_ref not working (#28) * [Bug] Remove duplicated option T2V-01 in MinimaxTextToVideoNode (#31) * Add utils to map from pydantic model fields to comfy node inputs (#30) * add veo2, bump av req (#32) * Add Recraft nodes (#29) * Add Kling Nodes (#12) * Add Camera Concepts (luma_concepts) to Luma Video nodes (#33) * Add Runway nodes (#17) * Convert Minimax node to use VIDEO output type (#34) * Standard `CATEGORY` system for api nodes (#35) * Set `Content-Type` header when uploading files (#36) * add better error propagation to veo2 (#37) * Add Realistic Image and Logo Raster styles for Recraft v3 (#38) * Fix runway image upload and progress polling (#39) * Fix image upload for Luma: only include `Content-Type` header field if it's set explicitly (#40) * Moved Luma nodes to nodes_luma.py (#47) * Moved Recraft nodes to nodes_recraft.py (#48) * Add Pixverse nodes (#46) * Move and fix BFL nodes to node_bfl.py (#49) * Move and edit Minimax node to nodes_minimax.py (#50) * Add Minimax Image to Video node + Cleanup (#51) * Add Recraft Text to Vector node, add Save SVG node to handle its output (#53) * Added pixverse_template support to Pixverse Text to Video node (#54) * Added Recraft Controls + Recraft Color RGB nodes (#57) * split remaining nodes out of nodes_api, make utility lib, refactor ideogram (#61) * Add types and doctstrings to utils file (#64) * Fix: `PollingOperation` progress bar update progress by absolute value (#65) * Use common download function in kling nodes module (#67) * Fix: Luma video nodes in `api nodes/image` category (#68) * Set request type explicitly (#66) * Add `control_after_generate` to all seed inputs (#69) * Fix bug: deleting `Content-Type` when property does not exist (#73) * Add preview to Save SVG node (#74) * change default poll interval (#76), rework veo2 * Add Pixverse and updated Kling types (#75) * Added Pixverse Image to VIdeo node (#77) * Add Pixverse Transition Video node (#79) * Proper ray-1-6 support as fix has been applied in backend (#80) * Added Recraft Style - Infinite Style Library node (#82) * add ideogram v3 (#83) * [Kling] Split Camera Control config to its own node (#81) * Add Pika i2v and t2v nodes (#52) * Temporary Fix for Runway (#87) * Added Stability Stable Image Ultra node (#86) * Remove Runway nodes (#88) * Fix: Prompt text can't be validated in Kling nodes when using primitive nodes (#90) * Fix: typo in node name "Stabiliy" => "Stability" (#91) * Add String (Multiline) node (#93) * Update Pika Duration and Resolution options (#94) * Change base branch to master. Not main. (#95) * Fix UploadRequest file_name param (#98) * Removed Infinite Style Library until later (#99) * fix ideogram style types (#100) * fix multi image return (#101) * add metadata saving to SVG (#102) * Bump templates version to include API node template workflows (#104) * Fix: `download_url_to_video_output` return type (#103) * fix 4o generation bug (#106) * Serve SVG files directly (#107) * Add a bunch of nodes, 3 ready to use, the rest waiting for endpoint support (#108) * Revert "Serve SVG files directly" (#111) * Expose 4 remaining Recraft nodes (#112) * [Kling] Add `Duration` and `Video ID` outputs (#105) * Fix: datamodel-codegen sets string#binary type to non-existent `bytes_aliased` variable (#114) * Fix: Dall-e 2 not setting request content-type dynamically (#113) * Default request timeout: one hour. (#116) * Add Kling nodes: camera control, start-end frame, lip-sync, video extend (#115) * Add 8 nodes - 4 BFL, 4 Stability (#117) * Fix error for Recraft ImageToImage error for nonexistent random_seed param (#118) * Add remaining Pika nodes (#119) * Make controls input work for Recraft Image to Image node (#120) * Use upstream PR: Support saving Comfy VIDEO type to buffer (#123) * Use Upstream PR: "Fix: Error creating video when sliced audio tensor chunks are non-c-contiguous" (#127) * Improve audio upload utils (#128) * Fix: Nested `AnyUrl` in request model cannot be serialized (Kling, Runway) (#129) * Show errors and API output URLs to the user (change log levels) (#131) * Fix: Luma I2I fails when weight is <=0.01 (#132) * Change category of `LumaConcepts` node from image to video (#133) * Fix: `image.shape` accessed before `image` is null-checked (#134) * Apply small fixes and most prompt validation (if needed to avoid API error) (#135) * Node name/category modifications (#140) * Add back Recraft Style - Infinite Style Library node (#141) * Fixed Kling: Check attributes of pydantic types. (#144) * Bump `comfyui-workflow-templates` version (#142) * [Kling] Print response data when error validating response (#146) * Fix: error validating Kling image response, trying to use `"key" in` on Pydantic class instance (#147) * [Kling] Fix: Correct/verify supported subset of input combos in Kling nodes (#149) * [Kling] Fix typo in node description (#150) * [Kling] Fix: CFG min/max not being enforced (#151) * Rebase launch-rebase (private) on prep-branch (public copy of master) (#153) * Bump templates version (#154) * Fix: Kling image gen nodes don't return entire batch when `n` > 1 (#152) * Remove pixverse_template from PixVerse Transition Video node (#155) * Invert image_weight value on Luma Image to Image node (#156) * Invert and resize mask for Ideogram V3 node to match masking conventions (#158) * [Kling] Fix: image generation nodes not returning Tuple (#159) * [Bug] [Kling] Fix Kling camera control (#161) * Kling Image Gen v2 + improve node descriptions for Flux/OpenAI (#160) * [Kling] Don't return video_id from dual effect video (#162) * Bump frontend to 1.18.8 (#163) * Use 3.9 compat syntax (#164) * Use Python 3.10 * add example env var * Update templates to 0.1.11 * Bump frontend to 1.18.9 --------- Co-authored-by: Robin Huang <robin.j.huang@gmail.com> Co-authored-by: Christian Byrne <cbyrne@comfy.org> Co-authored-by: thot experiment <94414189+thot-experiment@users.noreply.github.com>
1218 lines
39 KiB
Python
1218 lines
39 KiB
Python
from __future__ import annotations
|
||
from inspect import cleandoc
|
||
from comfy.utils import ProgressBar
|
||
from comfy.comfy_types.node_typing import IO
|
||
from comfy_api_nodes.apis.recraft_api import (
|
||
RecraftImageGenerationRequest,
|
||
RecraftImageGenerationResponse,
|
||
RecraftImageSize,
|
||
RecraftModel,
|
||
RecraftStyle,
|
||
RecraftStyleV3,
|
||
RecraftColor,
|
||
RecraftColorChain,
|
||
RecraftControls,
|
||
RecraftIO,
|
||
get_v3_substyles,
|
||
)
|
||
from comfy_api_nodes.apis.client import (
|
||
ApiEndpoint,
|
||
HttpMethod,
|
||
SynchronousOperation,
|
||
EmptyRequest,
|
||
)
|
||
from comfy_api_nodes.apinode_utils import (
|
||
bytesio_to_image_tensor,
|
||
download_url_to_bytesio,
|
||
tensor_to_bytesio,
|
||
resize_mask_to_image,
|
||
validate_string,
|
||
)
|
||
import folder_paths
|
||
import json
|
||
import os
|
||
import torch
|
||
from io import BytesIO
|
||
from PIL import UnidentifiedImageError
|
||
|
||
|
||
def handle_recraft_file_request(
|
||
image: torch.Tensor,
|
||
path: str,
|
||
mask: torch.Tensor=None,
|
||
total_pixels=4096*4096,
|
||
timeout=1024,
|
||
request=None,
|
||
auth_token=None
|
||
) -> list[BytesIO]:
|
||
"""
|
||
Handle sending common Recraft file-only request to get back file bytes.
|
||
"""
|
||
if request is None:
|
||
request = EmptyRequest()
|
||
|
||
files = {
|
||
'image': tensor_to_bytesio(image, total_pixels=total_pixels).read()
|
||
}
|
||
if mask is not None:
|
||
files['mask'] = tensor_to_bytesio(mask, total_pixels=total_pixels).read()
|
||
|
||
operation = SynchronousOperation(
|
||
endpoint=ApiEndpoint(
|
||
path=path,
|
||
method=HttpMethod.POST,
|
||
request_model=type(request),
|
||
response_model=RecraftImageGenerationResponse,
|
||
),
|
||
request=request,
|
||
files=files,
|
||
content_type="multipart/form-data",
|
||
auth_token=auth_token,
|
||
multipart_parser=recraft_multipart_parser,
|
||
)
|
||
response: RecraftImageGenerationResponse = operation.execute()
|
||
all_bytesio = []
|
||
if response.image is not None:
|
||
all_bytesio.append(download_url_to_bytesio(response.image.url, timeout=timeout))
|
||
else:
|
||
for data in response.data:
|
||
all_bytesio.append(download_url_to_bytesio(data.url, timeout=timeout))
|
||
|
||
return all_bytesio
|
||
|
||
|
||
def recraft_multipart_parser(data, parent_key=None, formatter: callable=None, converted_to_check: list[list]=None, is_list=False) -> dict:
|
||
"""
|
||
Formats data such that multipart/form-data will work with requests library
|
||
when both files and data are present.
|
||
|
||
The OpenAI client that Recraft uses has a bizarre way of serializing lists:
|
||
|
||
It does NOT keep track of indeces of each list, so for background_color, that must be serialized as:
|
||
'background_color[rgb][]' = [0, 0, 255]
|
||
where the array is assigned to a key that has '[]' at the end, to signal it's an array.
|
||
|
||
This has the consequence of nested lists having the exact same key, forcing arrays to merge; all colors inputs fall under the same key:
|
||
if 1 color -> 'controls[colors][][rgb][]' = [0, 0, 255]
|
||
if 2 colors -> 'controls[colors][][rgb][]' = [0, 0, 255, 255, 0, 0]
|
||
if 3 colors -> 'controls[colors][][rgb][]' = [0, 0, 255, 255, 0, 0, 0, 255, 0]
|
||
etc.
|
||
Whoever made this serialization up at OpenAI added the constraint that lists must be of uniform length on objects of same 'type'.
|
||
"""
|
||
# Modification of a function that handled a different type of multipart parsing, big ups:
|
||
# https://gist.github.com/kazqvaizer/4cebebe5db654a414132809f9f88067b
|
||
|
||
def handle_converted_lists(data, parent_key, lists_to_check=tuple[list]):
|
||
# if list already exists exists, just extend list with data
|
||
for check_list in lists_to_check:
|
||
for conv_tuple in check_list:
|
||
if conv_tuple[0] == parent_key and type(conv_tuple[1]) is list:
|
||
conv_tuple[1].append(formatter(data))
|
||
return True
|
||
return False
|
||
|
||
if converted_to_check is None:
|
||
converted_to_check = []
|
||
|
||
|
||
if formatter is None:
|
||
formatter = lambda v: v # Multipart representation of value
|
||
|
||
if type(data) is not dict:
|
||
# if list already exists exists, just extend list with data
|
||
added = handle_converted_lists(data, parent_key, converted_to_check)
|
||
if added:
|
||
return {}
|
||
# otherwise if is_list, create new list with data
|
||
if is_list:
|
||
return {parent_key: [formatter(data)]}
|
||
# return new key with data
|
||
return {parent_key: formatter(data)}
|
||
|
||
converted = []
|
||
next_check = [converted]
|
||
next_check.extend(converted_to_check)
|
||
|
||
for key, value in data.items():
|
||
current_key = key if parent_key is None else f"{parent_key}[{key}]"
|
||
if type(value) is dict:
|
||
converted.extend(recraft_multipart_parser(value, current_key, formatter, next_check).items())
|
||
elif type(value) is list:
|
||
for ind, list_value in enumerate(value):
|
||
iter_key = f"{current_key}[]"
|
||
converted.extend(recraft_multipart_parser(list_value, iter_key, formatter, next_check, is_list=True).items())
|
||
else:
|
||
converted.append((current_key, formatter(value)))
|
||
|
||
return dict(converted)
|
||
|
||
|
||
class handle_recraft_image_output:
|
||
"""
|
||
Catch an exception related to receiving SVG data instead of image, when Infinite Style Library style_id is in use.
|
||
"""
|
||
def __init__(self):
|
||
pass
|
||
|
||
def __enter__(self):
|
||
pass
|
||
|
||
def __exit__(self, exc_type, exc_val, exc_tb):
|
||
if exc_type is not None and exc_type is UnidentifiedImageError:
|
||
raise Exception("Received output data was not an image; likely an SVG. If you used style_id, make sure it is not a Vector art style.")
|
||
|
||
|
||
class SVG:
|
||
"""
|
||
Stores SVG representations via a list of BytesIO objects.
|
||
"""
|
||
def __init__(self, data: list[BytesIO]):
|
||
self.data = data
|
||
|
||
def combine(self, other: SVG):
|
||
return SVG(self.data + other.data)
|
||
|
||
@staticmethod
|
||
def combine_all(svgs: list[SVG]):
|
||
all_svgs = []
|
||
for svg in svgs:
|
||
all_svgs.extend(svg.data)
|
||
return SVG(all_svgs)
|
||
|
||
|
||
class SaveSVGNode:
|
||
"""
|
||
Save SVG files on disk.
|
||
"""
|
||
|
||
def __init__(self):
|
||
self.output_dir = folder_paths.get_output_directory()
|
||
self.type = "output"
|
||
self.prefix_append = ""
|
||
|
||
RETURN_TYPES = ()
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "save_svg"
|
||
CATEGORY = "api node/image/Recraft"
|
||
OUTPUT_NODE = True
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"svg": (RecraftIO.SVG,),
|
||
"filename_prefix": ("STRING", {"default": "svg/ComfyUI", "tooltip": "The prefix for the file to save. This may include formatting information such as %date:yyyy-MM-dd% or %Empty Latent Image.width% to include values from nodes."})
|
||
},
|
||
"hidden": {
|
||
"prompt": "PROMPT",
|
||
"extra_pnginfo": "EXTRA_PNGINFO"
|
||
}
|
||
}
|
||
|
||
def save_svg(self, svg: SVG, filename_prefix="svg/ComfyUI", prompt=None, extra_pnginfo=None):
|
||
filename_prefix += self.prefix_append
|
||
full_output_folder, filename, counter, subfolder, filename_prefix = folder_paths.get_save_image_path(filename_prefix, self.output_dir)
|
||
results = list()
|
||
|
||
# Prepare metadata JSON
|
||
metadata_dict = {}
|
||
if prompt is not None:
|
||
metadata_dict["prompt"] = prompt
|
||
if extra_pnginfo is not None:
|
||
metadata_dict.update(extra_pnginfo)
|
||
|
||
# Convert metadata to JSON string
|
||
metadata_json = json.dumps(metadata_dict, indent=2) if metadata_dict else None
|
||
|
||
for batch_number, svg_bytes in enumerate(svg.data):
|
||
filename_with_batch_num = filename.replace("%batch_num%", str(batch_number))
|
||
file = f"{filename_with_batch_num}_{counter:05}_.svg"
|
||
|
||
# Read SVG content
|
||
svg_bytes.seek(0)
|
||
svg_content = svg_bytes.read().decode('utf-8')
|
||
|
||
# Inject metadata if available
|
||
if metadata_json:
|
||
# Create metadata element with CDATA section
|
||
metadata_element = f""" <metadata>
|
||
<![CDATA[
|
||
{metadata_json}
|
||
]]>
|
||
</metadata>
|
||
"""
|
||
# Insert metadata after opening svg tag using regex
|
||
import re
|
||
svg_content = re.sub(r'(<svg[^>]*>)', r'\1\n' + metadata_element, svg_content)
|
||
|
||
# Write the modified SVG to file
|
||
with open(os.path.join(full_output_folder, file), 'wb') as svg_file:
|
||
svg_file.write(svg_content.encode('utf-8'))
|
||
|
||
results.append({
|
||
"filename": file,
|
||
"subfolder": subfolder,
|
||
"type": self.type
|
||
})
|
||
counter += 1
|
||
return { "ui": { "images": results } }
|
||
|
||
|
||
class RecraftColorRGBNode:
|
||
"""
|
||
Create Recraft Color by choosing specific RGB values.
|
||
"""
|
||
|
||
RETURN_TYPES = (RecraftIO.COLOR,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
RETURN_NAMES = ("recraft_color",)
|
||
FUNCTION = "create_color"
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"r": (IO.INT, {
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 255,
|
||
"tooltip": "Red value of color."
|
||
}),
|
||
"g": (IO.INT, {
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 255,
|
||
"tooltip": "Green value of color."
|
||
}),
|
||
"b": (IO.INT, {
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 255,
|
||
"tooltip": "Blue value of color."
|
||
}),
|
||
},
|
||
"optional": {
|
||
"recraft_color": (RecraftIO.COLOR,),
|
||
}
|
||
}
|
||
|
||
def create_color(self, r: int, g: int, b: int, recraft_color: RecraftColorChain=None):
|
||
recraft_color = recraft_color.clone() if recraft_color else RecraftColorChain()
|
||
recraft_color.add(RecraftColor(r, g, b))
|
||
return (recraft_color, )
|
||
|
||
|
||
class RecraftControlsNode:
|
||
"""
|
||
Create Recraft Controls for customizing Recraft generation.
|
||
"""
|
||
|
||
RETURN_TYPES = (RecraftIO.CONTROLS,)
|
||
RETURN_NAMES = ("recraft_controls",)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "create_controls"
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
},
|
||
"optional": {
|
||
"colors": (RecraftIO.COLOR,),
|
||
"background_color": (RecraftIO.COLOR,),
|
||
}
|
||
}
|
||
|
||
def create_controls(self, colors: RecraftColorChain=None, background_color: RecraftColorChain=None):
|
||
return (RecraftControls(colors=colors, background_color=background_color), )
|
||
|
||
|
||
class RecraftStyleV3RealisticImageNode:
|
||
"""
|
||
Select realistic_image style and optional substyle.
|
||
"""
|
||
|
||
RETURN_TYPES = (RecraftIO.STYLEV3,)
|
||
RETURN_NAMES = ("recraft_style",)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "create_style"
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
RECRAFT_STYLE = RecraftStyleV3.realistic_image
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"substyle": (get_v3_substyles(s.RECRAFT_STYLE),),
|
||
}
|
||
}
|
||
|
||
def create_style(self, substyle: str):
|
||
if substyle == "None":
|
||
substyle = None
|
||
return (RecraftStyle(self.RECRAFT_STYLE, substyle),)
|
||
|
||
|
||
class RecraftStyleV3DigitalIllustrationNode(RecraftStyleV3RealisticImageNode):
|
||
"""
|
||
Select digital_illustration style and optional substyle.
|
||
"""
|
||
|
||
RECRAFT_STYLE = RecraftStyleV3.digital_illustration
|
||
|
||
|
||
class RecraftStyleV3VectorIllustrationNode(RecraftStyleV3RealisticImageNode):
|
||
"""
|
||
Select vector_illustration style and optional substyle.
|
||
"""
|
||
|
||
RECRAFT_STYLE = RecraftStyleV3.vector_illustration
|
||
|
||
|
||
class RecraftStyleV3LogoRasterNode(RecraftStyleV3RealisticImageNode):
|
||
"""
|
||
Select vector_illustration style and optional substyle.
|
||
"""
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"substyle": (get_v3_substyles(s.RECRAFT_STYLE, include_none=False),),
|
||
}
|
||
}
|
||
|
||
RECRAFT_STYLE = RecraftStyleV3.logo_raster
|
||
|
||
|
||
class RecraftStyleInfiniteStyleLibrary:
|
||
"""
|
||
Select style based on preexisting UUID from Recraft's Infinite Style Library.
|
||
"""
|
||
|
||
RETURN_TYPES = (RecraftIO.STYLEV3,)
|
||
RETURN_NAMES = ("recraft_style",)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "create_style"
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"style_id": (IO.STRING, {
|
||
"default": "",
|
||
"tooltip": "UUID of style from Infinite Style Library.",
|
||
})
|
||
}
|
||
}
|
||
|
||
def create_style(self, style_id: str):
|
||
if not style_id:
|
||
raise Exception("The style_id input cannot be empty.")
|
||
return (RecraftStyle(style_id=style_id),)
|
||
|
||
|
||
class RecraftTextToImageNode:
|
||
"""
|
||
Generates images synchronously based on prompt and resolution.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"prompt": (
|
||
IO.STRING,
|
||
{
|
||
"multiline": True,
|
||
"default": "",
|
||
"tooltip": "Prompt for the image generation.",
|
||
},
|
||
),
|
||
"size": (
|
||
[res.value for res in RecraftImageSize],
|
||
{
|
||
"default": RecraftImageSize.res_1024x1024,
|
||
"tooltip": "The size of the generated image.",
|
||
},
|
||
),
|
||
"n": (
|
||
IO.INT,
|
||
{
|
||
"default": 1,
|
||
"min": 1,
|
||
"max": 6,
|
||
"tooltip": "The number of images to generate.",
|
||
},
|
||
),
|
||
"seed": (
|
||
IO.INT,
|
||
{
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 0xFFFFFFFFFFFFFFFF,
|
||
"control_after_generate": True,
|
||
"tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
||
},
|
||
),
|
||
},
|
||
"optional": {
|
||
"recraft_style": (RecraftIO.STYLEV3,),
|
||
"negative_prompt": (
|
||
IO.STRING,
|
||
{
|
||
"default": "",
|
||
"forceInput": True,
|
||
"tooltip": "An optional text description of undesired elements on an image.",
|
||
},
|
||
),
|
||
"recraft_controls": (
|
||
RecraftIO.CONTROLS,
|
||
{
|
||
"tooltip": "Optional additional controls over the generation via the Recraft Controls node."
|
||
},
|
||
),
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
prompt: str,
|
||
size: str,
|
||
n: int,
|
||
seed,
|
||
recraft_style: RecraftStyle = None,
|
||
negative_prompt: str = None,
|
||
recraft_controls: RecraftControls = None,
|
||
auth_token=None,
|
||
**kwargs,
|
||
):
|
||
validate_string(prompt, strip_whitespace=False, max_length=1000)
|
||
default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
||
if recraft_style is None:
|
||
recraft_style = default_style
|
||
|
||
controls_api = None
|
||
if recraft_controls:
|
||
controls_api = recraft_controls.create_api_model()
|
||
|
||
if not negative_prompt:
|
||
negative_prompt = None
|
||
|
||
operation = SynchronousOperation(
|
||
endpoint=ApiEndpoint(
|
||
path="/proxy/recraft/image_generation",
|
||
method=HttpMethod.POST,
|
||
request_model=RecraftImageGenerationRequest,
|
||
response_model=RecraftImageGenerationResponse,
|
||
),
|
||
request=RecraftImageGenerationRequest(
|
||
prompt=prompt,
|
||
negative_prompt=negative_prompt,
|
||
model=RecraftModel.recraftv3,
|
||
size=size,
|
||
n=n,
|
||
style=recraft_style.style,
|
||
substyle=recraft_style.substyle,
|
||
style_id=recraft_style.style_id,
|
||
controls=controls_api,
|
||
),
|
||
auth_token=auth_token,
|
||
)
|
||
response: RecraftImageGenerationResponse = operation.execute()
|
||
images = []
|
||
for data in response.data:
|
||
with handle_recraft_image_output():
|
||
image = bytesio_to_image_tensor(
|
||
download_url_to_bytesio(data.url, timeout=1024)
|
||
)
|
||
if len(image.shape) < 4:
|
||
image = image.unsqueeze(0)
|
||
images.append(image)
|
||
output_image = torch.cat(images, dim=0)
|
||
|
||
return (output_image,)
|
||
|
||
|
||
class RecraftImageToImageNode:
|
||
"""
|
||
Modify image based on prompt and strength.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"image": (IO.IMAGE, ),
|
||
"prompt": (
|
||
IO.STRING,
|
||
{
|
||
"multiline": True,
|
||
"default": "",
|
||
"tooltip": "Prompt for the image generation.",
|
||
},
|
||
),
|
||
"n": (
|
||
IO.INT,
|
||
{
|
||
"default": 1,
|
||
"min": 1,
|
||
"max": 6,
|
||
"tooltip": "The number of images to generate.",
|
||
},
|
||
),
|
||
"strength": (
|
||
IO.FLOAT,
|
||
{
|
||
"default": 0.5,
|
||
"min": 0.0,
|
||
"max": 1.0,
|
||
"step": 0.01,
|
||
"tooltip": "Defines the difference with the original image, should lie in [0, 1], where 0 means almost identical, and 1 means miserable similarity."
|
||
}
|
||
),
|
||
"seed": (
|
||
IO.INT,
|
||
{
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 0xFFFFFFFFFFFFFFFF,
|
||
"control_after_generate": True,
|
||
"tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
||
},
|
||
),
|
||
},
|
||
"optional": {
|
||
"recraft_style": (RecraftIO.STYLEV3,),
|
||
"negative_prompt": (
|
||
IO.STRING,
|
||
{
|
||
"default": "",
|
||
"forceInput": True,
|
||
"tooltip": "An optional text description of undesired elements on an image.",
|
||
},
|
||
),
|
||
"recraft_controls": (
|
||
RecraftIO.CONTROLS,
|
||
{
|
||
"tooltip": "Optional additional controls over the generation via the Recraft Controls node."
|
||
},
|
||
),
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
image: torch.Tensor,
|
||
prompt: str,
|
||
n: int,
|
||
strength: float,
|
||
seed,
|
||
auth_token=None,
|
||
recraft_style: RecraftStyle = None,
|
||
negative_prompt: str = None,
|
||
recraft_controls: RecraftControls = None,
|
||
**kwargs,
|
||
):
|
||
validate_string(prompt, strip_whitespace=False, max_length=1000)
|
||
default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
||
if recraft_style is None:
|
||
recraft_style = default_style
|
||
|
||
controls_api = None
|
||
if recraft_controls:
|
||
controls_api = recraft_controls.create_api_model()
|
||
|
||
if not negative_prompt:
|
||
negative_prompt = None
|
||
|
||
request = RecraftImageGenerationRequest(
|
||
prompt=prompt,
|
||
negative_prompt=negative_prompt,
|
||
model=RecraftModel.recraftv3,
|
||
n=n,
|
||
strength=round(strength, 2),
|
||
style=recraft_style.style,
|
||
substyle=recraft_style.substyle,
|
||
style_id=recraft_style.style_id,
|
||
controls=controls_api,
|
||
)
|
||
|
||
images = []
|
||
total = image.shape[0]
|
||
pbar = ProgressBar(total)
|
||
for i in range(total):
|
||
sub_bytes = handle_recraft_file_request(
|
||
image=image[i],
|
||
path="/proxy/recraft/images/imageToImage",
|
||
request=request,
|
||
auth_token=auth_token,
|
||
)
|
||
with handle_recraft_image_output():
|
||
images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
||
pbar.update(1)
|
||
|
||
images_tensor = torch.cat(images, dim=0)
|
||
return (images_tensor, )
|
||
|
||
|
||
class RecraftImageInpaintingNode:
|
||
"""
|
||
Modify image based on prompt and mask.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"image": (IO.IMAGE, ),
|
||
"mask": (IO.MASK, ),
|
||
"prompt": (
|
||
IO.STRING,
|
||
{
|
||
"multiline": True,
|
||
"default": "",
|
||
"tooltip": "Prompt for the image generation.",
|
||
},
|
||
),
|
||
"n": (
|
||
IO.INT,
|
||
{
|
||
"default": 1,
|
||
"min": 1,
|
||
"max": 6,
|
||
"tooltip": "The number of images to generate.",
|
||
},
|
||
),
|
||
"seed": (
|
||
IO.INT,
|
||
{
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 0xFFFFFFFFFFFFFFFF,
|
||
"control_after_generate": True,
|
||
"tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
||
},
|
||
),
|
||
},
|
||
"optional": {
|
||
"recraft_style": (RecraftIO.STYLEV3,),
|
||
"negative_prompt": (
|
||
IO.STRING,
|
||
{
|
||
"default": "",
|
||
"forceInput": True,
|
||
"tooltip": "An optional text description of undesired elements on an image.",
|
||
},
|
||
),
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
image: torch.Tensor,
|
||
mask: torch.Tensor,
|
||
prompt: str,
|
||
n: int,
|
||
seed,
|
||
auth_token=None,
|
||
recraft_style: RecraftStyle = None,
|
||
negative_prompt: str = None,
|
||
**kwargs,
|
||
):
|
||
validate_string(prompt, strip_whitespace=False, max_length=1000)
|
||
default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
||
if recraft_style is None:
|
||
recraft_style = default_style
|
||
|
||
if not negative_prompt:
|
||
negative_prompt = None
|
||
|
||
request = RecraftImageGenerationRequest(
|
||
prompt=prompt,
|
||
negative_prompt=negative_prompt,
|
||
model=RecraftModel.recraftv3,
|
||
n=n,
|
||
style=recraft_style.style,
|
||
substyle=recraft_style.substyle,
|
||
style_id=recraft_style.style_id,
|
||
)
|
||
|
||
# prepare mask tensor
|
||
mask = resize_mask_to_image(mask, image, allow_gradient=False, add_channel_dim=True)
|
||
|
||
images = []
|
||
total = image.shape[0]
|
||
pbar = ProgressBar(total)
|
||
for i in range(total):
|
||
sub_bytes = handle_recraft_file_request(
|
||
image=image[i],
|
||
mask=mask[i:i+1],
|
||
path="/proxy/recraft/images/inpaint",
|
||
request=request,
|
||
auth_token=auth_token,
|
||
)
|
||
with handle_recraft_image_output():
|
||
images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
||
pbar.update(1)
|
||
|
||
images_tensor = torch.cat(images, dim=0)
|
||
return (images_tensor, )
|
||
|
||
|
||
class RecraftTextToVectorNode:
|
||
"""
|
||
Generates SVG synchronously based on prompt and resolution.
|
||
"""
|
||
|
||
RETURN_TYPES = (RecraftIO.SVG,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"prompt": (
|
||
IO.STRING,
|
||
{
|
||
"multiline": True,
|
||
"default": "",
|
||
"tooltip": "Prompt for the image generation.",
|
||
},
|
||
),
|
||
"substyle": (get_v3_substyles(RecraftStyleV3.vector_illustration),),
|
||
"size": (
|
||
[res.value for res in RecraftImageSize],
|
||
{
|
||
"default": RecraftImageSize.res_1024x1024,
|
||
"tooltip": "The size of the generated image.",
|
||
},
|
||
),
|
||
"n": (
|
||
IO.INT,
|
||
{
|
||
"default": 1,
|
||
"min": 1,
|
||
"max": 6,
|
||
"tooltip": "The number of images to generate.",
|
||
},
|
||
),
|
||
"seed": (
|
||
IO.INT,
|
||
{
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 0xFFFFFFFFFFFFFFFF,
|
||
"control_after_generate": True,
|
||
"tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
||
},
|
||
),
|
||
},
|
||
"optional": {
|
||
"negative_prompt": (
|
||
IO.STRING,
|
||
{
|
||
"default": "",
|
||
"forceInput": True,
|
||
"tooltip": "An optional text description of undesired elements on an image.",
|
||
},
|
||
),
|
||
"recraft_controls": (
|
||
RecraftIO.CONTROLS,
|
||
{
|
||
"tooltip": "Optional additional controls over the generation via the Recraft Controls node."
|
||
},
|
||
),
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
prompt: str,
|
||
substyle: str,
|
||
size: str,
|
||
n: int,
|
||
seed,
|
||
negative_prompt: str = None,
|
||
recraft_controls: RecraftControls = None,
|
||
auth_token=None,
|
||
**kwargs,
|
||
):
|
||
validate_string(prompt, strip_whitespace=False, max_length=1000)
|
||
# create RecraftStyle so strings will be formatted properly (i.e. "None" will become None)
|
||
recraft_style = RecraftStyle(RecraftStyleV3.vector_illustration, substyle=substyle)
|
||
|
||
controls_api = None
|
||
if recraft_controls:
|
||
controls_api = recraft_controls.create_api_model()
|
||
|
||
if not negative_prompt:
|
||
negative_prompt = None
|
||
|
||
operation = SynchronousOperation(
|
||
endpoint=ApiEndpoint(
|
||
path="/proxy/recraft/image_generation",
|
||
method=HttpMethod.POST,
|
||
request_model=RecraftImageGenerationRequest,
|
||
response_model=RecraftImageGenerationResponse,
|
||
),
|
||
request=RecraftImageGenerationRequest(
|
||
prompt=prompt,
|
||
negative_prompt=negative_prompt,
|
||
model=RecraftModel.recraftv3,
|
||
size=size,
|
||
n=n,
|
||
style=recraft_style.style,
|
||
substyle=recraft_style.substyle,
|
||
controls=controls_api,
|
||
),
|
||
auth_token=auth_token,
|
||
)
|
||
response: RecraftImageGenerationResponse = operation.execute()
|
||
svg_data = []
|
||
for data in response.data:
|
||
svg_data.append(download_url_to_bytesio(data.url, timeout=1024))
|
||
|
||
return (SVG(svg_data),)
|
||
|
||
|
||
class RecraftVectorizeImageNode:
|
||
"""
|
||
Generates SVG synchronously from an input image.
|
||
"""
|
||
|
||
RETURN_TYPES = (RecraftIO.SVG,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"image": (IO.IMAGE, ),
|
||
},
|
||
"optional": {
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
image: torch.Tensor,
|
||
auth_token=None,
|
||
**kwargs,
|
||
):
|
||
svgs = []
|
||
total = image.shape[0]
|
||
pbar = ProgressBar(total)
|
||
for i in range(total):
|
||
sub_bytes = handle_recraft_file_request(
|
||
image=image[i],
|
||
path="/proxy/recraft/images/vectorize",
|
||
auth_token=auth_token,
|
||
)
|
||
svgs.append(SVG(sub_bytes))
|
||
pbar.update(1)
|
||
|
||
return (SVG.combine_all(svgs), )
|
||
|
||
|
||
class RecraftReplaceBackgroundNode:
|
||
"""
|
||
Replace background on image, based on provided prompt.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"image": (IO.IMAGE, ),
|
||
"prompt": (
|
||
IO.STRING,
|
||
{
|
||
"multiline": True,
|
||
"default": "",
|
||
"tooltip": "Prompt for the image generation.",
|
||
},
|
||
),
|
||
"n": (
|
||
IO.INT,
|
||
{
|
||
"default": 1,
|
||
"min": 1,
|
||
"max": 6,
|
||
"tooltip": "The number of images to generate.",
|
||
},
|
||
),
|
||
"seed": (
|
||
IO.INT,
|
||
{
|
||
"default": 0,
|
||
"min": 0,
|
||
"max": 0xFFFFFFFFFFFFFFFF,
|
||
"control_after_generate": True,
|
||
"tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
||
},
|
||
),
|
||
},
|
||
"optional": {
|
||
"recraft_style": (RecraftIO.STYLEV3,),
|
||
"negative_prompt": (
|
||
IO.STRING,
|
||
{
|
||
"default": "",
|
||
"forceInput": True,
|
||
"tooltip": "An optional text description of undesired elements on an image.",
|
||
},
|
||
),
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
image: torch.Tensor,
|
||
prompt: str,
|
||
n: int,
|
||
seed,
|
||
auth_token=None,
|
||
recraft_style: RecraftStyle = None,
|
||
negative_prompt: str = None,
|
||
**kwargs,
|
||
):
|
||
default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
||
if recraft_style is None:
|
||
recraft_style = default_style
|
||
|
||
if not negative_prompt:
|
||
negative_prompt = None
|
||
|
||
request = RecraftImageGenerationRequest(
|
||
prompt=prompt,
|
||
negative_prompt=negative_prompt,
|
||
model=RecraftModel.recraftv3,
|
||
n=n,
|
||
style=recraft_style.style,
|
||
substyle=recraft_style.substyle,
|
||
style_id=recraft_style.style_id,
|
||
)
|
||
|
||
images = []
|
||
total = image.shape[0]
|
||
pbar = ProgressBar(total)
|
||
for i in range(total):
|
||
sub_bytes = handle_recraft_file_request(
|
||
image=image[i],
|
||
path="/proxy/recraft/images/replaceBackground",
|
||
request=request,
|
||
auth_token=auth_token,
|
||
)
|
||
images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
||
pbar.update(1)
|
||
|
||
images_tensor = torch.cat(images, dim=0)
|
||
return (images_tensor, )
|
||
|
||
|
||
class RecraftRemoveBackgroundNode:
|
||
"""
|
||
Remove background from image, and return processed image and mask.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE, IO.MASK)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"image": (IO.IMAGE, ),
|
||
},
|
||
"optional": {
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
image: torch.Tensor,
|
||
auth_token=None,
|
||
**kwargs,
|
||
):
|
||
images = []
|
||
total = image.shape[0]
|
||
pbar = ProgressBar(total)
|
||
for i in range(total):
|
||
sub_bytes = handle_recraft_file_request(
|
||
image=image[i],
|
||
path="/proxy/recraft/images/removeBackground",
|
||
auth_token=auth_token,
|
||
)
|
||
images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
||
pbar.update(1)
|
||
|
||
images_tensor = torch.cat(images, dim=0)
|
||
# use alpha channel as masks, in B,H,W format
|
||
masks_tensor = images_tensor[:,:,:,-1:].squeeze(-1)
|
||
return (images_tensor, masks_tensor)
|
||
|
||
|
||
class RecraftCrispUpscaleNode:
|
||
"""
|
||
Upscale image synchronously.
|
||
Enhances a given raster image using ‘crisp upscale’ tool, increasing image resolution, making the image sharper and cleaner.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
RECRAFT_PATH = "/proxy/recraft/images/crispUpscale"
|
||
|
||
@classmethod
|
||
def INPUT_TYPES(s):
|
||
return {
|
||
"required": {
|
||
"image": (IO.IMAGE, ),
|
||
},
|
||
"optional": {
|
||
},
|
||
"hidden": {
|
||
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||
},
|
||
}
|
||
|
||
def api_call(
|
||
self,
|
||
image: torch.Tensor,
|
||
auth_token=None,
|
||
**kwargs,
|
||
):
|
||
images = []
|
||
total = image.shape[0]
|
||
pbar = ProgressBar(total)
|
||
for i in range(total):
|
||
sub_bytes = handle_recraft_file_request(
|
||
image=image[i],
|
||
path=self.RECRAFT_PATH,
|
||
auth_token=auth_token,
|
||
)
|
||
images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
||
pbar.update(1)
|
||
|
||
images_tensor = torch.cat(images, dim=0)
|
||
return (images_tensor,)
|
||
|
||
|
||
class RecraftCreativeUpscaleNode(RecraftCrispUpscaleNode):
|
||
"""
|
||
Upscale image synchronously.
|
||
Enhances a given raster image using ‘creative upscale’ tool, boosting resolution with a focus on refining small details and faces.
|
||
"""
|
||
|
||
RETURN_TYPES = (IO.IMAGE,)
|
||
DESCRIPTION = cleandoc(__doc__ or "") # Handle potential None value
|
||
FUNCTION = "api_call"
|
||
API_NODE = True
|
||
CATEGORY = "api node/image/Recraft"
|
||
|
||
RECRAFT_PATH = "/proxy/recraft/images/creativeUpscale"
|
||
|
||
|
||
# A dictionary that contains all nodes you want to export with their names
|
||
# NOTE: names should be globally unique
|
||
NODE_CLASS_MAPPINGS = {
|
||
"RecraftTextToImageNode": RecraftTextToImageNode,
|
||
"RecraftImageToImageNode": RecraftImageToImageNode,
|
||
"RecraftImageInpaintingNode": RecraftImageInpaintingNode,
|
||
"RecraftTextToVectorNode": RecraftTextToVectorNode,
|
||
"RecraftVectorizeImageNode": RecraftVectorizeImageNode,
|
||
"RecraftRemoveBackgroundNode": RecraftRemoveBackgroundNode,
|
||
"RecraftReplaceBackgroundNode": RecraftReplaceBackgroundNode,
|
||
"RecraftCrispUpscaleNode": RecraftCrispUpscaleNode,
|
||
"RecraftCreativeUpscaleNode": RecraftCreativeUpscaleNode,
|
||
"RecraftStyleV3RealisticImage": RecraftStyleV3RealisticImageNode,
|
||
"RecraftStyleV3DigitalIllustration": RecraftStyleV3DigitalIllustrationNode,
|
||
"RecraftStyleV3LogoRaster": RecraftStyleV3LogoRasterNode,
|
||
"RecraftStyleV3InfiniteStyleLibrary": RecraftStyleInfiniteStyleLibrary,
|
||
"RecraftColorRGB": RecraftColorRGBNode,
|
||
"RecraftControls": RecraftControlsNode,
|
||
"SaveSVG": SaveSVGNode,
|
||
}
|
||
|
||
# A dictionary that contains the friendly/humanly readable titles for the nodes
|
||
NODE_DISPLAY_NAME_MAPPINGS = {
|
||
"RecraftTextToImageNode": "Recraft Text to Image",
|
||
"RecraftImageToImageNode": "Recraft Image to Image",
|
||
"RecraftImageInpaintingNode": "Recraft Image Inpainting",
|
||
"RecraftTextToVectorNode": "Recraft Text to Vector",
|
||
"RecraftVectorizeImageNode": "Recraft Vectorize Image",
|
||
"RecraftRemoveBackgroundNode": "Recraft Remove Background",
|
||
"RecraftReplaceBackgroundNode": "Recraft Replace Background",
|
||
"RecraftCrispUpscaleNode": "Recraft Crisp Upscale Image",
|
||
"RecraftCreativeUpscaleNode": "Recraft Creative Upscale Image",
|
||
"RecraftStyleV3RealisticImage": "Recraft Style - Realistic Image",
|
||
"RecraftStyleV3DigitalIllustration": "Recraft Style - Digital Illustration",
|
||
"RecraftStyleV3LogoRaster": "Recraft Style - Logo Raster",
|
||
"RecraftStyleV3InfiniteStyleLibrary": "Recraft Style - Infinite Style Library",
|
||
"RecraftColorRGB": "Recraft Color RGB",
|
||
"RecraftControls": "Recraft Controls",
|
||
"SaveSVG": "Save SVG",
|
||
}
|