ImageGen / ui /shared /ui_components.py
RioShiina's picture
Upload folder using huggingface_hub
e62785a verified
import gradio as gr
from comfy_integration.nodes import SAMPLER_CHOICES, SCHEDULER_CHOICES
from core.settings import (
MAX_LORAS, LORA_SOURCE_CHOICES, MAX_EMBEDDINGS, MAX_CONDITIONINGS,
MAX_CONTROLNETS, MAX_IPADAPTERS, RESOLUTION_MAP, ARCHITECTURES_CONFIG,
MODEL_MAP_CHECKPOINT, MODEL_TYPE_MAP, FEATURES_CONFIG, ARCH_CATEGORIES_MAP
)
import yaml
import os
from functools import lru_cache
default_model_name = list(MODEL_MAP_CHECKPOINT.keys())[0] if MODEL_MAP_CHECKPOINT else None
default_m_type = MODEL_TYPE_MAP.get(default_model_name, "SDXL") if default_model_name else "SDXL"
default_architectures_dict = ARCHITECTURES_CONFIG.get('architectures', {})
default_arch_model_type = default_architectures_dict.get(default_m_type, {}).get("model_type", default_m_type.lower().replace(" ", "").replace(".", ""))
default_arch_features = FEATURES_CONFIG.get(default_arch_model_type, FEATURES_CONFIG.get('default', {}))
default_enabled_chains = default_arch_features.get('enabled_chains', [])
@lru_cache(maxsize=1)
def get_ipadapter_config_from_yaml():
try:
_PROJECT_ROOT = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
_IPADAPTER_LIST_PATH = os.path.join(_PROJECT_ROOT, 'yaml', 'ipadapter.yaml')
with open(_IPADAPTER_LIST_PATH, 'r', encoding='utf-8') as f:
config = yaml.safe_load(f)
return config
except Exception as e:
print(f"Warning: Could not load ipadapter.yaml for UI components: {e}")
return {}
def get_ipadapter_presets(arch="SDXL"):
config = get_ipadapter_config_from_yaml()
presets = []
if config:
std_presets = config.get("IPAdapter_presets", {}).get(arch, [])
face_presets = config.get("IPAdapter_FaceID_presets", {}).get(arch, [])
if std_presets:
presets.extend(std_presets)
if face_presets:
presets.extend(face_presets)
return presets if presets else ["STANDARD (medium strength)"]
def create_model_architecture_filter_ui(prefix):
components = {}
ordered_architectures = ARCHITECTURES_CONFIG.get("architecture_order", [])
choices = ["ALL"] + ordered_architectures
components[f'model_arch_{prefix}'] = gr.Radio(
label="Model Architecture",
choices=choices,
value="ALL",
interactive=True
)
return components
def create_category_filter_ui(prefix):
valid_cats = list(set(cat for cats in ARCH_CATEGORIES_MAP.values() for cat in cats))
cat_choices = ["ALL"] + sorted(valid_cats)
components = {}
components[f'model_cat_{prefix}'] = gr.Dropdown(
label="Filter Models",
choices=cat_choices,
value="ALL",
interactive=True,
scale=1
)
return components
def create_base_parameter_ui(prefix, defaults=None):
if defaults is None:
defaults = {}
components = {}
with gr.Row():
components[f'aspect_ratio_{prefix}'] = gr.Dropdown(
label="Aspect Ratio",
choices=list(RESOLUTION_MAP.get('sdxl', {}).keys()),
value="1:1 (Square)",
interactive=True
)
with gr.Row():
components[f'width_{prefix}'] = gr.Number(label="Width", value=defaults.get('w', 1024), interactive=True)
components[f'height_{prefix}'] = gr.Number(label="Height", value=defaults.get('h', 1024), interactive=True)
with gr.Row():
components[f'sampler_{prefix}'] = gr.Dropdown(label="Sampler", choices=SAMPLER_CHOICES, value=SAMPLER_CHOICES[0])
components[f'scheduler_{prefix}'] = gr.Dropdown(label="Scheduler", choices=SCHEDULER_CHOICES, value='normal' if 'normal' in SCHEDULER_CHOICES else SCHEDULER_CHOICES[0])
with gr.Row():
components[f'steps_{prefix}'] = gr.Slider(label="Steps", minimum=1, maximum=100, step=1, value=28)
components[f'cfg_{prefix}'] = gr.Slider(label="CFG Scale", minimum=1.0, maximum=20.0, step=0.1, value=7.5)
with gr.Row():
components[f'seed_{prefix}'] = gr.Number(label="Seed (-1 for random)", value=-1, precision=0)
components[f'batch_size_{prefix}'] = gr.Slider(label="Batch Size", minimum=1, maximum=16, step=1, value=1)
with gr.Row():
components[f'clip_skip_{prefix}'] = gr.Slider(label="Clip Skip", minimum=1, maximum=2, step=1, value=1, visible=False, interactive=True)
components[f'guidance_{prefix}'] = gr.Slider(label="Guidance (FLUX)", minimum=1.0, maximum=10.0, step=0.1, value=3.5, visible=False, interactive=True)
components[f'zero_gpu_{prefix}'] = gr.Number(label="ZeroGPU Duration (s)", value=None, placeholder="Default: 60s, Max: 120s", info="Optional: Set how long to reserve the GPU.")
return components
def create_lora_settings_ui(prefix: str):
components = {}
lora_rows, lora_sources, lora_ids, lora_scales, lora_uploads = [], [], [], [], []
with gr.Accordion("LoRA Settings", open=False, visible=('lora' in default_enabled_chains)) as lora_accordion:
components[f'lora_accordion_{prefix}'] = lora_accordion
gr.Markdown("πŸ’‘ **Tip:** When downloading from Civitai, please use the **Version ID**, not the Model ID. You can find the Version ID in the URL (e.g., `civitai.com/models/123?modelVersionId=456`) or under the model's download button.")
components[f'lora_count_state_{prefix}'] = gr.State(1)
for i in range(MAX_LORAS):
with gr.Row(visible=i==0) as row:
source = gr.Dropdown(label=f"LoRA Source {i+1}", choices=LORA_SOURCE_CHOICES, value=LORA_SOURCE_CHOICES[0], scale=1)
lora_id = gr.Textbox(label=f"Civitai Version ID / File", placeholder="Civitai Version ID or Filename", scale=2, type="text")
scale = gr.Slider(label=f"Scale", minimum=-2.0, maximum=2.0, step=0.05, value=0.8, scale=1)
upload = gr.UploadButton(label="Upload", file_types=[".safetensors"], scale=1)
lora_rows.append(row)
lora_sources.append(source)
lora_ids.append(lora_id)
lora_scales.append(scale)
lora_uploads.append(upload)
with gr.Row():
components[f'add_lora_button_{prefix}'] = gr.Button("Add LoRA", variant="secondary")
components[f'delete_lora_button_{prefix}'] = gr.Button("Remove LoRA", variant="secondary", visible=False)
components[f'lora_rows_{prefix}'] = lora_rows
components[f'lora_sources_{prefix}'] = lora_sources
components[f'lora_ids_{prefix}'] = lora_ids
components[f'lora_scales_{prefix}'] = lora_scales
components[f'lora_uploads_{prefix}'] = lora_uploads
all_lora_components_flat = []
for i in range(MAX_LORAS):
all_lora_components_flat.extend([lora_sources[i], lora_ids[i], lora_scales[i], lora_uploads[i]])
components[f'all_lora_components_flat_{prefix}'] = all_lora_components_flat
return components
def create_controlnet_ui(prefix: str, max_units=MAX_CONTROLNETS):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("ControlNet Settings", open=False, visible=('controlnet' in default_enabled_chains)) as accordion:
components[key('controlnet_accordion')] = accordion
cn_rows, images, series, types, strengths, filepaths = [], [], [], [], [], []
components.update({
key('controlnet_rows'): cn_rows,
key('controlnet_images'): images,
key('controlnet_series'): series,
key('controlnet_types'): types,
key('controlnet_strengths'): strengths,
key('controlnet_filepaths'): filepaths
})
for i in range(max_units):
with gr.Row(visible=(i < 1)) as row:
with gr.Column(scale=1):
images.append(gr.Image(label=f"Control Image {i+1}", type="pil", sources=["upload"], height=256))
with gr.Column(scale=2):
types.append(gr.Dropdown(label="Type", choices=[], interactive=True, allow_custom_value=True))
series.append(gr.Dropdown(label="Series", choices=[], interactive=True, allow_custom_value=True))
strengths.append(gr.Slider(label="Strength", minimum=0.0, maximum=2.0, step=0.05, value=1.0, interactive=True))
filepaths.append(gr.State(None))
cn_rows.append(row)
with gr.Row():
components[key('add_controlnet_button')] = gr.Button("✚ Add ControlNet")
components[key('delete_controlnet_button')] = gr.Button("βž– Delete ControlNet", visible=False)
components[key('controlnet_count_state')] = gr.State(1)
all_cn_components_flat = []
for i in range(max_units):
all_cn_components_flat.extend([
images[i], types[i], series[i], strengths[i], filepaths[i]
])
components[key('all_controlnet_components_flat')] = all_cn_components_flat
return components
def create_diffsynth_controlnet_ui(prefix: str, max_units=MAX_CONTROLNETS):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("DiffSynth ControlNet Settings", open=False, visible=('controlnet_model_patch' in default_enabled_chains)) as accordion:
components[key('diffsynth_controlnet_accordion')] = accordion
cn_rows, images, series, types, strengths, filepaths = [], [], [], [], [], []
components.update({
key('diffsynth_controlnet_rows'): cn_rows,
key('diffsynth_controlnet_images'): images,
key('diffsynth_controlnet_series'): series,
key('diffsynth_controlnet_types'): types,
key('diffsynth_controlnet_strengths'): strengths,
key('diffsynth_controlnet_filepaths'): filepaths
})
for i in range(max_units):
with gr.Row(visible=(i < 1)) as row:
with gr.Column(scale=1):
images.append(gr.Image(label=f"Control Image {i+1}", type="pil", sources=["upload"], height=256))
with gr.Column(scale=2):
types.append(gr.Dropdown(label="Type", choices=[], interactive=True, allow_custom_value=True))
series.append(gr.Dropdown(label="Series", choices=[], interactive=True, allow_custom_value=True))
strengths.append(gr.Slider(label="Strength", minimum=0.0, maximum=2.0, step=0.05, value=1.0, interactive=True))
filepaths.append(gr.State(None))
cn_rows.append(row)
with gr.Row():
components[key('add_diffsynth_controlnet_button')] = gr.Button("✚ Add DiffSynth ControlNet")
components[key('delete_diffsynth_controlnet_button')] = gr.Button("βž– Delete DiffSynth ControlNet", visible=False)
components[key('diffsynth_controlnet_count_state')] = gr.State(1)
all_cn_components_flat = []
for i in range(max_units):
all_cn_components_flat.extend([
images[i], types[i], series[i], strengths[i], filepaths[i]
])
components[key('all_diffsynth_controlnet_components_flat')] = all_cn_components_flat
return components
def create_ipadapter_ui(prefix: str, max_units=MAX_IPADAPTERS):
components = {}
key = lambda name: f"{name}_{prefix}"
sdxl_presets = get_ipadapter_presets("SDXL")
default_preset = sdxl_presets[0] if sdxl_presets else None
with gr.Accordion("IPAdapter Settings", open=False, visible=('ipadapter' in default_enabled_chains)) as accordion:
components[key('ipadapter_accordion')] = accordion
gr.Markdown("Powered by [cubiq/ComfyUI_IPAdapter_plus](https://github.com/cubiq/ComfyUI_IPAdapter_plus).")
with gr.Row():
components[key('ipadapter_final_preset')] = gr.Dropdown(
label="Preset (for all images)",
choices=sdxl_presets,
value=default_preset,
interactive=True,
allow_custom_value=True
)
components[key('ipadapter_embeds_scaling')] = gr.Dropdown(
label="Embeds Scaling",
choices=['V only', 'K+V', 'K+V w/ C penalty', 'K+mean(V) w/ C penalty'],
value='V only',
interactive=True
)
with gr.Row():
components[key('ipadapter_combine_method')] = gr.Dropdown(
label="Combine Method",
choices=["concat", "add", "subtract", "average", "norm average", "max", "min"],
value="concat",
interactive=True
)
components[key('ipadapter_final_weight')] = gr.Slider(label="Final Weight", minimum=0.0, maximum=2.0, step=0.05, value=1.0, interactive=True)
components[key('ipadapter_final_lora_strength')] = gr.Slider(label="Final LoRA Strength", minimum=0.0, maximum=2.0, step=0.05, value=0.6, interactive=True, visible=False)
gr.Markdown("---")
ipa_rows, images, weights, lora_strengths = [], [], [], []
components.update({
key('ipadapter_rows'): ipa_rows,
key('ipadapter_images'): images,
key('ipadapter_weights'): weights,
key('ipadapter_lora_strengths'): lora_strengths
})
for i in range(max_units):
with gr.Row(visible=(i < 1)) as row:
with gr.Column(scale=1):
images.append(gr.Image(label=f"IPAdapter Image {i+1}", type="pil", sources=["upload"], height=256))
with gr.Column(scale=2):
weights.append(gr.Slider(label="Weight", minimum=0.0, maximum=2.0, step=0.05, value=1.0, interactive=True))
lora_strengths.append(gr.Slider(label="LoRA Strength", minimum=0.0, maximum=2.0, step=0.05, value=0.6, interactive=True, visible=False))
ipa_rows.append(row)
with gr.Row():
components[key('add_ipadapter_button')] = gr.Button("✚ Add IPAdapter")
components[key('delete_ipadapter_button')] = gr.Button("βž– Delete IPAdapter", visible=False)
components[key('ipadapter_count_state')] = gr.State(1)
all_ipa_components_flat = images + weights + lora_strengths
all_ipa_components_flat += [
components[key('ipadapter_final_preset')],
components[key('ipadapter_final_weight')],
components[key('ipadapter_final_lora_strength')],
components[key('ipadapter_embeds_scaling')],
components[key('ipadapter_combine_method')],
]
components[key('all_ipadapter_components_flat')] = all_ipa_components_flat
return components
def create_flux1_ipadapter_ui(prefix: str, max_units=MAX_IPADAPTERS):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("IPAdapter Settings (FLUX.1)", open=False, visible=('flux1_ipadapter' in default_enabled_chains)) as accordion:
components[key('flux1_ipadapter_accordion')] = accordion
ipa_rows, images, weights, start_percents, end_percents = [], [], [], [], []
components.update({
key('flux1_ipadapter_rows'): ipa_rows,
key('flux1_ipadapter_images'): images,
key('flux1_ipadapter_weights'): weights,
key('flux1_ipadapter_start_percents'): start_percents,
key('flux1_ipadapter_end_percents'): end_percents,
})
for i in range(max_units):
with gr.Row(visible=(i < 1)) as row:
with gr.Column(scale=1):
images.append(gr.Image(label=f"IPAdapter Image {i+1}", type="pil", sources=["upload"], height=256))
with gr.Column(scale=2):
weights.append(gr.Slider(label="Weight", minimum=0.0, maximum=2.0, step=0.05, value=0.6, interactive=True))
with gr.Row():
start_percents.append(gr.Slider(label="Start At", minimum=0.0, maximum=1.0, step=0.01, value=0.0, interactive=True))
end_percents.append(gr.Slider(label="End At", minimum=0.0, maximum=1.0, step=0.01, value=0.6, interactive=True))
ipa_rows.append(row)
with gr.Row():
components[key('add_flux1_ipadapter_button')] = gr.Button("✚ Add IPAdapter (FLUX)")
components[key('delete_flux1_ipadapter_button')] = gr.Button("βž– Delete IPAdapter (FLUX)", visible=False)
components[key('flux1_ipadapter_count_state')] = gr.State(1)
all_flux1_ipa_components_flat = images + weights + start_percents + end_percents
components[key('all_flux1_ipadapter_components_flat')] = all_flux1_ipa_components_flat
return components
def create_sd3_ipadapter_ui(prefix: str, max_units=MAX_IPADAPTERS):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("IPAdapter Settings (SD3)", open=False, visible=('sd3_ipadapter' in default_enabled_chains)) as accordion:
components[key('sd3_ipadapter_accordion')] = accordion
ipa_rows, images, weights, start_percents, end_percents = [], [], [], [], []
components.update({
key('sd3_ipadapter_rows'): ipa_rows,
key('sd3_ipadapter_images'): images,
key('sd3_ipadapter_weights'): weights,
key('sd3_ipadapter_start_percents'): start_percents,
key('sd3_ipadapter_end_percents'): end_percents,
})
for i in range(max_units):
with gr.Row(visible=(i < 1)) as row:
with gr.Column(scale=1):
images.append(gr.Image(label=f"IPAdapter Image {i+1}", type="pil", sources=["upload"], height=256))
with gr.Column(scale=2):
weights.append(gr.Slider(label="Weight", minimum=0.0, maximum=2.0, step=0.05, value=0.5, interactive=True))
with gr.Row():
start_percents.append(gr.Slider(label="Start At", minimum=0.0, maximum=1.0, step=0.01, value=0.0, interactive=True))
end_percents.append(gr.Slider(label="End At", minimum=0.0, maximum=1.0, step=0.01, value=1.0, interactive=True))
ipa_rows.append(row)
with gr.Row():
components[key('add_sd3_ipadapter_button')] = gr.Button("✚ Add IPAdapter (SD3)")
components[key('delete_sd3_ipadapter_button')] = gr.Button("βž– Delete IPAdapter (SD3)", visible=False)
components[key('sd3_ipadapter_count_state')] = gr.State(1)
all_sd3_ipa_components_flat = images + weights + start_percents + end_percents
components[key('all_sd3_ipadapter_components_flat')] = all_sd3_ipa_components_flat
return components
def create_style_ui(prefix: str):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("Style Settings (FLUX.1)", open=False, visible=('style' in default_enabled_chains)) as accordion:
components[key('style_accordion')] = accordion
style_rows, images, strengths = [], [], []
components.update({
key('style_rows'): style_rows,
key('style_images'): images,
key('style_strengths'): strengths
})
for i in range(5):
with gr.Row(visible=(i < 1)) as row:
with gr.Column(scale=1):
images.append(gr.Image(label=f"Style Image {i+1}", type="pil", sources=["upload"], height=256))
with gr.Column(scale=2):
strengths.append(gr.Slider(label="Strength", minimum=0.0, maximum=2.0, step=0.05, value=1.0, interactive=True))
style_rows.append(row)
with gr.Row():
components[key('add_style_button')] = gr.Button("✚ Add Style (FLUX)")
components[key('delete_style_button')] = gr.Button("βž– Delete Style (FLUX)", visible=False)
components[key('style_count_state')] = gr.State(1)
all_style_components_flat = images + strengths
components[key('all_style_components_flat')] = all_style_components_flat
return components
def create_embedding_ui(prefix: str):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("Embedding Settings", open=False, visible=('embedding' in default_enabled_chains)) as accordion:
components[key('embedding_accordion')] = accordion
gr.Markdown("πŸ’‘ **Tip:** Embeddings are automatically added to your prompt using `embedding:filename` syntax. When downloading from Civitai, please use the **Version ID**, not the Model ID. You can find the Version ID in the URL (e.g., `civitai.com/models/123?modelVersionId=456`) or under the model's download button. For instance, using the Version ID `456` from the example above would automatically append `embedding:civitai_456` to your positive prompt.")
embedding_rows, sources, ids, files, upload_buttons = [], [], [], [], []
components.update({
key('embedding_rows'): embedding_rows,
key('embeddings_sources'): sources,
key('embeddings_ids'): ids,
key('embeddings_files'): files,
key('embeddings_uploads'): upload_buttons
})
for i in range(MAX_EMBEDDINGS):
with gr.Row(visible=(i < 1)) as row:
sources.append(gr.Dropdown(label=f"Embedding Source {i+1}", choices=LORA_SOURCE_CHOICES, value="Civitai", scale=1, interactive=True))
ids.append(gr.Textbox(label="Civitai Version ID / File", placeholder="Civitai Version ID or Filename", scale=3, interactive=True, type="text"))
upload_btn = gr.UploadButton("Upload", file_types=[".safetensors"], scale=1)
files.append(gr.State(None))
upload_buttons.append(upload_btn)
embedding_rows.append(row)
with gr.Row():
components[key('add_embedding_button')] = gr.Button("✚ Add Embedding")
components[key('delete_embedding_button')] = gr.Button("βž– Delete Embedding", visible=False)
components[key('embedding_count_state')] = gr.State(1)
all_embedding_components_flat = []
for i in range(MAX_EMBEDDINGS):
all_embedding_components_flat.extend([sources[i], ids[i], files[i]])
components[key('all_embedding_components_flat')] = all_embedding_components_flat
return components
def create_conditioning_ui(prefix: str):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("Conditioning Settings", open=False, visible=('conditioning' in default_enabled_chains)) as accordion:
components[key('conditioning_accordion')] = accordion
gr.Markdown("πŸ’‘ **Tip:** Define rectangular areas and assign specific prompts to them. Coordinates (X, Y) start from the top-left corner.")
cond_rows, prompts, widths, heights, xs, ys, strengths = [], [], [], [], [], [], []
components.update({
key('conditioning_rows'): cond_rows,
key('conditioning_prompts'): prompts,
key('conditioning_widths'): widths,
key('conditioning_heights'): heights,
key('conditioning_xs'): xs,
key('conditioning_ys'): ys,
key('conditioning_strengths'): strengths
})
for i in range(MAX_CONDITIONINGS):
with gr.Column(visible=(i < 1)) as row_wrapper:
prompts.append(gr.Textbox(label=f"Area Prompt {i+1}", lines=2, interactive=True))
with gr.Row():
xs.append(gr.Number(label="X", value=0, interactive=True, step=8, scale=1))
ys.append(gr.Number(label="Y", value=0, interactive=True, step=8, scale=1))
widths.append(gr.Number(label="Width", value=512, interactive=True, step=8, scale=1))
heights.append(gr.Number(label="Height", value=512, interactive=True, step=8, scale=1))
strengths.append(gr.Slider(label="Strength", minimum=0.1, maximum=2.0, step=0.05, value=1.0, interactive=True, scale=2))
cond_rows.append(row_wrapper)
with gr.Row():
components[key('add_conditioning_button')] = gr.Button("✚ Add Area")
components[key('delete_conditioning_button')] = gr.Button("βž– Delete Area", visible=False)
components[key('conditioning_count_state')] = gr.State(1)
all_cond_components_flat = prompts + widths + heights + xs + ys + strengths
components[key('all_conditioning_components_flat')] = all_cond_components_flat
return components
def create_vae_override_ui(prefix: str):
components = {}
key = lambda name: f"{name}_{prefix}"
source_choices = ["None"] + LORA_SOURCE_CHOICES
with gr.Accordion("VAE Settings (Override)", open=False) as accordion:
components[key('vae_accordion')] = accordion
gr.Markdown("πŸ’‘ **Tip:** When downloading from Civitai, please use the **Version ID**, not the Model ID. You can find the Version ID in the URL (e.g., `civitai.com/models/123?modelVersionId=456`) or under the model's download button.")
with gr.Row():
components[key('vae_source')] = gr.Dropdown(
label="VAE Source",
choices=source_choices,
value="None",
scale=1,
interactive=True
)
components[key('vae_id')] = gr.Textbox(
label="Civitai Version ID / File",
placeholder="Civitai Version ID or Filename",
scale=3,
interactive=True,
type="text"
)
upload_btn = gr.UploadButton(
"Upload",
file_types=[".safetensors"],
scale=1
)
components[key('vae_upload_button')] = upload_btn
components[key('vae_file')] = gr.State(None)
return components
def create_reference_latent_ui(prefix: str, max_units=10):
components = {}
key = lambda name: f"{name}_{prefix}"
with gr.Accordion("Reference Edit Settings", open=False, visible=('reference_latent' in default_enabled_chains)) as ref_accordion:
components[key('reference_latent_accordion')] = ref_accordion
gr.Markdown("πŸ’‘ **Tip:** For multimodal models (like FLUX.2 or OmniGen), this feature enables powerful editing and combining capabilities. In txt2img mode, adding a single reference image performs an **Image Edit**, while adding multiple images performs an **Image Combine**.")
ref_image_groups = []
ref_image_inputs = []
with gr.Row():
for i in range(max_units):
with gr.Column(visible=(i < 1), min_width=160) as img_col:
img_comp = gr.Image(type="pil", label=f"Ref. {i+1}", sources=["upload"], height=150)
ref_image_groups.append(img_col)
ref_image_inputs.append(img_comp)
components[key('reference_latent_rows')] = ref_image_groups
components[key('reference_latent_images')] = ref_image_inputs
with gr.Row():
components[key('add_reference_latent_button')] = gr.Button("✚ Add Reference Image")
components[key('delete_reference_latent_button')] = gr.Button("βž– Delete Reference Image", visible=False)
components[key('reference_latent_count_state')] = gr.State(1)
components[key('all_reference_latent_components_flat')] = ref_image_inputs
return components