stable-diffusion-webui-rand.../scripts/randomize.py
MMaker 895fea03f0
Support new hires. fix
Breaking change, requires latest webui version
Closes #11
2023-01-02 17:06:22 -05:00

280 lines
12 KiB
Python

import math
import random
import gradio as gr
from modules import scripts, sd_models, shared
from modules.processing import (StableDiffusionProcessing,
StableDiffusionProcessingTxt2Img)
from modules.shared import opts, cmd_opts, state
from modules.sd_models import checkpoints_list
from modules.sd_samplers import all_samplers_map, samplers
from modules.hypernetworks import hypernetwork
try:
from scripts.xy_grid import build_samplers_dict # type: ignore
except ImportError:
def build_samplers_dict():
return {}
class RandomizeScript(scripts.Script):
def __init__(self) -> None:
super().__init__()
def title(self):
return 'Randomize'
def show(self, is_img2img):
if not is_img2img:
return scripts.AlwaysVisible
def ui(self, is_img2img):
randomize_enabled, randomize_param_sampler_name, randomize_param_cfg_scale, randomize_param_steps, randomize_param_width, randomize_param_height, randomize_hires_chance, randomize_hr_hr_upscaler, randomize_hr_hr_scale, randomize_hr_denoising_strength, randomize_other_CLIP_stop_at_last_layers, randomize_other_sd_model_checkpoint, randomize_other_sd_hypernetwork, randomize_other_sd_hypernetwork_strength, randomize_other_eta_noise_seed_delta, randomize_other_styles = self._create_ui()
return [randomize_enabled, randomize_param_sampler_name, randomize_param_cfg_scale, randomize_param_steps, randomize_param_width, randomize_param_height, randomize_hires_chance, randomize_hr_hr_upscaler, randomize_hr_hr_scale, randomize_hr_denoising_strength, randomize_other_CLIP_stop_at_last_layers, randomize_other_sd_model_checkpoint, randomize_other_sd_hypernetwork, randomize_other_sd_hypernetwork_strength, randomize_other_eta_noise_seed_delta, randomize_other_styles]
def process(
self,
p: StableDiffusionProcessing,
randomize_enabled: bool,
# randomize_param_seed: str,
randomize_param_sampler_name: str,
randomize_param_cfg_scale: str,
randomize_param_steps: str,
randomize_param_width: str,
randomize_param_height: str,
randomize_hires_chance: str,
randomize_hr_hr_upscaler: str,
randomize_hr_hr_scale: str,
randomize_hr_denoising_strength: str,
randomize_other_CLIP_stop_at_last_layers: str,
randomize_other_sd_model_checkpoint: str,
randomize_other_sd_hypernetwork: str,
randomize_other_sd_hypernetwork_strength: str,
randomize_other_eta_noise_seed_delta: str,
randomize_other_styles: str,
**kwargs
):
if randomize_enabled and isinstance(p, StableDiffusionProcessingTxt2Img):
self.hypernetwork = opts.sd_hypernetwork
self.hypernetwork_strength = opts.sd_hypernetwork_strength
all_opts = {k: v for k, v in locals().items() if k not in ['self', 'p', 'randomize_enabled', 'batch_number', 'prompts', 'seeds', 'subseeds']}
# NOTE (mmaker): Can we update these in the UI?
for param, val in self._list_params(all_opts, prefix='randomize_other_'):
if param == 'sd_model_checkpoint':
sd_model_checkpoint = self._opt({param: val}, p)
if sd_model_checkpoint:
sd_models.reload_model_weights(shared.sd_model, sd_model_checkpoint)
p.sd_model = shared.sd_model
if param == 'sd_hypernetwork':
hypernetwork.load_hypernetwork(self._opt({param: val}, p))
if param == 'sd_hypernetwork_strength':
hypernetwork.apply_strength(self._opt({param: val}, p))
opts.data[param] = self._opt({param: val}, p) # type: ignore
if param == 'styles':
p.styles = self._opt({param: val}, p) # type: ignore
self._apply_styles(p)
def process_batch(
self,
p: StableDiffusionProcessing,
randomize_enabled: bool,
# randomize_param_seed: str,
randomize_param_sampler_name: str,
randomize_param_cfg_scale: str,
randomize_param_steps: str,
randomize_param_width: str,
randomize_param_height: str,
randomize_hires_chance: str,
randomize_hr_hr_upscaler: str,
randomize_hr_hr_scale: str,
randomize_hr_denoising_strength: str,
randomize_other_CLIP_stop_at_last_layers: str,
randomize_other_sd_model_checkpoint: str,
randomize_other_sd_hypernetwork: str,
randomize_other_sd_hypernetwork_strength: str,
randomize_other_eta_noise_seed_delta: str,
randomize_other_styles: str,
**kwargs
):
if randomize_enabled and isinstance(p, StableDiffusionProcessingTxt2Img):
self.CLIP_stop_at_last_layers = opts.CLIP_stop_at_last_layers
self.eta_noise_seed_delta = opts.eta_noise_seed_delta
# TODO (mmaker): Fix this jank. Don't do this.
all_opts = {k: v for k, v in locals().items() if k not in ['self', 'p', 'randomize_enabled', 'batch_number', 'prompts', 'seeds', 'subseeds']}
# Base params
for param, val in self._list_params(all_opts):
self._set_attr(p, param, val)
# Other params
for param, val in self._list_params(all_opts, prefix='randomize_other_'):
if param in ['CLIP_stop_at_last_layers', 'eta_noise_seed_delta']:
opts.data[param] = self._opt({param: val}, p) # type: ignore
if len(randomize_hires_chance.strip()) > 0:
fix_job_count = False
if p.enable_hr:
fix_job_count = True
if random.random() < float(randomize_hires_chance or 0):
setattr(p, 'enable_hr', True)
if not p.denoising_strength:
setattr(p, 'denoising_strength', 0.75)
for param, val in self._list_params(all_opts, prefix='randomize_hr_'):
self._set_attr(p, param, val)
p.init(p.all_prompts, p.all_seeds, p.all_subseeds)
if fix_job_count:
state.job_count = math.floor(state.job_count / 2)
else:
setattr(p, 'enable_hr', False)
p.init(p.all_prompts, p.all_seeds, p.all_subseeds)
if fix_job_count:
state.job_count = math.floor(state.job_count / 2)
else:
return
def postprocess(self, p, processed, *args):
if isinstance(p, StableDiffusionProcessingTxt2Img):
# I don't think these checks are needed, but just for good measure for now
if hasattr(self, 'hypernetwork'):
hypernetwork.load_hypernetwork(self.hypernetwork)
hypernetwork.apply_strength(self.hypernetwork_strength)
if hasattr(self, 'CLIP_stop_at_last_layers'):
opts.data["CLIP_stop_at_last_layers"] = self.CLIP_stop_at_last_layers # type: ignore
opts.data["eta_noise_seed_delta"] = self.eta_noise_seed_delta # type: ignore
def _set_attr(self, p, param, val):
try:
# Backwards compat
if param == 'sampler_name' and hasattr(p, 'sampler_index') and not hasattr(p, 'sampler_name'):
param = 'sampler_index'
opt = self._opt({param: val}, p)
if opt is not None:
if param in ['seed']:
setattr(p, 'all_seeds', [self._opt({param: val}, p) for _ in range(0, len(getattr(p, 'all_seeds')))]) # NOTE (mmaker): Is this correct?
else:
setattr(p, param, opt)
else:
print(f'Skipping randomizing param `{param}` -- incorrect value')
except (TypeError, IndexError) as exception:
print(f'Failed to randomize param `{param}` -- incorrect value?', exception)
def _list_params(self, opts, prefix='randomize_param_'):
for k, v in opts.items():
if k.startswith(prefix) and v is not None and len(v) > 0:
yield k.replace(prefix,''), v
def _opt(self, opt, p):
opt_name = list(opt.keys())[0]
opt_val = list(opt.values())[0].strip()
opt_arr: list[str] = [x.strip() for x in opt_val.split(',')]
if self._is_num(opt_arr[0]) and len(opt_arr) == 3 and opt_name not in ['seed']:
vals = [float(v) for v in opt_arr]
rand = self._rand(vals[0], vals[1], vals[2])
if rand.is_integer():
return int(rand)
else:
return round(float(rand), max(0, int(opt_arr[2][::-1].find('.'))))
else:
if opt_name == 'sampler_name':
if opt_val == '*':
return random.choice([s.name for s in samplers])
random_sampler = random.choice(opt_arr)
if random_sampler in all_samplers_map:
return random_sampler
if opt_name == 'sampler_index':
samplers_dict = build_samplers_dict()
if len(samplers_dict) > 0:
if opt_val == '*':
return random.choice(list(samplers_dict.values()))
return samplers_dict.get(random.choice(opt_arr).lower(), None)
else:
return None
if opt_name == 'seed':
return int(random.choice(opt_arr))
if opt_name == 'sd_model_checkpoint':
if opt_val == '*':
return random.choice(list(checkpoints_list.values()))
return sd_models.get_closet_checkpoint_match(random.choice(opt_arr))
if opt_name == 'sd_hypernetwork':
if opt_val.lower() == 'none':
return None
if opt_val == '*':
return random.choice(list(hypernetwork.list_hypernetworks(cmd_opts.hypernetwork_dir).keys()))
return hypernetwork.find_closest_hypernetwork_name(random.choice(opt_arr))
if opt_name == 'styles':
if opt_val == '*':
return [random.choice([k for k, v in shared.prompt_styles.styles.items() if k != 'None'])]
return [random.choice(opt_arr)]
if opt_name == 'hr_upscaler':
if opt_val == '*':
return random.choice([*shared.latent_upscale_modes, *[x.name for x in shared.sd_upscalers]])
return random.choice(opt_arr)
return None
def _rand(self, start: float, stop: float, step: float) -> float:
return random.randint(0, int((stop - start) / step)) * step + start
def _is_num(self, val: str):
if val.isdigit():
return True
else:
try:
float(val)
return True
except ValueError:
return False
def _apply_styles(self, p: StableDiffusionProcessing):
# Copy-pasted code from processing.py
# Not at all smart but there's no way to run this again cleanly currently
# Needs to be ran again since normally, this is ran *before* scripts
if type(p.prompt) == list:
p.all_prompts = [shared.prompt_styles.apply_styles_to_prompt(x, p.styles) for x in p.prompt]
else:
p.all_prompts = p.batch_size * p.n_iter * [shared.prompt_styles.apply_styles_to_prompt(p.prompt, p.styles)]
if type(p.negative_prompt) == list:
p.all_negative_prompts = [shared.prompt_styles.apply_negative_styles_to_prompt(x, p.styles) for x in p.negative_prompt]
else:
p.all_negative_prompts = p.batch_size * p.n_iter * [shared.prompt_styles.apply_negative_styles_to_prompt(p.negative_prompt, p.styles)]
def _create_ui(self):
hint_minmax = 'Range of stepped values (min, max, step)'
hint_list = 'Comma separated list OR * for all'
hint_float = 'Float value from 0 to 1'
with gr.Group():
with gr.Accordion('Randomize', open=False):
randomize_enabled = gr.Checkbox(label='Enable', value=False)
# randomize_param_seed = gr.Textbox(label='Seed', value='', placeholder=hint_list)
randomize_param_sampler_name = gr.Textbox(label='Sampler', value='', placeholder=hint_list)
randomize_param_cfg_scale = gr.Textbox(label='CFG Scale', value='', placeholder=hint_minmax)
randomize_param_steps = gr.Textbox(label='Steps', value='', placeholder=hint_minmax)
randomize_param_width = gr.Textbox(label='Width', value='', placeholder=hint_minmax)
randomize_param_height = gr.Textbox(label='Height', value='', placeholder=hint_minmax)
randomize_hires_chance = gr.Textbox(label='Highres. percentage chance', value='', placeholder=hint_float)
randomize_hr_hr_upscaler = gr.Textbox(label='Highres. Upscaler', value='', placeholder=hint_list)
randomize_hr_hr_scale = gr.Textbox(label='Highres. Upscale by', value='', placeholder=hint_minmax)
randomize_hr_denoising_strength = gr.Textbox(label='Highres. Denoising Strength', value='', placeholder=hint_minmax)
randomize_other_CLIP_stop_at_last_layers = gr.Textbox(label='Stop at CLIP layers', value='', placeholder=hint_minmax)
randomize_other_sd_model_checkpoint = gr.Textbox(label='Checkpoint name', value='', placeholder=hint_list)
randomize_other_sd_hypernetwork = gr.Textbox(label='Hypernetwork', value='', placeholder=hint_list)
randomize_other_sd_hypernetwork_strength = gr.Textbox(label='Hypernetwork strength', value='', placeholder=hint_minmax)
randomize_other_eta_noise_seed_delta = gr.Textbox(label='Eta noise seed delta', value='', placeholder=hint_minmax)
randomize_other_styles = gr.Textbox(label='Styles', value='', placeholder=hint_list)
return randomize_enabled, randomize_param_sampler_name, randomize_param_cfg_scale, randomize_param_steps, randomize_param_width, randomize_param_height, randomize_hires_chance, randomize_hr_hr_upscaler, randomize_hr_hr_scale, randomize_hr_denoising_strength, randomize_other_CLIP_stop_at_last_layers, randomize_other_sd_model_checkpoint, randomize_other_sd_hypernetwork, randomize_other_sd_hypernetwork_strength, randomize_other_eta_noise_seed_delta, randomize_other_styles