2022-09-03 19:32:45 +03:00
from collections import namedtuple
from copy import copy
2022-10-06 13:55:21 +03:00
from itertools import permutations , chain
2022-09-03 19:32:45 +03:00
import random
2022-10-06 13:55:21 +03:00
import csv
from io import StringIO
2022-09-26 16:46:18 +03:00
from PIL import Image
2022-09-09 17:54:04 +03:00
import numpy as np
2022-09-03 19:32:45 +03:00
import modules . scripts as scripts
import gradio as gr
2023-01-16 17:36:56 +03:00
from modules import images , paths , sd_samplers , processing , sd_models , sd_vae
2022-11-19 12:01:51 +03:00
from modules . processing import process_images , Processed , StableDiffusionProcessingTxt2Img
2022-09-03 19:32:45 +03:00
from modules . shared import opts , cmd_opts , state
2022-09-17 13:49:36 +03:00
import modules . shared as shared
2022-09-03 19:32:45 +03:00
import modules . sd_samplers
2022-09-17 13:49:36 +03:00
import modules . sd_models
2022-12-18 18:47:02 +03:00
import modules . sd_vae
import glob
import os
2022-09-06 10:11:25 +03:00
import re
2022-09-03 19:32:45 +03:00
2023-01-16 17:36:56 +03:00
from modules . ui_components import ToolButton
2022-09-03 19:32:45 +03:00
2023-01-16 17:36:56 +03:00
fill_values_symbol = " \U0001f4d2 " # 📒
2023-01-16 08:41:58 +03:00
2022-09-03 19:32:45 +03:00
def apply_field ( field ) :
def fun ( p , x , xs ) :
setattr ( p , field , x )
return fun
def apply_prompt ( p , x , xs ) :
2022-10-11 13:59:56 +03:00
if xs [ 0 ] not in p . prompt and xs [ 0 ] not in p . negative_prompt :
raise RuntimeError ( f " Prompt S/R did not find { xs [ 0 ] } in prompt or negative prompt. " )
2022-10-11 13:16:57 +03:00
2022-09-03 19:32:45 +03:00
p . prompt = p . prompt . replace ( xs [ 0 ] , x )
2022-09-09 08:58:31 +03:00
p . negative_prompt = p . negative_prompt . replace ( xs [ 0 ] , x )
2022-09-03 19:32:45 +03:00
2022-10-04 05:20:09 +03:00
def apply_order ( p , x , xs ) :
token_order = [ ]
2022-10-04 09:18:00 +03:00
# Initally grab the tokens from the prompt, so they can be replaced in order of earliest seen
2022-10-04 05:20:09 +03:00
for token in x :
token_order . append ( ( p . prompt . find ( token ) , token ) )
token_order . sort ( key = lambda t : t [ 0 ] )
2022-10-04 08:07:36 +03:00
prompt_parts = [ ]
# Split the prompt up, taking out the tokens
for _ , token in token_order :
n = p . prompt . find ( token )
prompt_parts . append ( p . prompt [ 0 : n ] )
p . prompt = p . prompt [ n + len ( token ) : ]
# Rebuild the prompt with the tokens in the order we want
prompt_tmp = " "
for idx , part in enumerate ( prompt_parts ) :
prompt_tmp + = part
prompt_tmp + = x [ idx ]
p . prompt = prompt_tmp + p . prompt
2022-09-03 19:32:45 +03:00
def apply_sampler ( p , x , xs ) :
2022-11-27 13:43:10 +03:00
sampler_name = sd_samplers . samplers_map . get ( x . lower ( ) , None )
2022-11-27 13:17:39 +03:00
if sampler_name is None :
2022-09-03 19:32:45 +03:00
raise RuntimeError ( f " Unknown sampler: { x } " )
2022-11-27 13:17:39 +03:00
p . sampler_name = sampler_name
2022-09-03 19:32:45 +03:00
2022-10-09 20:20:35 +03:00
def confirm_samplers ( p , xs ) :
for x in xs :
2022-11-27 13:43:10 +03:00
if x . lower ( ) not in sd_samplers . samplers_map :
2022-10-09 20:20:35 +03:00
raise RuntimeError ( f " Unknown sampler: { x } " )
2022-09-03 19:32:45 +03:00
2022-09-17 13:49:36 +03:00
def apply_checkpoint ( p , x , xs ) :
2022-09-29 00:31:53 +03:00
info = modules . sd_models . get_closet_checkpoint_match ( x )
2022-10-09 20:20:35 +03:00
if info is None :
raise RuntimeError ( f " Unknown checkpoint: { x } " )
2022-09-17 13:49:36 +03:00
modules . sd_models . reload_model_weights ( shared . sd_model , info )
2022-10-09 20:20:35 +03:00
def confirm_checkpoints ( p , xs ) :
for x in xs :
if modules . sd_models . get_closet_checkpoint_match ( x ) is None :
raise RuntimeError ( f " Unknown checkpoint: { x } " )
2022-10-09 18:58:55 +03:00
def apply_clip_skip ( p , x , xs ) :
2022-10-09 21:57:17 +03:00
opts . data [ " CLIP_stop_at_last_layers " ] = x
2022-10-09 18:58:55 +03:00
2022-12-18 18:47:02 +03:00
def apply_upscale_latent_space ( p , x , xs ) :
if x . lower ( ) . strip ( ) != ' 0 ' :
opts . data [ " use_scale_latent_for_hires_fix " ] = True
else :
opts . data [ " use_scale_latent_for_hires_fix " ] = False
def find_vae ( name : str ) :
2023-01-14 19:56:09 +03:00
if name . lower ( ) in [ ' auto ' , ' automatic ' ] :
return modules . sd_vae . unspecified
if name . lower ( ) == ' none ' :
return None
2022-12-18 18:47:02 +03:00
else :
2023-01-14 19:56:09 +03:00
choices = [ x for x in sorted ( modules . sd_vae . vae_dict , key = lambda x : len ( x ) ) if name . lower ( ) . strip ( ) in x . lower ( ) ]
if len ( choices ) == 0 :
print ( f " No VAE found for { name } ; using automatic " )
return modules . sd_vae . unspecified
2022-12-18 18:47:02 +03:00
else :
2023-01-14 19:56:09 +03:00
return modules . sd_vae . vae_dict [ choices [ 0 ] ]
2022-12-18 18:47:02 +03:00
def apply_vae ( p , x , xs ) :
2023-01-14 19:56:09 +03:00
modules . sd_vae . reload_vae_weights ( shared . sd_model , vae_file = find_vae ( x ) )
2022-12-18 18:47:02 +03:00
def apply_styles ( p : StableDiffusionProcessingTxt2Img , x : str , _ ) :
p . styles = x . split ( ' , ' )
2022-09-17 13:49:36 +03:00
2022-09-03 19:32:45 +03:00
def format_value_add_label ( p , opt , x ) :
2022-09-09 18:05:43 +03:00
if type ( x ) == float :
x = round ( x , 8 )
2022-09-03 19:32:45 +03:00
return f " { opt . label } : { x } "
def format_value ( p , opt , x ) :
2022-09-09 18:05:43 +03:00
if type ( x ) == float :
x = round ( x , 8 )
2022-09-03 19:32:45 +03:00
return x
2022-10-04 09:18:00 +03:00
def format_value_join_list ( p , opt , x ) :
return " , " . join ( x )
2022-09-09 17:54:04 +03:00
def do_nothing ( p , x , xs ) :
pass
2022-10-04 09:18:00 +03:00
2022-09-09 17:54:04 +03:00
def format_nothing ( p , opt , x ) :
return " "
2022-09-03 19:32:45 +03:00
2022-10-04 09:18:00 +03:00
def str_permutations ( x ) :
""" dummy function for specifying it in AxisOption ' s type when you want to get a list of permutations """
return x
2023-01-16 17:36:56 +03:00
class AxisOption :
def __init__ ( self , label , type , apply , format_value = format_value_add_label , confirm = None , cost = 0.0 , choices = None ) :
self . label = label
self . type = type
self . apply = apply
self . format_value = format_value
self . confirm = confirm
self . cost = cost
self . choices = choices
class AxisOptionImg2Img ( AxisOption ) :
2023-01-21 22:43:37 +03:00
def __init__ ( self , * args , * * kwargs ) :
super ( ) . __init__ ( * args , * * kwargs )
self . is_img2img = True
class AxisOptionTxt2Img ( AxisOption ) :
2023-01-16 17:36:56 +03:00
def __init__ ( self , * args , * * kwargs ) :
super ( ) . __init__ ( * args , * * kwargs )
self . is_img2img = False
2022-09-03 19:32:45 +03:00
axis_options = [
2023-01-16 17:36:56 +03:00
AxisOption ( " Nothing " , str , do_nothing , format_value = format_nothing ) ,
AxisOption ( " Seed " , int , apply_field ( " seed " ) ) ,
AxisOption ( " Var. seed " , int , apply_field ( " subseed " ) ) ,
AxisOption ( " Var. strength " , float , apply_field ( " subseed_strength " ) ) ,
AxisOption ( " Steps " , int , apply_field ( " steps " ) ) ,
2023-01-23 01:08:08 +03:00
AxisOptionTxt2Img ( " Hires steps " , int , apply_field ( " hr_second_pass_steps " ) ) ,
2023-01-16 17:36:56 +03:00
AxisOption ( " CFG Scale " , float , apply_field ( " cfg_scale " ) ) ,
AxisOption ( " Prompt S/R " , str , apply_prompt , format_value = format_value ) ,
AxisOption ( " Prompt order " , str_permutations , apply_order , format_value = format_value_join_list ) ,
2023-01-21 22:43:37 +03:00
AxisOptionTxt2Img ( " Sampler " , str , apply_sampler , format_value = format_value , confirm = confirm_samplers , choices = lambda : [ x . name for x in sd_samplers . samplers ] ) ,
AxisOptionImg2Img ( " Sampler " , str , apply_sampler , format_value = format_value , confirm = confirm_samplers , choices = lambda : [ x . name for x in sd_samplers . samplers_for_img2img ] ) ,
2023-01-16 17:36:56 +03:00
AxisOption ( " Checkpoint name " , str , apply_checkpoint , format_value = format_value , confirm = confirm_checkpoints , cost = 1.0 , choices = lambda : list ( sd_models . checkpoints_list ) ) ,
AxisOption ( " Sigma Churn " , float , apply_field ( " s_churn " ) ) ,
AxisOption ( " Sigma min " , float , apply_field ( " s_tmin " ) ) ,
AxisOption ( " Sigma max " , float , apply_field ( " s_tmax " ) ) ,
AxisOption ( " Sigma noise " , float , apply_field ( " s_noise " ) ) ,
AxisOption ( " Eta " , float , apply_field ( " eta " ) ) ,
AxisOption ( " Clip skip " , int , apply_clip_skip ) ,
AxisOption ( " Denoising " , float , apply_field ( " denoising_strength " ) ) ,
2023-01-21 22:43:37 +03:00
AxisOptionTxt2Img ( " Hires upscaler " , str , apply_field ( " hr_upscaler " ) , choices = lambda : [ * shared . latent_upscale_modes , * [ x . name for x in shared . sd_upscalers ] ] ) ,
AxisOptionImg2Img ( " Cond. Image Mask Weight " , float , apply_field ( " inpainting_mask_weight " ) ) ,
2023-01-16 17:36:56 +03:00
AxisOption ( " VAE " , str , apply_vae , cost = 0.7 , choices = lambda : list ( sd_vae . vae_dict ) ) ,
AxisOption ( " Styles " , str , apply_styles , choices = lambda : list ( shared . prompt_styles . styles ) ) ,
2022-09-03 19:32:45 +03:00
]
2023-01-16 08:40:57 +03:00
def draw_xy_grid ( p , xs , ys , x_labels , y_labels , cell , draw_legend , include_lone_images , swap_axes_processing_order ) :
2022-09-17 14:55:40 +03:00
ver_texts = [ [ images . GridAnnotation ( y ) ] for y in y_labels ]
hor_texts = [ [ images . GridAnnotation ( x ) ] for x in x_labels ]
2022-09-03 19:32:45 +03:00
2022-10-13 02:12:12 +03:00
# Temporary list of all the images that are generated to be populated into the grid.
# Will be filled with empty images for any individual step that fails to process properly
2023-01-16 08:40:57 +03:00
image_cache = [ None ] * ( len ( xs ) * len ( ys ) )
2022-10-13 02:12:12 +03:00
processed_result = None
cell_mode = " P "
2023-01-16 08:40:57 +03:00
cell_size = ( 1 , 1 )
2022-09-03 19:32:45 +03:00
2022-09-14 13:08:05 +03:00
state . job_count = len ( xs ) * len ( ys ) * p . n_iter
2022-09-06 02:09:01 +03:00
2023-01-16 08:40:57 +03:00
def process_cell ( x , y , ix , iy ) :
nonlocal image_cache , processed_result , cell_mode , cell_size
state . job = f " { ix + iy * len ( xs ) + 1 } out of { len ( xs ) * len ( ys ) } "
processed : Processed = cell ( x , y )
try :
# this dereference will throw an exception if the image was not processed
# (this happens in cases such as if the user stops the process from the UI)
processed_image = processed . images [ 0 ]
if processed_result is None :
# Use our first valid processed result as a template container to hold our full results
processed_result = copy ( processed )
cell_mode = processed_image . mode
cell_size = processed_image . size
processed_result . images = [ Image . new ( cell_mode , cell_size ) ]
image_cache [ ix + iy * len ( xs ) ] = processed_image
if include_lone_images :
processed_result . images . append ( processed_image )
processed_result . all_prompts . append ( processed . prompt )
processed_result . all_seeds . append ( processed . seed )
processed_result . infotexts . append ( processed . infotexts [ 0 ] )
except :
image_cache [ ix + iy * len ( xs ) ] = Image . new ( cell_mode , cell_size )
if swap_axes_processing_order :
2022-09-03 19:32:45 +03:00
for ix , x in enumerate ( xs ) :
2023-01-16 08:40:57 +03:00
for iy , y in enumerate ( ys ) :
process_cell ( x , y , ix , iy )
else :
for iy , y in enumerate ( ys ) :
for ix , x in enumerate ( xs ) :
process_cell ( x , y , ix , iy )
2022-10-13 02:12:12 +03:00
if not processed_result :
print ( " Unexpected error: draw_xy_grid failed to return even a single processed image " )
2023-01-16 17:36:56 +03:00
return Processed ( p , [ ] )
2022-09-03 19:32:45 +03:00
2022-10-13 02:12:12 +03:00
grid = images . image_grid ( image_cache , rows = len ( ys ) )
2022-09-14 15:01:16 +03:00
if draw_legend :
2022-10-13 02:12:12 +03:00
grid = images . draw_grid_annotations ( grid , cell_size [ 0 ] , cell_size [ 1 ] , hor_texts , ver_texts )
2022-09-03 19:32:45 +03:00
2022-10-13 02:12:12 +03:00
processed_result . images [ 0 ] = grid
2022-09-03 19:32:45 +03:00
2022-10-13 02:12:12 +03:00
return processed_result
2022-09-03 19:32:45 +03:00
2022-10-16 22:10:07 +03:00
class SharedSettingsStackHelper ( object ) :
def __enter__ ( self ) :
self . CLIP_stop_at_last_layers = opts . CLIP_stop_at_last_layers
2022-12-18 18:47:02 +03:00
self . vae = opts . sd_vae
2022-10-16 22:10:07 +03:00
def __exit__ ( self , exc_type , exc_value , tb ) :
2023-01-14 19:56:09 +03:00
opts . data [ " sd_vae " ] = self . vae
2023-01-16 09:27:52 +03:00
modules . sd_models . reload_model_weights ( )
modules . sd_vae . reload_vae_weights ( )
2022-09-03 19:32:45 +03:00
2022-10-16 22:10:07 +03:00
opts . data [ " CLIP_stop_at_last_layers " ] = self . CLIP_stop_at_last_layers
2022-09-03 19:32:45 +03:00
2022-09-06 10:11:25 +03:00
re_range = re . compile ( r " \ s*([+-]? \ s* \ d+) \ s*- \ s*([+-]? \ s* \ d+)(?: \ s* \ (([+-] \ d+) \ s* \ ))? \ s* " )
2022-09-09 17:54:04 +03:00
re_range_float = re . compile ( r " \ s*([+-]? \ s* \ d+(?:. \ d*)?) \ s*- \ s*([+-]? \ s* \ d+(?:. \ d*)?)(?: \ s* \ (([+-] \ d+(?:. \ d*)?) \ s* \ ))? \ s* " )
2022-09-06 10:11:25 +03:00
2022-09-14 14:56:26 +03:00
re_range_count = re . compile ( r " \ s*([+-]? \ s* \ d+) \ s*- \ s*([+-]? \ s* \ d+)(?: \ s* \ [( \ d+) \ s* \ ])? \ s* " )
re_range_count_float = re . compile ( r " \ s*([+-]? \ s* \ d+(?:. \ d*)?) \ s*- \ s*([+-]? \ s* \ d+(?:. \ d*)?)(?: \ s* \ [( \ d+(?:. \ d*)?) \ s* \ ])? \ s* " )
2023-01-04 19:19:11 +03:00
2022-09-03 19:32:45 +03:00
class Script ( scripts . Script ) :
def title ( self ) :
return " X/Y plot "
def ui ( self , is_img2img ) :
2023-01-21 23:58:59 +03:00
self . current_axis_options = [ x for x in axis_options if type ( x ) == AxisOption or x . is_img2img == is_img2img ]
2022-09-03 19:32:45 +03:00
with gr . Row ( ) :
2023-01-16 08:41:58 +03:00
with gr . Column ( scale = 19 ) :
with gr . Row ( ) :
2023-01-21 23:58:59 +03:00
x_type = gr . Dropdown ( label = " X type " , choices = [ x . label for x in self . current_axis_options ] , value = self . current_axis_options [ 1 ] . label , type = " index " , elem_id = self . elem_id ( " x_type " ) )
2023-01-16 08:41:58 +03:00
x_values = gr . Textbox ( label = " X values " , lines = 1 , elem_id = self . elem_id ( " x_values " ) )
2023-01-16 17:36:56 +03:00
fill_x_button = ToolButton ( value = fill_values_symbol , elem_id = " xy_grid_fill_x_tool_button " , visible = False )
2023-01-16 08:41:58 +03:00
with gr . Row ( ) :
2023-01-21 23:58:59 +03:00
y_type = gr . Dropdown ( label = " Y type " , choices = [ x . label for x in self . current_axis_options ] , value = self . current_axis_options [ 0 ] . label , type = " index " , elem_id = self . elem_id ( " y_type " ) )
2023-01-16 08:41:58 +03:00
y_values = gr . Textbox ( label = " Y values " , lines = 1 , elem_id = self . elem_id ( " y_values " ) )
2023-01-16 17:36:56 +03:00
fill_y_button = ToolButton ( value = fill_values_symbol , elem_id = " xy_grid_fill_y_tool_button " , visible = False )
2023-01-22 00:58:45 +03:00
with gr . Row ( variant = " compact " , elem_id = " axis_options " ) :
2023-01-16 17:36:56 +03:00
draw_legend = gr . Checkbox ( label = ' Draw legend ' , value = True , elem_id = self . elem_id ( " draw_legend " ) )
include_lone_images = gr . Checkbox ( label = ' Include Separate Images ' , value = False , elem_id = self . elem_id ( " include_lone_images " ) )
no_fixed_seeds = gr . Checkbox ( label = ' Keep -1 for seeds ' , value = False , elem_id = self . elem_id ( " no_fixed_seeds " ) )
swap_axes_button = gr . Button ( value = " Swap axes " , elem_id = " xy_grid_swap_axes_button " )
2022-09-03 19:32:45 +03:00
2023-01-16 08:41:58 +03:00
def swap_axes ( x_type , x_values , y_type , y_values ) :
2023-01-21 23:58:59 +03:00
return self . current_axis_options [ y_type ] . label , y_values , self . current_axis_options [ x_type ] . label , x_values
2023-01-16 08:41:58 +03:00
swap_args = [ x_type , x_values , y_type , y_values ]
swap_axes_button . click ( swap_axes , inputs = swap_args , outputs = swap_args )
2023-01-16 17:36:56 +03:00
def fill ( x_type ) :
2023-01-21 23:58:59 +03:00
axis = self . current_axis_options [ x_type ]
2023-01-16 17:36:56 +03:00
return " , " . join ( axis . choices ( ) ) if axis . choices else gr . update ( )
fill_x_button . click ( fn = fill , inputs = [ x_type ] , outputs = [ x_values ] )
fill_y_button . click ( fn = fill , inputs = [ y_type ] , outputs = [ y_values ] )
def select_axis ( x_type ) :
2023-01-21 23:58:59 +03:00
return gr . Button . update ( visible = self . current_axis_options [ x_type ] . choices is not None )
2023-01-16 17:36:56 +03:00
x_type . change ( fn = select_axis , inputs = [ x_type ] , outputs = [ fill_x_button ] )
y_type . change ( fn = select_axis , inputs = [ y_type ] , outputs = [ fill_y_button ] )
2022-10-12 03:05:20 +03:00
return [ x_type , x_values , y_type , y_values , draw_legend , include_lone_images , no_fixed_seeds ]
2022-09-24 08:09:59 +03:00
2022-10-12 03:05:20 +03:00
def run ( self , p , x_type , x_values , y_type , y_values , draw_legend , include_lone_images , no_fixed_seeds ) :
2022-10-07 02:31:36 +03:00
if not no_fixed_seeds :
modules . processing . fix_seed ( p )
2022-10-10 21:24:11 +03:00
if not opts . return_grid :
p . batch_size = 1
2022-09-03 19:32:45 +03:00
def process_axis ( opt , vals ) :
2022-09-29 21:16:12 +03:00
if opt . label == ' Nothing ' :
return [ 0 ]
2022-10-06 20:16:21 +03:00
valslist = [ x . strip ( ) for x in chain . from_iterable ( csv . reader ( StringIO ( vals ) ) ) ]
2022-09-03 19:32:45 +03:00
if opt . type == int :
valslist_ext = [ ]
for val in valslist :
2022-09-06 10:11:25 +03:00
m = re_range . fullmatch ( val )
2022-09-14 14:56:26 +03:00
mc = re_range_count . fullmatch ( val )
2022-09-06 10:11:25 +03:00
if m is not None :
start = int ( m . group ( 1 ) )
end = int ( m . group ( 2 ) ) + 1
step = int ( m . group ( 3 ) ) if m . group ( 3 ) is not None else 1
2022-09-03 19:32:45 +03:00
valslist_ext + = list ( range ( start , end , step ) )
2022-09-14 14:56:26 +03:00
elif mc is not None :
start = int ( mc . group ( 1 ) )
end = int ( mc . group ( 2 ) )
num = int ( mc . group ( 3 ) ) if mc . group ( 3 ) is not None else 1
2022-09-24 08:23:01 +03:00
valslist_ext + = [ int ( x ) for x in np . linspace ( start = start , stop = end , num = num ) . tolist ( ) ]
2022-09-03 19:32:45 +03:00
else :
valslist_ext . append ( val )
valslist = valslist_ext
2022-09-09 17:54:04 +03:00
elif opt . type == float :
valslist_ext = [ ]
for val in valslist :
m = re_range_float . fullmatch ( val )
2022-09-14 14:56:26 +03:00
mc = re_range_count_float . fullmatch ( val )
2022-09-09 17:54:04 +03:00
if m is not None :
start = float ( m . group ( 1 ) )
end = float ( m . group ( 2 ) )
step = float ( m . group ( 3 ) ) if m . group ( 3 ) is not None else 1
valslist_ext + = np . arange ( start , end + step , step ) . tolist ( )
2022-09-14 14:56:26 +03:00
elif mc is not None :
start = float ( mc . group ( 1 ) )
end = float ( mc . group ( 2 ) )
num = int ( mc . group ( 3 ) ) if mc . group ( 3 ) is not None else 1
2022-09-24 08:23:01 +03:00
valslist_ext + = np . linspace ( start = start , stop = end , num = num ) . tolist ( )
2022-09-09 17:54:04 +03:00
else :
valslist_ext . append ( val )
valslist = valslist_ext
2022-10-04 09:18:00 +03:00
elif opt . type == str_permutations :
valslist = list ( permutations ( valslist ) )
2022-09-03 19:32:45 +03:00
valslist = [ opt . type ( x ) for x in valslist ]
2022-10-08 08:30:49 +03:00
# Confirm options are valid before starting
2022-10-09 20:20:35 +03:00
if opt . confirm :
opt . confirm ( p , valslist )
2022-09-03 19:32:45 +03:00
return valslist
2023-01-21 23:58:59 +03:00
x_opt = self . current_axis_options [ x_type ]
2022-09-03 19:32:45 +03:00
xs = process_axis ( x_opt , x_values )
2023-01-21 23:58:59 +03:00
y_opt = self . current_axis_options [ y_type ]
2022-09-03 19:32:45 +03:00
ys = process_axis ( y_opt , y_values )
2022-09-24 08:09:59 +03:00
def fix_axis_seeds ( axis_opt , axis_list ) :
2023-01-04 19:19:11 +03:00
if axis_opt . label in [ ' Seed ' , ' Var. seed ' ] :
2022-09-24 08:09:59 +03:00
return [ int ( random . randrange ( 4294967294 ) ) if val is None or val == ' ' or val == - 1 else val for val in axis_list ]
else :
return axis_list
2022-09-24 08:23:01 +03:00
if not no_fixed_seeds :
2022-09-24 08:09:59 +03:00
xs = fix_axis_seeds ( x_opt , xs )
ys = fix_axis_seeds ( y_opt , ys )
if x_opt . label == ' Steps ' :
total_steps = sum ( xs ) * len ( ys )
elif y_opt . label == ' Steps ' :
total_steps = sum ( ys ) * len ( xs )
else :
total_steps = p . steps * len ( xs ) * len ( ys )
2022-10-15 00:26:38 +03:00
if isinstance ( p , StableDiffusionProcessingTxt2Img ) and p . enable_hr :
2023-01-23 01:08:08 +03:00
if x_opt . label == " Hires steps " :
total_steps + = sum ( xs ) * len ( ys )
elif y_opt . label == " Hires steps " :
total_steps + = sum ( ys ) * len ( xs )
elif p . hr_second_pass_steps :
total_steps + = p . hr_second_pass_steps * len ( xs ) * len ( ys )
else :
total_steps * = 2
total_steps * = p . n_iter
2022-10-15 00:26:38 +03:00
2023-01-23 01:08:08 +03:00
image_cell_count = p . n_iter * p . batch_size
cell_console_text = f " ; { image_cell_count } images per cell " if image_cell_count > 1 else " "
print ( f " X/Y plot will create { len ( xs ) * len ( ys ) * image_cell_count } images on a { len ( xs ) } x { len ( ys ) } grid { cell_console_text } . (Total steps to process: { total_steps } ) " )
shared . total_tqdm . updateTotal ( total_steps )
2022-09-24 08:09:59 +03:00
2023-01-04 19:19:11 +03:00
grid_infotext = [ None ]
2023-01-16 08:40:57 +03:00
# If one of the axes is very slow to change between (like SD model
# checkpoint), then make sure it is in the outer iteration of the nested
# `for` loop.
swap_axes_processing_order = x_opt . cost > y_opt . cost
2022-09-03 19:32:45 +03:00
def cell ( x , y ) :
2023-01-16 06:43:34 +03:00
if shared . state . interrupted :
return Processed ( p , [ ] , p . seed , " " )
2022-09-03 19:32:45 +03:00
pc = copy ( p )
x_opt . apply ( pc , x , xs )
y_opt . apply ( pc , y , ys )
2023-01-04 19:19:11 +03:00
res = process_images ( pc )
if grid_infotext [ 0 ] is None :
pc . extra_generation_params = copy ( pc . extra_generation_params )
if x_opt . label != ' Nothing ' :
pc . extra_generation_params [ " X Type " ] = x_opt . label
pc . extra_generation_params [ " X Values " ] = x_values
if x_opt . label in [ " Seed " , " Var. seed " ] and not no_fixed_seeds :
pc . extra_generation_params [ " Fixed X Values " ] = " , " . join ( [ str ( x ) for x in xs ] )
if y_opt . label != ' Nothing ' :
pc . extra_generation_params [ " Y Type " ] = y_opt . label
pc . extra_generation_params [ " Y Values " ] = y_values
if y_opt . label in [ " Seed " , " Var. seed " ] and not no_fixed_seeds :
pc . extra_generation_params [ " Fixed Y Values " ] = " , " . join ( [ str ( y ) for y in ys ] )
2022-09-03 19:32:45 +03:00
2023-01-04 19:19:11 +03:00
grid_infotext [ 0 ] = processing . create_infotext ( pc , pc . all_prompts , pc . all_seeds , pc . all_subseeds )
2022-10-01 08:02:29 +03:00
2023-01-04 19:19:11 +03:00
return res
2022-10-01 08:02:29 +03:00
2022-10-16 22:10:07 +03:00
with SharedSettingsStackHelper ( ) :
processed = draw_xy_grid (
p ,
xs = xs ,
ys = ys ,
x_labels = [ x_opt . format_value ( p , x_opt , x ) for x in xs ] ,
y_labels = [ y_opt . format_value ( p , y_opt , y ) for y in ys ] ,
cell = cell ,
draw_legend = draw_legend ,
2023-01-16 08:40:57 +03:00
include_lone_images = include_lone_images ,
swap_axes_processing_order = swap_axes_processing_order
2022-10-16 22:10:07 +03:00
)
2022-09-03 19:32:45 +03:00
2022-09-04 03:38:24 +03:00
if opts . grid_save :
2023-01-04 19:19:11 +03:00
images . save_image ( processed . images [ 0 ] , p . outpath_grids , " xy_grid " , info = grid_infotext [ 0 ] , extension = opts . grid_format , prompt = p . prompt , seed = processed . seed , grid = True , p = p )
2022-09-17 13:49:36 +03:00
2022-09-03 19:32:45 +03:00
return processed