Professional Noob
Update app.py
fb1c443 verified
Raw
History Blame Contribute Delete
36.7 kB
import os
import re
import gc
import traceback
import gradio as gr
import numpy as np
import spaces
import torch
import random
from PIL import Image
from typing import Iterable, Optional
from huggingface_hub import hf_hub_download
from safetensors.torch import load_file as safetensors_load_file
from gradio.themes import Soft
from gradio.themes.utils import colors, fonts, sizes
# ============================================================
# Theme
# ============================================================
colors.orange_red = colors.Color(
name="orange_red",
c50="#FFF0E5",
c100="#FFE0CC",
c200="#FFC299",
c300="#FFA366",
c400="#FF8533",
c500="#FF4500",
c600="#E63E00",
c700="#CC3700",
c800="#B33000",
c900="#992900",
c950="#802200",
)
class OrangeRedTheme(Soft):
def __init__(
self,
*,
primary_hue: colors.Color | str = colors.gray,
secondary_hue: colors.Color | str = colors.orange_red,
neutral_hue: colors.Color | str = colors.slate,
text_size: sizes.Size | str = sizes.text_lg,
font: fonts.Font | str | Iterable[fonts.Font | str] = (
fonts.GoogleFont("Outfit"),
"Arial",
"sans-serif",
),
font_mono: fonts.Font | str | Iterable[fonts.Font | str] = (
fonts.GoogleFont("IBM Plex Mono"),
"ui-monospace",
"monospace",
),
):
super().__init__(
primary_hue=primary_hue,
secondary_hue=secondary_hue,
neutral_hue=neutral_hue,
text_size=text_size,
font=font,
font_mono=font_mono,
)
super().set(
background_fill_primary="*primary_50",
background_fill_primary_dark="*primary_900",
body_background_fill="linear-gradient(135deg, *primary_200, *primary_100)",
body_background_fill_dark="linear-gradient(135deg, *primary_900, *primary_800)",
button_primary_text_color="white",
button_primary_text_color_hover="white",
button_primary_background_fill="linear-gradient(90deg, *secondary_500, *secondary_600)",
button_primary_background_fill_hover="linear-gradient(90deg, *secondary_600, *secondary_700)",
button_primary_background_fill_dark="linear-gradient(90deg, *secondary_600, *secondary_700)",
button_primary_background_fill_hover_dark="linear-gradient(90deg, *secondary_500, *secondary_600)",
button_secondary_text_color="black",
button_secondary_text_color_hover="white",
button_secondary_background_fill="linear-gradient(90deg, *primary_300, *primary_300)",
button_secondary_background_fill_hover="linear-gradient(90deg, *primary_400, *primary_400)",
button_secondary_background_fill_dark="linear-gradient(90deg, *primary_500, *primary_600)",
button_secondary_background_fill_hover_dark="linear-gradient(90deg, *primary_500, *primary_500)",
slider_color="*secondary_500",
slider_color_dark="*secondary_600",
block_title_text_weight="600",
block_border_width="3px",
block_shadow="*shadow_drop_lg",
button_primary_shadow="*shadow_drop_lg",
button_large_padding="11px",
color_accent_soft="*primary_100",
block_label_background_fill="*primary_200",
)
orange_red_theme = OrangeRedTheme()
# ============================================================
# Device
# ============================================================
device = torch.device("cuda" if torch.cuda.is_available() else "cpu")
print("CUDA_VISIBLE_DEVICES=", os.environ.get("CUDA_VISIBLE_DEVICES"))
print("torch.__version__ =", torch.__version__)
print("torch.version.cuda =", torch.version.cuda)
print("cuda available:", torch.cuda.is_available())
print("cuda device count:", torch.cuda.device_count())
if torch.cuda.is_available():
print("current device:", torch.cuda.current_device())
print("device name:", torch.cuda.get_device_name(torch.cuda.current_device()))
print("Using device:", device)
# ============================================================
# AIO version (Space variable)
# ============================================================
AIO_REPO_ID = "Pr0f3ssi0n4ln00b/Phr00t-Qwen-Rapid-AIO"
DEFAULT_AIO_VERSION = "v19"
_VER_RE = re.compile(r"^v\d+$")
_DIGITS_RE = re.compile(r"^\d+$")
def _normalize_version(raw: str) -> Optional[str]:
if raw is None:
return None
s = str(raw).strip()
if not s:
return None
if _VER_RE.fullmatch(s):
return s
# forgiving: allow "21" -> "v21"
if _DIGITS_RE.fullmatch(s):
return f"v{s}"
return None
_AIO_ENV_RAW = os.environ.get("AIO_VERSION", "")
_AIO_ENV_NORM = _normalize_version(_AIO_ENV_RAW)
AIO_VERSION = _AIO_ENV_NORM or DEFAULT_AIO_VERSION
AIO_VERSION_SOURCE = "env" if _AIO_ENV_NORM else "default(v19)"
print(f"AIO_VERSION (env raw) = {_AIO_ENV_RAW!r}")
print(f"AIO_VERSION (normalized) = {_AIO_ENV_NORM!r}")
print(f"Using AIO_VERSION = {AIO_VERSION} ({AIO_VERSION_SOURCE})")
# ============================================================
# Pipeline
# ============================================================
from diffusers import FlowMatchEulerDiscreteScheduler # noqa: F401
from qwenimage.pipeline_qwenimage_edit_plus import QwenImageEditPlusPipeline
from qwenimage.transformer_qwenimage import QwenImageTransformer2DModel
from qwenimage.qwen_fa3_processor import QwenDoubleStreamAttnProcessorFA3
dtype = torch.bfloat16
def _load_pipe_with_version(version: str) -> QwenImageEditPlusPipeline:
sub = f"{version}/transformer"
print(f"📦 Loading AIO transformer: {AIO_REPO_ID} / {sub}")
p = QwenImageEditPlusPipeline.from_pretrained(
"Qwen/Qwen-Image-Edit-2511",
transformer=QwenImageTransformer2DModel.from_pretrained(
AIO_REPO_ID,
subfolder=sub,
torch_dtype=dtype,
device_map="cuda",
),
torch_dtype=dtype,
).to(device)
return p
# Forgiving load: try env/default version, fallback to v19 if it fails
try:
pipe = _load_pipe_with_version(AIO_VERSION)
except Exception as e:
print("❌ Failed to load requested AIO_VERSION. Falling back to v19.")
print("---- exception ----")
print(traceback.format_exc())
print("-------------------")
AIO_VERSION = DEFAULT_AIO_VERSION
AIO_VERSION_SOURCE = "fallback_to_v19"
pipe = _load_pipe_with_version(AIO_VERSION)
# Apply FA3 Optimization
try:
pipe.transformer.set_attn_processor(QwenDoubleStreamAttnProcessorFA3())
print("Flash Attention 3 Processor set successfully.")
except Exception as e:
print(f"Warning: Could not set FA3 processor: {e}")
MAX_SEED = np.iinfo(np.int32).max
# ============================================================
# LoRA adapters + presets
# ============================================================
NONE_LORA = "None"
ADAPTER_SPECS = {
"Consistance": {
"type": "single",
"repo": "Pr0f3ssi0n4ln00b/QIE_2511_Consistency_Lora",
"weights": "qe2511_consis_alpha_patched.safetensors",
"adapter_name": "Consistency",
"strength": 0.6,
},
"Semirealistic-photo-detailer": {
"type": "single",
"repo": "rzgar/Qwen-Image-Edit-semi-realistic-detailer",
"weights": "Qwen-Image-Edit-Anime-Semi-Realistic-Detailer-v1.safetensors",
"adapter_name": "semirealistic",
"strength": 1.0,
},
"AnyPose": {
"type": "package",
"requires_two_images": True,
"image2_label": "Upload Pose Reference (Image 2)",
"parts": [
{
"repo": "lilylilith/AnyPose",
"weights": "2511-AnyPose-base-000006250.safetensors",
"adapter_name": "anypose-base",
"strength": 0.7,
},
{
"repo": "lilylilith/AnyPose",
"weights": "2511-AnyPose-helper-00006000.safetensors",
"adapter_name": "anypose-helper",
"strength": 0.7,
},
],
},
"Any2Real_2601": {
"type": "single",
"repo": "lrzjason/Anything2Real_2601",
"weights": "anything2real_2601_A_final_patched.safetensors",
"adapter_name": "photoreal",
"strength": 1.0,
},
"Hyperrealistic-Portrait": {
"type": "single",
"repo": "prithivMLmods/Qwen-Image-Edit-2511-Hyper-Realistic-Portrait",
"weights": "HRP_20.safetensors",
"adapter_name": "HRPortrait",
"strength": 1.0,
},
"Ultrarealistic-Portrait": {
"type": "single",
"repo": "prithivMLmods/Qwen-Image-Edit-2511-Ultra-Realistic-Portrait",
"weights": "URP_20.safetensors",
"adapter_name": "URPortrait",
"strength": 1.0,
},
"BFS-Best-FaceSwap": {
"type": "single",
"requires_two_images": True,
"image2_label": "Upload Head/Face Donor (Image 2)",
"repo": "Alissonerdx/BFS-Best-Face-Swap",
"weights": "bfs_head_v5_2511_original.safetensors",
"adapter_name": "BFS-Best-Faceswap",
"strength": 1.0,
"needs_alpha_fix": True, # <-- fixes KeyError 'img_in.alpha'
},
"BFS-Best-FaceSwap-merge": {
"type": "single",
"requires_two_images": True,
"image2_label": "Upload Head/Face Donor (Image 2)",
"repo": "Alissonerdx/BFS-Best-Face-Swap",
"weights": "bfs_head_v5_2511_merged_version_rank_32_fp32.safetensors",
"adapter_name": "BFS-Best-Faceswap-merge",
"strength": 1.1,
"needs_alpha_fix": True, # <-- fixes KeyError 'img_in.alpha'
},
"F2P": {
"type": "single",
"repo": "DiffSynth-Studio/Qwen-Image-Edit-F2P",
"weights": "edit_0928_lora_step40000.safetensors",
"adapter_name": "F2P",
"strength": 1.0,
},
"Multiple-Angles": {
"type": "single",
"repo": "dx8152/Qwen-Edit-2509-Multiple-angles",
"weights": "镜头转换.safetensors",
"adapter_name": "multiple-angles",
"strength": 1.0,
},
"Light-Restoration": {
"type": "single",
"repo": "dx8152/Qwen-Image-Edit-2509-Light_restoration",
"weights": "移除光影.safetensors",
"adapter_name": "light-restoration",
"strength": 1.0,
},
"Relight": {
"type": "single",
"repo": "dx8152/Qwen-Image-Edit-2509-Relight",
"weights": "Qwen-Edit-Relight.safetensors",
"adapter_name": "relight",
"strength": 1.0,
},
"Multi-Angle-Lighting": {
"type": "single",
"repo": "dx8152/Qwen-Edit-2509-Multi-Angle-Lighting",
"weights": "多角度灯光-251116.safetensors",
"adapter_name": "multi-angle-lighting",
"strength": 1.0,
},
"Edit-Skin": {
"type": "single",
"repo": "tlennon-ie/qwen-edit-skin",
"weights": "qwen-edit-skin_1.1_000002750.safetensors",
"adapter_name": "edit-skin",
"strength": 1.0,
},
"Next-Scene": {
"type": "single",
"repo": "lovis93/next-scene-qwen-image-lora-2509",
"weights": "next-scene_lora-v2-3000.safetensors",
"adapter_name": "next-scene",
"strength": 1.0,
},
"Flat-Log": {
"type": "single",
"repo": "tlennon-ie/QwenEdit2509-FlatLogColor",
"weights": "QwenEdit2509-FlatLogColor.safetensors",
"adapter_name": "flat-log",
"strength": 1.0,
},
"Upscale-Image": {
"type": "single",
"repo": "vafipas663/Qwen-Edit-2509-Upscale-LoRA",
"weights": "qwen-edit-enhance_64-v3_000001000.safetensors",
"adapter_name": "upscale-image",
"strength": 1.0,
},
"Upscale2K": {
"type": "single",
"repo": "valiantcat/Qwen-Image-Edit-2509-Upscale2K",
"weights": "qwen_image_edit_2509_upscale.safetensors",
"adapter_name": "upscale-2k",
"strength": 1.0,
"target_long_edge": 2048,
},
}
LORA_PRESET_PROMPTS = {
"Any2Real_2601": "change the picture 1 to realistic photograph",
"Semirealistic-photo-detailer": "transform the image to semi-realistic image",
"AnyPose": "Make the person in image 1 do the exact same pose of the person in image 2. Changing the style and background of the image of the person in image 1 is undesirable, so don't do it. The new pose should be pixel accurate to the pose we are trying to copy. The position of the arms and head and legs should be the same as the pose we are trying to copy. Change the field of view and angle to match exactly image 2. Head tilt and eye gaze pose should match the person in image 2.",
"Hyperrealistic-Portrait": "Transform the image into an ultra-realistic photorealistic portrait with strict identity preservation, facing straight to the camera. Enhance pore-level skin textures, realistic moisture effects, and natural wet hair clumping against the skin. Apply cool-toned soft-box lighting with subtle highlights and shadows, maintain realistic green-hazel eye catchlights without synthetic gloss, and preserve soft natural lip texture. Use shallow depth of field with a clean bokeh background, an 85mm macro photographic look, and raw photo grading without retouching to maintain realism and original details.",
"Ultrarealistic-Portrait": "Transform the image into an ultra-realistic glamour portrait while strictly preserving the subject’s identity. Apply a close-up composition with a slight head tilt and a hand near the face, enhance cinematic directional lighting with dramatic fashion-style highlights, and refine makeup details including glowing skin, glossy lips, luminous highlighter, and defined eyes. Increase skin realism with detailed epidermal textures such as micropores, microhairs, subtle oil sheen, natural highlights, soft wrinkles, and subsurface scattering. Maintain a luxury fashion-magazine look in a 9:16 aspect ratio, preserving realism, facial structure, and original details without over-smoothing or retouching.",
"Upscale2K": "Upscale this picture to 4K resolution.",
"BFS-Best-FaceSwap": "head_swap: start with Picture 1 as the base image, keeping its lighting, environment, and background. remove the head from Picture 1 completely and replace it with the head from Picture 2, strictly preserving the hair, eye color, and nose structure of Picture 2. copy the eye direction, head rotation, and micro-expressions from Picture 1. high quality, sharp details, 4k",
"BFS-Best-FaceSwap-merge": "head_swap: start with Picture 1 as the base image, keeping its lighting, environment, and background. remove the head from Picture 1 completely and replace it with the head from Picture 2, strictly preserving the hair, eye color, and nose structure of Picture 2. copy the eye direction, head rotation, and micro-expressions from Picture 1. high quality, sharp details, 4k",
}
# Track what is currently loaded in memory (adapter_name values)
LOADED_ADAPTERS = set()
# ============================================================
# Helpers: resolution
# ============================================================
def _round8(x: int) -> int:
return max(8, (int(x) // 8) * 8)
def compute_dimensions(image: Image.Image, long_edge: int) -> tuple[int, int]:
w, h = image.size
if w >= h:
new_w = long_edge
new_h = int(round(long_edge * (h / w)))
else:
new_h = long_edge
new_w = int(round(long_edge * (w / h)))
return _round8(new_w), _round8(new_h)
def get_target_long_edge_for_lora(lora_adapter: str) -> int:
spec = ADAPTER_SPECS.get(lora_adapter, {})
return int(spec.get("target_long_edge", 1024))
# ============================================================
# Helpers: multi-input routing + gallery normalization
# ============================================================
def lora_requires_two_images(lora_adapter: str) -> bool:
return bool(ADAPTER_SPECS.get(lora_adapter, {}).get("requires_two_images", False))
def image2_label_for_lora(lora_adapter: str) -> str:
return str(ADAPTER_SPECS.get(lora_adapter, {}).get("image2_label", "Upload Reference (Image 2)"))
def _to_pil_rgb(x) -> Optional[Image.Image]:
"""
Accepts PIL / numpy / (image, caption) tuples from gr.Gallery and returns PIL RGB.
Gradio Gallery commonly yields tuples like (image, caption).
"""
if x is None:
return None
# Gallery often returns (image, caption)
if isinstance(x, tuple) and len(x) >= 1:
x = x[0]
if x is None:
return None
if isinstance(x, Image.Image):
return x.convert("RGB")
if isinstance(x, np.ndarray):
return Image.fromarray(x).convert("RGB")
# Best-effort fallback
try:
return Image.fromarray(np.array(x)).convert("RGB")
except Exception:
return None
def build_labeled_images(
img1: Image.Image,
img2: Optional[Image.Image],
extra_imgs: Optional[list[Image.Image]],
) -> dict[str, Image.Image]:
"""
Creates labels image_1, image_2, image_3... based on what is actually uploaded:
- img1 is always image_1
- img2 becomes image_2 only if present
- extras start immediately after the last present base box
The pipeline receives images in this exact order.
"""
labeled: dict[str, Image.Image] = {}
idx = 1
labeled[f"image_{idx}"] = img1
idx += 1
if img2 is not None:
labeled[f"image_{idx}"] = img2
idx += 1
if extra_imgs:
for im in extra_imgs:
if im is None:
continue
labeled[f"image_{idx}"] = im
idx += 1
return labeled
# ============================================================
# Helpers: BFS alpha key fix
# ============================================================
def _inject_missing_alpha_keys(state_dict: dict) -> dict:
"""
Diffusers' Qwen LoRA converter expects '<module>.alpha' keys.
BFS safetensors omits them. We inject alpha = rank (neutral scaling).
IMPORTANT: diffusers may strip 'diffusion_model.' before lookup, so we
inject BOTH:
- diffusion_model.xxx.alpha
- xxx.alpha
"""
bases = {}
for k, v in state_dict.items():
if not isinstance(v, torch.Tensor):
continue
if k.endswith(".lora_down.weight") and v.ndim >= 1:
base = k[: -len(".lora_down.weight")]
rank = int(v.shape[0])
bases[base] = rank
for base, rank in bases.items():
alpha_tensor = torch.tensor(float(rank), dtype=torch.float32)
full_alpha = f"{base}.alpha"
if full_alpha not in state_dict:
state_dict[full_alpha] = alpha_tensor
if base.startswith("diffusion_model."):
stripped_base = base[len("diffusion_model.") :]
stripped_alpha = f"{stripped_base}.alpha"
if stripped_alpha not in state_dict:
state_dict[stripped_alpha] = alpha_tensor
return state_dict
def _filter_to_diffusers_lora_keys(state_dict: dict) -> tuple[dict, dict]:
"""Return (filtered_state_dict, stats).
Some ComfyUI/Qwen safetensors (especially "merged" variants) include non-LoRA
delta/patch keys like `*.diff` and `*.diff_b` alongside real LoRA tensors.
Diffusers' internal Qwen LoRA converter is strict: any leftover keys cause an
error (`state_dict should be empty...`).
This helper keeps only the keys Diffusers can consume as a LoRA:
- `*.lora_up.weight`
- `*.lora_down.weight`
- (rare) `*.lora_mid.weight`
- alpha keys: `*.alpha` (or `*.lora_alpha` which we normalize to `*.alpha`)
It also drops known patch keys (`*.diff`, `*.diff_b`) and everything else.
"""
keep_suffixes = (
".lora_up.weight",
".lora_down.weight",
".lora_mid.weight",
".alpha",
".lora_alpha",
)
dropped_patch = 0
dropped_other = 0
kept = 0
normalized_alpha = 0
out: dict[str, torch.Tensor] = {}
for k, v in state_dict.items():
if not isinstance(v, torch.Tensor):
# Ignore non-tensor entries if any.
dropped_other += 1
continue
# Drop ComfyUI "delta" keys that Diffusers' LoRA loader will never consume.
if k.endswith(".diff") or k.endswith(".diff_b"):
dropped_patch += 1
continue
if not k.endswith(keep_suffixes):
dropped_other += 1
continue
if k.endswith(".lora_alpha"):
# Normalize common alt name to what Diffusers expects.
base = k[: -len(".lora_alpha")]
k2 = f"{base}.alpha"
out[k2] = v.float() if v.dtype != torch.float32 else v
normalized_alpha += 1
kept += 1
continue
out[k] = v
kept += 1
stats = {
"kept": kept,
"dropped_patch": dropped_patch,
"dropped_other": dropped_other,
"normalized_alpha": normalized_alpha,
}
return out, stats
def _duplicate_stripped_prefix_keys(state_dict: dict, prefix: str = "diffusion_model.") -> dict:
"""Ensure both prefixed and unprefixed variants exist for LoRA-related keys.
Diffusers' Qwen LoRA conversion may strip `diffusion_model.` when looking up
modules. Some exports only include prefixed keys. To be maximally compatible,
we duplicate LoRA keys (and alpha) in stripped form when missing.
"""
out = dict(state_dict)
for k, v in list(state_dict.items()):
if not k.startswith(prefix):
continue
stripped = k[len(prefix) :]
if stripped not in out:
out[stripped] = v
return out
def _load_lora_weights_with_fallback(repo: str, weight_name: str, adapter_name: str, needs_alpha_fix: bool = False):
"""
Normal path: pipe.load_lora_weights(repo, weight_name=..., adapter_name=...)
BFS fallback: download safetensors, inject missing alpha keys, then load from dict.
"""
try:
pipe.load_lora_weights(repo, weight_name=weight_name, adapter_name=adapter_name)
return
except (KeyError, ValueError) as e:
# KeyError: missing required alpha keys (common in BFS)
# ValueError: Diffusers Qwen converter found leftover keys (e.g. .diff/.diff_b)
if not needs_alpha_fix:
raise
print(
"⚠️ LoRA load failed (will try safe dict fallback). "
f"Adapter={adapter_name!r} file={weight_name!r} error={type(e).__name__}: {e}"
)
local_path = hf_hub_download(repo_id=repo, filename=weight_name)
sd = safetensors_load_file(local_path)
# 1) Inject required `<module>.alpha` keys (neutral scaling alpha=rank).
sd = _inject_missing_alpha_keys(sd)
# 2) Keep only LoRA + alpha keys; drop ComfyUI patch/delta keys.
sd, stats = _filter_to_diffusers_lora_keys(sd)
# 3) Duplicate stripped keys (remove `diffusion_model.`) for compatibility.
sd = _duplicate_stripped_prefix_keys(sd)
print(
"🧹 LoRA dict cleanup stats: "
f"kept={stats['kept']} dropped_patch={stats['dropped_patch']} "
f"dropped_other={stats['dropped_other']} normalized_alpha={stats['normalized_alpha']}"
)
pipe.load_lora_weights(sd, adapter_name=adapter_name)
return
# ============================================================
# LoRA loader: single/package + strengths
# ============================================================
def _ensure_loaded_and_get_active_adapters(selected_lora: str):
spec = ADAPTER_SPECS.get(selected_lora)
if not spec:
raise gr.Error(f"Configuration not found for: {selected_lora}")
adapter_names = []
adapter_weights = []
if spec.get("type") == "package":
parts = spec.get("parts", [])
if not parts:
raise gr.Error(f"Package spec has no parts: {selected_lora}")
for part in parts:
repo = part["repo"]
weights = part["weights"]
adapter_name = part["adapter_name"]
strength = float(part.get("strength", 1.0))
needs_alpha_fix = bool(part.get("needs_alpha_fix", False))
if adapter_name not in LOADED_ADAPTERS:
print(f"--- Downloading and Loading Adapter Part: {selected_lora} / {adapter_name} ---")
try:
_load_lora_weights_with_fallback(
repo=repo,
weight_name=weights,
adapter_name=adapter_name,
needs_alpha_fix=needs_alpha_fix,
)
LOADED_ADAPTERS.add(adapter_name)
except Exception as e:
raise gr.Error(f"Failed to load adapter part {selected_lora}/{adapter_name}: {e}")
else:
print(f"--- Adapter part already loaded: {selected_lora} / {adapter_name} ---")
adapter_names.append(adapter_name)
adapter_weights.append(strength)
else:
repo = spec["repo"]
weights = spec["weights"]
adapter_name = spec["adapter_name"]
strength = float(spec.get("strength", 1.0))
needs_alpha_fix = bool(spec.get("needs_alpha_fix", False))
if adapter_name not in LOADED_ADAPTERS:
print(f"--- Downloading and Loading Adapter: {selected_lora} ---")
try:
_load_lora_weights_with_fallback(
repo=repo,
weight_name=weights,
adapter_name=adapter_name,
needs_alpha_fix=needs_alpha_fix,
)
LOADED_ADAPTERS.add(adapter_name)
except Exception as e:
raise gr.Error(f"Failed to load adapter {selected_lora}: {e}")
else:
print(f"--- Adapter {selected_lora} is already loaded. ---")
adapter_names = [adapter_name]
adapter_weights = [strength]
return adapter_names, adapter_weights
# ============================================================
# UI handlers
# ============================================================
def on_lora_change_ui(selected_lora, current_prompt):
# Preset prompt (fill only if empty)
if selected_lora != NONE_LORA:
preset = LORA_PRESET_PROMPTS.get(selected_lora, "")
if preset and (current_prompt is None or str(current_prompt).strip() == ""):
prompt_update = gr.update(value=preset)
else:
prompt_update = gr.update(value=current_prompt)
else:
prompt_update = gr.update(value=current_prompt)
# Image2 visibility/label
if lora_requires_two_images(selected_lora):
img2_update = gr.update(visible=True, label=image2_label_for_lora(selected_lora))
else:
img2_update = gr.update(visible=False, value=None, label="Upload Reference (Image 2)")
return prompt_update, img2_update
# ============================================================
# Inference
# ============================================================
@spaces.GPU
def infer(
input_image_1,
input_image_2,
input_images_extra, # gallery multi-image box
prompt,
lora_adapter,
seed,
randomize_seed,
guidance_scale,
steps,
progress=gr.Progress(track_tqdm=True),
):
gc.collect()
if torch.cuda.is_available():
torch.cuda.empty_cache()
if input_image_1 is None:
raise gr.Error("Please upload Image 1.")
# Handle "None"
if lora_adapter == NONE_LORA:
try:
pipe.set_adapters([], adapter_weights=[])
except Exception:
if LOADED_ADAPTERS:
pipe.set_adapters(list(LOADED_ADAPTERS), adapter_weights=[0.0] * len(LOADED_ADAPTERS))
else:
adapter_names, adapter_weights = _ensure_loaded_and_get_active_adapters(lora_adapter)
pipe.set_adapters(adapter_names, adapter_weights=adapter_weights)
if randomize_seed:
seed = random.randint(0, MAX_SEED)
generator = torch.Generator(device=device).manual_seed(seed)
negative_prompt = (
"worst quality, low quality, bad anatomy, bad hands, text, error, missing fingers, "
"extra digit, fewer digits, cropped, jpeg artifacts, signature, watermark, username, blurry"
)
img1 = input_image_1.convert("RGB")
img2 = input_image_2.convert("RGB") if input_image_2 is not None else None
# Normalize extra images (Gallery) to PIL RGB (handles tuples from Gallery)
extra_imgs: list[Image.Image] = []
if input_images_extra:
for item in input_images_extra:
pil = _to_pil_rgb(item)
if pil is not None:
extra_imgs.append(pil)
# Enforce existing 2-image LoRA behavior (image_1 + image_2 required)
if lora_requires_two_images(lora_adapter) and img2 is None:
raise gr.Error("This LoRA needs two images. Please upload Image 2 as well.")
# Label images as image_1, image_2, image_3...
labeled = build_labeled_images(img1, img2, extra_imgs)
# Pass to pipeline in labeled order. Keep single-image call when only one is present.
pipe_images = list(labeled.values())
if len(pipe_images) == 1:
pipe_images = pipe_images[0]
# Resolution derived from Image 1 (base/body/target)
target_long_edge = get_target_long_edge_for_lora(lora_adapter)
width, height = compute_dimensions(img1, target_long_edge)
try:
print(
"[DEBUG][infer] submitting request | "
f"lora_adapter={lora_adapter!r} seed={seed} prompt={prompt!r}"
)
result = pipe(
image=pipe_images,
prompt=prompt,
negative_prompt=negative_prompt,
height=height,
width=width,
num_inference_steps=steps,
generator=generator,
true_cfg_scale=guidance_scale,
).images[0]
return result, seed
finally:
gc.collect()
if torch.cuda.is_available():
torch.cuda.empty_cache()
@spaces.GPU
def infer_example(input_image, prompt, lora_adapter):
if input_image is None:
return None, 0
input_pil = input_image.convert("RGB")
guidance_scale = 1.0
steps = 4
# Examples don't supply Image 2 or extra images; and example list doesn't include AnyPose/BFS.
result, seed = infer(input_pil, None, None, prompt, lora_adapter, 0, True, guidance_scale, steps)
return result, seed
# ============================================================
# UI
# ============================================================
css = """
#col-container {
margin: 0 auto;
max-width: 960px;
}
#main-title h1 {font-size: 2.1em !important;}
"""
aio_status_line = (
f"**AIO transformer version:** `{AIO_VERSION}` "
f"({AIO_VERSION_SOURCE}; env `AIO_VERSION`={_AIO_ENV_RAW!r})"
)
with gr.Blocks() as demo:
with gr.Column(elem_id="col-container"):
gr.Markdown("# **Qwen-Image-Edit-2511-LoRAs-Fast**", elem_id="main-title")
gr.Markdown(
"Perform diverse image edits using specialized "
"[LoRA](https://huggingface.co/models?other=base_model:adapter:Qwen/Qwen-Image-Edit-2511) adapters for the "
"[Qwen-Image-Edit](https://huggingface.co/Qwen/Qwen-Image-Edit-2511) model. Uses a Diffusers compatible extraction of the transformers from Phr00t's Rapid AIO merge. If a different AIO version is desired, copy the space and set the space variable to change version.'"
)
gr.Markdown(aio_status_line)
with gr.Row(equal_height=True):
with gr.Column():
input_image_1 = gr.Image(label="Upload Image 1 (Base / Target)", type="pil", height=290)
input_image_2 = gr.Image(label="Upload Reference (Image 2)", type="pil", height=290, visible=False)
input_images_extra = gr.Gallery(
label="Upload Additional Images (auto-indexed after Image 1/2)",
type="pil",
height=290,
columns=4,
rows=2,
interactive=True,
)
prompt = gr.Text(
label="Edit Prompt",
show_label=True,
placeholder="e.g., transform into photo..",
)
run_button = gr.Button("Edit Image", variant="primary")
with gr.Column():
output_image = gr.Image(label="Output Image", interactive=False, format="png", height=353)
with gr.Row():
lora_choices = [NONE_LORA] + list(ADAPTER_SPECS.keys())
lora_adapter = gr.Dropdown(
label="Choose Editing Style",
choices=lora_choices,
value=NONE_LORA,
)
with gr.Accordion("Advanced Settings", open=False, visible=True):
seed = gr.Slider(label="Seed", minimum=0, maximum=MAX_SEED, step=1, value=0)
randomize_seed = gr.Checkbox(label="Randomize Seed", value=True)
guidance_scale = gr.Slider(label="Guidance Scale", minimum=1.0, maximum=10.0, step=0.1, value=1.0)
steps = gr.Slider(label="Inference Steps", minimum=1, maximum=50, step=1, value=4)
# On LoRA selection: preset prompt + toggle Image 2
lora_adapter.change(
fn=on_lora_change_ui,
inputs=[lora_adapter, prompt],
outputs=[prompt, input_image_2],
)
gr.Examples(
examples=[
["examples/5.jpg", "Remove shadows and relight the image using soft lighting.", "Light-Restoration"],
["examples/4.jpg", "Use a subtle golden-hour filter with smooth light diffusion.", "Relight"],
["examples/2.jpeg", "Rotate the camera 45 degrees to the left.", "Multiple-Angles"],
[
"examples/12.jpg",
"flatcolor Desaturate the image and lower the contrast to create a flat, ungraded look similar to a camera log profile. Preserve details in the highlights and shadows.",
"Flat-Log",
],
["examples/7.jpg", "Light source from the Right Rear", "Multi-Angle-Lighting"],
["examples/10.jpeg", "Upscale the image.", "Upscale-Image"],
["examples/7.jpg", "Light source from the Below", "Multi-Angle-Lighting"],
["examples/2.jpeg", "Switch the camera to a top-down right corner view.", "Multiple-Angles"],
[
"examples/9.jpg",
"The camera moves slightly forward as sunlight breaks through the clouds, casting a soft glow around the character's silhouette in the mist. Realistic cinematic style, atmospheric depth.",
"Next-Scene",
],
["examples/8.jpg", "Make the subjects skin details more prominent and natural.", "Edit-Skin"],
["examples/6.jpg", "Switch the camera to a bottom-up view.", "Multiple-Angles"],
["examples/6.jpg", "Rotate the camera 180 degrees upside down.", "Multiple-Angles"],
["examples/4.jpg", "Rotate the camera 45 degrees to the right.", "Multiple-Angles"],
["examples/4.jpg", "Switch the camera to a top-down view.", "Multiple-Angles"],
["examples/4.jpg", "Switch the camera to a wide-angle lens.", "Multiple-Angles"],
["examples/11.jpg", "Upscale this picture to 4K resolution.", "Upscale2K"],
],
inputs=[input_image_1, prompt, lora_adapter],
outputs=[output_image, seed],
fn=infer_example,
cache_examples=False,
label="Examples",
)
run_button.click(
fn=infer,
inputs=[
input_image_1,
input_image_2,
input_images_extra,
prompt,
lora_adapter,
seed,
randomize_seed,
guidance_scale,
steps,
],
outputs=[output_image, seed],
)
if __name__ == "__main__":
demo.queue(max_size=30).launch(
css=css,
theme=orange_red_theme,
mcp_server=True,
ssr_mode=False,
show_error=True,
)