Spaces:
Paused
Paused
import gradio as gr | |
import numpy as np | |
import re | |
import itertools | |
import os | |
import imageio | |
import imageio.plugins.ffmpeg | |
import ffmpeg | |
from PIL import Image, ImageDraw, ImageFont | |
from diffusers_helper.utils import generate_timestamp | |
from modules.video_queue import JobType | |
# --- Helper Dictionaries & Functions --- | |
xy_plot_axis_options = { | |
# "type": [ | |
# "dropdown(checkboxGroup), textbox or number", | |
# "empty if textbox, dtype if number, [] if dropdown", | |
# "standard values", | |
# "True if multi axis - like prompt replace, False is only on one axis - like steps" | |
# ], | |
"Nothing": ["nothing", "", "", True], | |
"Model type": ["dropdown", ["Original", "F1"], ["Original", "F1"], False], | |
"End frame influence": ["number", "float", "0.05-0.95[3]", False], | |
"Latent type": ["dropdown", ["Black", "White", "Noise", "Green Screen"], ["Black", "Noise"], False], | |
"Prompt add": ["textbox", "", "", True], | |
"Prompt replace": ["textbox", "", "", True], | |
"Blend sections": ["number", "int", "3-7 [3]", False], | |
"Steps": ["number", "int", "15-30 [3]", False], | |
"Seed": ["number", "int", "1000-10000 [3]", False], | |
"Use teacache": ["dropdown", [True, False], [True, False], False], | |
"TeaCache steps": ["number", "int", "5-25 [3]", False], | |
"TeaCache rel_l1_thresh": ["number", "float", "0.01-0.3 [3]", False], | |
"Use MagCache": ["dropdown", [True, False], [True, False], False], | |
"MagCache Threshold": ["number", "float", "0.01-1.0 [3]", False], | |
"MagCache Max Consecutive Skips": ["number", "int", "1-5 [3]", False], | |
"MagCache Retention Ratio": ["number", "float", "0.0-1.0 [3]", False], | |
# "CFG": ["number", "float", "", False], | |
"Distilled CFG Scale": ["number", "float", "5-15 [3]", False], | |
# "RS": ["number", "float", "", False], | |
# "Use weighted embeddings": ["dropdown", [True, False], [True, False], False], | |
} | |
text_to_base_keys = { | |
"Model type": "model_type", | |
"End frame influence": "end_frame_strength_original", | |
"Latent type": "latent_type", | |
"Prompt add": "prompt", | |
"Prompt replace": "prompt", | |
"Blend sections": "blend_sections", | |
"Steps": "steps", | |
"Seed": "seed", | |
"Use teacache": "use_teacache", | |
"TeaCache steps":"teacache_num_steps", | |
"TeaCache rel_l1_thresh":"teacache_rel_l1_thresh", | |
"Use MagCache": "use_magcache", | |
"MagCache Threshold": "magcache_threshold", | |
"MagCache Max Consecutive Skips": "magcache_max_consecutive_skips", | |
"MagCache Retention Ratio": "magcache_retention_ratio", | |
"Latent window size": "latent_window_size", | |
# "CFG": "", | |
"Distilled CFG Scale": "gs", | |
# "RS": "", | |
# "Use weighted embeddings": "", | |
} | |
def xy_plot_parse_input(text): | |
text = text.strip() | |
if ',' in text: | |
return [x.strip() for x in text.split(",")] | |
match = re.match(r'^\s*(-?\d*\.?\d*)\s*-\s*(-?\d*\.?\d*)\s*\[\s*(\d+)\s*\]$', text) | |
if match: | |
start, end, count = map(float, match.groups()) | |
result = np.linspace(start, end, int(count)) | |
if np.allclose(result, np.round(result)): | |
result = np.round(result).astype(int) | |
return result.tolist() | |
return [] | |
def xy_plot_process( | |
job_queue, settings, # Added explicit dependencies | |
model_type, input_image, end_frame_image_original, | |
end_frame_strength_original, latent_type, | |
prompt, blend_sections, steps, total_second_length, | |
resolutionW, resolutionH, seed, randomize_seed, use_teacache, | |
teacache_num_steps, teacache_rel_l1_thresh, | |
use_magcache, magcache_threshold, magcache_max_consecutive_skips, magcache_retention_ratio, | |
latent_window_size, | |
cfg, gs, rs, gpu_memory_preservation, mp4_crf, | |
axis_x_switch, axis_x_value_text, axis_x_value_dropdown, | |
axis_y_switch, axis_y_value_text, axis_y_value_dropdown, | |
axis_z_switch, axis_z_value_text, axis_z_value_dropdown, | |
selected_loras, | |
*lora_slider_values | |
): | |
# print(model_type, input_image, latent_type, | |
# prompt, blend_sections, steps, total_second_length, | |
# resolutionW, resolutionH, seed, randomize_seed, use_teacache, | |
# latent_window_size, cfg, gs, rs, gpu_memory_preservation, | |
# mp4_crf, | |
# axis_x_switch, axis_x_value_text, axis_x_value_dropdown, | |
# axis_y_switch, axis_y_value_text, axis_y_value_dropdown, | |
# axis_z_switch, axis_z_value_text, axis_z_value_dropdown, sep=", ") | |
if axis_x_switch == "Nothing" and axis_y_switch == "Nothing" and axis_z_switch == "Nothing": | |
return "Not selected any axis for plot", gr.update() | |
if (axis_x_switch == "Nothing" or axis_y_switch == "Nothing") and axis_z_switch != "Nothing": | |
return "For using Z axis, first use X and Y axis", gr.update() | |
if axis_x_switch == "Nothing" and axis_y_switch != "Nothing": | |
return "For using Y axis, first use X axis", gr.update() | |
if xy_plot_axis_options[axis_x_switch][0] == "dropdown" and len(axis_x_value_dropdown) < 1: | |
return "No values for axis X", gr.update() | |
if xy_plot_axis_options[axis_y_switch][0] == "dropdown" and len(axis_y_value_dropdown) < 1: | |
return "No values for axis Y", gr.update() | |
if xy_plot_axis_options[axis_z_switch][0] == "dropdown" and len(axis_z_value_dropdown) < 1: | |
return "No values for axis Z", gr.update() | |
if not xy_plot_axis_options[axis_x_switch][3]: | |
if axis_x_switch == axis_y_switch: | |
return "Axis type on X and Y axis are same, you can't do that generation.<br>Multi axis supported only for \"Prompt add\" and \"Prompt replace\".", gr.update() | |
if axis_x_switch == axis_z_switch: | |
return "Axis type on X and Z axis are same, you can't do that generation.<br>Multi axis supported only for \"Prompt add\" and \"Prompt replace\".", gr.update() | |
if not xy_plot_axis_options[axis_y_switch][3]: | |
if axis_y_switch == axis_z_switch: | |
return "Axis type on Y and Z axis are same, you can't do that generation.<br>Multi axis supported only for \"Prompt add\" and \"Prompt replace\".", gr.update() | |
base_generator_vars = { | |
"model_type": model_type, | |
"input_image": input_image, | |
"end_frame_image": None, | |
"end_frame_strength": 1.0, | |
"input_video": None, | |
"end_frame_image_original": end_frame_image_original, | |
"end_frame_strength_original": end_frame_strength_original, | |
"prompt_text": prompt, | |
"n_prompt": "", | |
"seed": seed, | |
"total_second_length": total_second_length, | |
"latent_window_size": latent_window_size, | |
"steps": steps, | |
"cfg": cfg, | |
"gs": gs, | |
"rs": rs, | |
"use_teacache": use_teacache, | |
"teacache_num_steps": teacache_num_steps, | |
"teacache_rel_l1_thresh": teacache_rel_l1_thresh, | |
"use_magcache": use_magcache, | |
"magcache_threshold": magcache_threshold, | |
"magcache_max_consecutive_skips": magcache_max_consecutive_skips, | |
"magcache_retention_ratio": magcache_retention_ratio, | |
"has_input_image": True if input_image is not None else False, | |
"save_metadata_checked": True, | |
"blend_sections": blend_sections, | |
"latent_type": latent_type, | |
"selected_loras": selected_loras, | |
"resolutionW": resolutionW, | |
"resolutionH": resolutionH, | |
"lora_loaded_names": lora_names, | |
"lora_values": lora_slider_values | |
} | |
def xy_plot_convert_values(type, value_textbox, value_dropdown): | |
retVal = [] | |
if type[0] == "dropdown": | |
retVal = value_dropdown | |
elif type[0] == "textbox": | |
retVal = xy_plot_parse_input(value_textbox) | |
elif type[0] == "number": | |
if type[1] == "int": | |
retVal = [int(float(x)) for x in xy_plot_parse_input(value_textbox)] | |
else: | |
retVal = [float(x) for x in xy_plot_parse_input(value_textbox)] | |
return retVal | |
prompt_replace_initial_values = {} | |
all_axis_values = { | |
axis_x_switch+" -> X": xy_plot_convert_values(xy_plot_axis_options[axis_x_switch], axis_x_value_text, axis_x_value_dropdown) | |
} | |
if axis_x_switch == "Prompt replace": | |
prompt_replace_initial_values["X"] = all_axis_values[axis_x_switch+" -> X"][0] | |
if prompt_replace_initial_values["X"] not in base_generator_vars["prompt_text"]: | |
return "Prompt for replacing in X axis not present in generation prompt", gr.update() | |
if axis_y_switch != "Nothing": | |
all_axis_values[axis_y_switch+" -> Y"] = xy_plot_convert_values(xy_plot_axis_options[axis_y_switch], axis_y_value_text, axis_y_value_dropdown) | |
if axis_y_switch == "Prompt replace": | |
prompt_replace_initial_values["Y"] = all_axis_values[axis_y_switch+" -> Y"][0] | |
if prompt_replace_initial_values["Y"] not in base_generator_vars["prompt_text"]: | |
return "Prompt for replacing in Y axis not present in generation prompt", gr.update() | |
if axis_z_switch != "Nothing": | |
all_axis_values[axis_z_switch+" -> Z"] = xy_plot_convert_values(xy_plot_axis_options[axis_z_switch], axis_z_value_text, axis_z_value_dropdown) | |
if axis_z_switch == "Prompt replace": | |
prompt_replace_initial_values["Z"] = all_axis_values[axis_z_switch+" -> Z"][0] | |
if prompt_replace_initial_values["Z"] not in base_generator_vars["prompt_text"]: | |
return "Prompt for replacing in Z axis not present in generation prompt", gr.update() | |
active_axes = list(all_axis_values.keys()) | |
value_lists = [all_axis_values[axis] for axis in active_axes] | |
output_generator_vars = [] | |
combintion_plot = itertools.product(*value_lists) | |
for combo in combintion_plot: | |
vars_copy = base_generator_vars.copy() | |
for axis, value in zip(active_axes, combo): | |
splitted_axis_name = axis.split(" -> ") | |
if splitted_axis_name[0] == "Prompt add": | |
vars_copy["prompt_text"] = vars_copy["prompt_text"] + " " + str(value) | |
elif splitted_axis_name[0] == "Prompt replace": | |
orig_copy_prompt_text = vars_copy["prompt_text"] | |
vars_copy["prompt_text"] = orig_copy_prompt_text.replace(prompt_replace_initial_values[splitted_axis_name[1]], str(value)) | |
else: | |
vars_copy[text_to_base_keys[splitted_axis_name[0]]] = value | |
vars_copy[splitted_axis_name[1]+"_axis_on_plot"] = str(value) | |
worker_params = {k: v for k, v in vars_copy.items() if k not in ["X_axis_on_plot", "Y_axis_on_plot", "Z_axis_on_plot"]} | |
output_generator_vars.append(worker_params) | |
# print("----- BEFORE GENERATED VIDS VARS START -----") | |
# for v in output_generator_vars: | |
# print(v) | |
# print("------ BEFORE GENERATED VIDS VARS END ------") | |
job_queue.add_job( | |
params=base_generator_vars, | |
job_type=JobType.GRID, | |
child_job_params_list=output_generator_vars | |
) | |
return "Grid job added to the queue.", gr.update(visible=False) | |
# print("----- GENERATED VIDS VARS START -----") | |
# for v in output_generator_vars: | |
# print(v) | |
# print("------ GENERATED VIDS VARS END ------") | |
# -------------------------- connect with settings -------------------------- | |
# Ensure settings is available in this scope or passed in. | |
# Assuming 'settings' object is available from create_interface's scope. | |
output_dir_setting = settings.get("output_dir", "outputs") | |
mp4_crf_setting = settings.get("mp4_crf", 16) # Default CRF if not in settings | |
# -------------------------- connect with settings -------------------------- | |
def create_xy_plot_ui(lora_names, default_prompt, DUMMY_LORA_NAME): | |
""" | |
Creates the Gradio UI for the XY Plot functionality. | |
Returns a dictionary of key components to be used by the main interface. | |
""" | |
with gr.Group(visible=False) as xy_group: # The original was visible=False | |
with gr.Row(): | |
xy_plot_model_type = gr.Radio( | |
["Original", "F1"], | |
label="Model Type", | |
value="F1", | |
info="Select which model to use for generation" | |
) | |
with gr.Group(): | |
with gr.Row(): | |
with gr.Column(scale=1): | |
xy_plot_input_image = gr.Image( | |
sources='upload', | |
type="numpy", | |
label="Image (optional)", | |
height=420, | |
image_mode="RGB", | |
elem_classes="contain-image" | |
) | |
with gr.Column(scale=1): | |
xy_plot_end_frame_image_original = gr.Image( | |
sources='upload', | |
type="numpy", | |
label="End Frame (Optional)", | |
height=420, | |
elem_classes="contain-image", | |
image_mode="RGB", | |
show_download_button=False, | |
show_label=True, | |
container=True | |
) | |
with gr.Group(): | |
xy_plot_end_frame_strength_original = gr.Slider( | |
label="End Frame Influence", | |
minimum=0.05, | |
maximum=1.0, | |
value=1.0, | |
step=0.05, | |
info="Controls how strongly the end frame guides the generation. 1.0 is full influence." | |
) | |
with gr.Accordion("Latent Image Options", open=False): | |
xy_plot_latent_type = gr.Dropdown( | |
["Black", "White", "Noise", "Green Screen"], | |
label="Latent Image", | |
value="Black", | |
info="Used as a starting point if no image is provided" | |
) | |
xy_plot_prompt = gr.Textbox(label="Prompt", value=default_prompt) | |
with gr.Accordion("Prompt Parameters", open=False): | |
xy_plot_blend_sections = gr.Slider( | |
minimum=0, maximum=10, value=4, step=1, | |
label="Number of sections to blend between prompts" | |
) | |
with gr.Accordion("Generation Parameters", open=True): | |
with gr.Row(): | |
xy_plot_steps = gr.Slider(label="Steps", minimum=1, maximum=100, value=5, step=1) | |
xy_plot_total_second_length = gr.Slider(label="Video Length (Seconds)", minimum=0.1, maximum=120, value=1, step=0.1) | |
with gr.Row(): | |
xy_plot_seed = gr.Number(label="Seed", value=31337, precision=0) | |
xy_plot_randomize_seed = gr.Checkbox(label="Randomize", value=False, info="Generate a new random seed for each job") | |
with gr.Row("LoRAs"): | |
xy_plot_lora_selector = gr.Dropdown( | |
choices=lora_names, | |
label="Select LoRAs to Load", | |
multiselect=True, | |
value=[], | |
info="Select one or more LoRAs to use for this job" | |
) | |
xy_plot_lora_sliders = {} | |
for lora in lora_names: | |
xy_plot_lora_sliders[lora] = gr.Slider( | |
minimum=0.0, maximum=2.0, value=1.0, step=0.01, | |
label=f"{lora} Weight", visible=False, interactive=True | |
) | |
with gr.Accordion("Advanced Parameters", open=False): | |
with gr.Row("TeaCache"): | |
xy_plot_use_teacache = gr.Checkbox(label='Use TeaCache', value=True, info='Faster speed, but often makes hands and fingers slightly worse.') | |
xy_plot_teacache_num_steps = gr.Slider(label="TeaCache steps", minimum=1, maximum=50, step=1, value=25, visible=True, info='How many intermediate sections to keep in the cache') | |
xy_plot_teacache_rel_l1_thresh = gr.Slider(label="TeaCache rel_l1_thresh", minimum=0.01, maximum=1.0, step=0.01, value=0.15, visible=True, info='Relative L1 Threshold') | |
with gr.Row("MagCache"): | |
xy_plot_use_magcache = gr.Checkbox(label='Use MagCache', value=False, info='Faster speed, but may introduce artifacts. Uses pre-calibrated ratios.') | |
xy_plot_magcache_threshold = gr.Slider(label="MagCache Threshold", minimum=0.01, maximum=1.0, step=0.01, value=0.1, visible=False, info='Error tolerance for skipping steps. Lower = more skips, higher = fewer skips.') | |
xy_plot_magcache_max_consecutive_skips = gr.Slider(label="MagCache Max Consecutive Skips", minimum=1, maximum=10, step=1, value=2, visible=False, info='Maximum number of consecutive steps that can be skipped.') | |
xy_plot_magcache_retention_ratio = gr.Slider(label="MagCache Retention Ratio", minimum=0.0, maximum=1.0, step=0.01, value=0.25, visible=False, info='Ratio of initial steps to always calculate (not skip).') | |
# Mutual exclusivity logic for TeaCache and MagCache in XY Plot UI | |
xy_plot_use_teacache.change(lambda enabled: (gr.update(visible=enabled), gr.update(visible=enabled), gr.update(value=not enabled)), inputs=xy_plot_use_teacache, outputs=[xy_plot_teacache_num_steps, xy_plot_teacache_rel_l1_thresh, xy_plot_use_magcache]) | |
xy_plot_use_magcache.change(lambda enabled: (gr.update(visible=enabled), gr.update(visible=enabled), gr.update(visible=enabled), gr.update(value=not enabled)), inputs=xy_plot_use_magcache, outputs=[xy_plot_magcache_threshold, xy_plot_magcache_max_consecutive_skips, xy_plot_magcache_retention_ratio, xy_plot_use_teacache]) | |
xy_plot_latent_window_size = gr.Slider(label="Latent Window Size", minimum=1, maximum=33, value=9, step=1, visible=True, info='Change at your own risk, very experimental') | |
xy_plot_cfg = gr.Slider(label="CFG Scale", minimum=1.0, maximum=32.0, value=1.0, step=0.01, visible=False) | |
xy_plot_gs = gr.Slider(label="Distilled CFG Scale", minimum=1.0, maximum=32.0, value=10.0, step=0.01) | |
xy_plot_rs = gr.Slider(label="CFG Re-Scale", minimum=0.0, maximum=1.0, value=0.0, step=0.01, visible=False) | |
xy_plot_gpu_memory_preservation = gr.Slider(label="GPU Inference Preserved Memory (GB) (larger means slower)", minimum=1, maximum=128, value=6, step=0.1, info="Set this number to a larger value if you encounter OOM. Larger value causes slower speed.") | |
with gr.Accordion("Output Parameters", open=False): | |
xy_plot_mp4_crf = gr.Slider(label="MP4 Compression", minimum=0, maximum=100, value=16, step=1, info="Lower means better quality. 0 is uncompressed. Change to 16 if you get black outputs. ") | |
with gr.Accordion("Plot Parameters", open=True): | |
def xy_plot_axis_change(updated_value_type): | |
if xy_plot_axis_options[updated_value_type][0] == "textbox" or xy_plot_axis_options[updated_value_type][0] == "number": | |
return gr.update(visible=True, value=xy_plot_axis_options[updated_value_type][2]), gr.update(visible=False, value=[], choices=[]) | |
elif xy_plot_axis_options[updated_value_type][0] == "dropdown": | |
return gr.update(visible=False), gr.update(visible=True, value=xy_plot_axis_options[updated_value_type][2], choices=xy_plot_axis_options[updated_value_type][1]) | |
else: | |
return gr.update(visible=False), gr.update(visible=False, value=[], choices=[]) | |
with gr.Row(): | |
xy_plot_axis_x_switch = gr.Dropdown(label="X axis type for plotting", choices=list(xy_plot_axis_options.keys())) | |
xy_plot_axis_x_value_text = gr.Textbox(label="X axis comma separated text", visible=False) | |
xy_plot_axis_x_value_dropdown = gr.CheckboxGroup(label="X axis values", visible=False) #, multiselect=True) | |
with gr.Row(): | |
xy_plot_axis_y_switch = gr.Dropdown(label="Y axis type for plotting", choices=list(xy_plot_axis_options.keys())) | |
xy_plot_axis_y_value_text = gr.Textbox(label="Y axis comma separated text", visible=False) | |
xy_plot_axis_y_value_dropdown = gr.CheckboxGroup(label="Y axis values", visible=False) #, multiselect=True) | |
with gr.Row(visible=False): # not implemented Z axis | |
xy_plot_axis_z_switch = gr.Dropdown(label="Z axis type for plotting", choices=list(xy_plot_axis_options.keys())) | |
xy_plot_axis_z_value_text = gr.Textbox(label="Z axis comma separated text", visible=False) | |
xy_plot_axis_z_value_dropdown = gr.CheckboxGroup(label="Z axis values", visible=False) #, multiselect=True) | |
xy_plot_status = gr.HTML("") | |
xy_plot_output = gr.Video(autoplay=True, loop=True, sources=[], height=256, visible=False) | |
# --- ADD THE PROCESS BUTTON HERE --- | |
# This button is logically part of the XY plot group but will be controlled | |
# from interface.py. We place it here so it's encapsulated. | |
xy_plot_process_btn = gr.Button("Submit", visible=False) | |
# --- Internal Event Handlers --- | |
xy_plot_use_teacache.change(lambda enabled: (gr.update(visible=enabled), gr.update(visible=enabled)), inputs=xy_plot_use_teacache, outputs=[xy_plot_teacache_num_steps, xy_plot_teacache_rel_l1_thresh]) | |
xy_plot_axis_x_switch.change(fn=xy_plot_axis_change, inputs=[xy_plot_axis_x_switch], outputs=[xy_plot_axis_x_value_text, xy_plot_axis_x_value_dropdown]) | |
xy_plot_axis_y_switch.change(fn=xy_plot_axis_change, inputs=[xy_plot_axis_y_switch], outputs=[xy_plot_axis_y_value_text, xy_plot_axis_y_value_dropdown]) | |
xy_plot_axis_z_switch.change(fn=xy_plot_axis_change, inputs=[xy_plot_axis_z_switch], outputs=[xy_plot_axis_z_value_text, xy_plot_axis_z_value_dropdown]) | |
def xy_plot_update_lora_sliders(selected_loras): | |
updates = [] | |
actual_selected_loras_for_display = [lora for lora in selected_loras if lora != DUMMY_LORA_NAME] | |
updates.append(gr.update(value=actual_selected_loras_for_display)) | |
for lora_name_key in lora_names: | |
if lora_name_key == DUMMY_LORA_NAME: | |
updates.append(gr.update(visible=False)) | |
else: | |
updates.append(gr.update(visible=(lora_name_key in actual_selected_loras_for_display))) | |
return updates | |
xy_plot_lora_selector.change( | |
fn=xy_plot_update_lora_sliders, | |
inputs=[xy_plot_lora_selector], | |
outputs=[xy_plot_lora_selector] + [xy_plot_lora_sliders[lora] for lora in lora_names if lora in xy_plot_lora_sliders] | |
) | |
# --- Component Dictionary for Export --- | |
components = { | |
"group": xy_group, | |
"status": xy_plot_status, | |
"output": xy_plot_output, | |
"process_btn": xy_plot_process_btn, | |
# --- Inputs for the process button --- | |
"model_type": xy_plot_model_type, | |
"input_image": xy_plot_input_image, | |
"end_frame_image_original": xy_plot_end_frame_image_original, | |
"end_frame_strength_original": xy_plot_end_frame_strength_original, | |
"latent_type": xy_plot_latent_type, | |
"prompt": xy_plot_prompt, | |
"blend_sections": xy_plot_blend_sections, | |
"steps": xy_plot_steps, | |
"total_second_length": xy_plot_total_second_length, | |
"seed": xy_plot_seed, | |
"randomize_seed": xy_plot_randomize_seed, | |
"use_teacache": xy_plot_use_teacache, | |
"teacache_num_steps": xy_plot_teacache_num_steps, | |
"teacache_rel_l1_thresh": xy_plot_teacache_rel_l1_thresh, | |
"use_magcache": xy_plot_use_magcache, | |
"magcache_threshold": xy_plot_magcache_threshold, | |
"magcache_max_consecutive_skips": xy_plot_magcache_max_consecutive_skips, | |
"magcache_retention_ratio": xy_plot_magcache_retention_ratio, | |
"latent_window_size": xy_plot_latent_window_size, | |
"cfg": xy_plot_cfg, | |
"gs": xy_plot_gs, | |
"rs": xy_plot_rs, | |
"gpu_memory_preservation": xy_plot_gpu_memory_preservation, | |
"mp4_crf": xy_plot_mp4_crf, | |
"axis_x_switch": xy_plot_axis_x_switch, | |
"axis_x_value_text": xy_plot_axis_x_value_text, | |
"axis_x_value_dropdown": xy_plot_axis_x_value_dropdown, | |
"axis_y_switch": xy_plot_axis_y_switch, | |
"axis_y_value_text": xy_plot_axis_y_value_text, | |
"axis_y_value_dropdown": xy_plot_axis_y_value_dropdown, | |
"axis_z_switch": xy_plot_axis_z_switch, | |
"axis_z_value_text": xy_plot_axis_z_value_text, | |
"axis_z_value_dropdown": xy_plot_axis_z_value_dropdown, | |
"lora_selector": xy_plot_lora_selector, | |
"lora_sliders": xy_plot_lora_sliders, | |
} | |
return components |