prompt_matrix.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. import math
  2. import modules.scripts as scripts
  3. import gradio as gr
  4. from modules import images
  5. from modules.processing import process_images
  6. from modules.shared import opts, state
  7. import modules.sd_samplers
  8. def draw_xy_grid(xs, ys, x_label, y_label, cell):
  9. res = []
  10. ver_texts = [[images.GridAnnotation(y_label(y))] for y in ys]
  11. hor_texts = [[images.GridAnnotation(x_label(x))] for x in xs]
  12. first_processed = None
  13. state.job_count = len(xs) * len(ys)
  14. for iy, y in enumerate(ys):
  15. for ix, x in enumerate(xs):
  16. state.job = f"{ix + iy * len(xs) + 1} out of {len(xs) * len(ys)}"
  17. processed = cell(x, y)
  18. if first_processed is None:
  19. first_processed = processed
  20. res.append(processed.images[0])
  21. grid = images.image_grid(res, rows=len(ys))
  22. grid = images.draw_grid_annotations(grid, res[0].width, res[0].height, hor_texts, ver_texts)
  23. first_processed.images = [grid]
  24. return first_processed
  25. class Script(scripts.Script):
  26. def title(self):
  27. return "Prompt matrix"
  28. def ui(self, is_img2img):
  29. gr.HTML('<br />')
  30. with gr.Row():
  31. with gr.Column():
  32. put_at_start = gr.Checkbox(label='Put variable parts at start of prompt', value=False, elem_id=self.elem_id("put_at_start"))
  33. different_seeds = gr.Checkbox(label='Use different seed for each picture', value=False, elem_id=self.elem_id("different_seeds"))
  34. with gr.Column():
  35. prompt_type = gr.Radio(["positive", "negative"], label="Select prompt", elem_id=self.elem_id("prompt_type"), value="positive")
  36. variations_delimiter = gr.Radio(["comma", "space"], label="Select joining char", elem_id=self.elem_id("variations_delimiter"), value="comma")
  37. with gr.Column():
  38. margin_size = gr.Slider(label="Grid margins (px)", minimum=0, maximum=500, value=0, step=2, elem_id=self.elem_id("margin_size"))
  39. return [put_at_start, different_seeds, prompt_type, variations_delimiter, margin_size]
  40. def run(self, p, put_at_start, different_seeds, prompt_type, variations_delimiter, margin_size):
  41. modules.processing.fix_seed(p)
  42. # Raise error if promp type is not positive or negative
  43. if prompt_type not in ["positive", "negative"]:
  44. raise ValueError(f"Unknown prompt type {prompt_type}")
  45. # Raise error if variations delimiter is not comma or space
  46. if variations_delimiter not in ["comma", "space"]:
  47. raise ValueError(f"Unknown variations delimiter {variations_delimiter}")
  48. prompt = p.prompt if prompt_type == "positive" else p.negative_prompt
  49. original_prompt = prompt[0] if type(prompt) == list else prompt
  50. positive_prompt = p.prompt[0] if type(p.prompt) == list else p.prompt
  51. delimiter = ", " if variations_delimiter == "comma" else " "
  52. all_prompts = []
  53. prompt_matrix_parts = original_prompt.split("|")
  54. combination_count = 2 ** (len(prompt_matrix_parts) - 1)
  55. for combination_num in range(combination_count):
  56. selected_prompts = [text.strip().strip(',') for n, text in enumerate(prompt_matrix_parts[1:]) if combination_num & (1 << n)]
  57. if put_at_start:
  58. selected_prompts = selected_prompts + [prompt_matrix_parts[0]]
  59. else:
  60. selected_prompts = [prompt_matrix_parts[0]] + selected_prompts
  61. all_prompts.append(delimiter.join(selected_prompts))
  62. p.n_iter = math.ceil(len(all_prompts) / p.batch_size)
  63. p.do_not_save_grid = True
  64. print(f"Prompt matrix will create {len(all_prompts)} images using a total of {p.n_iter} batches.")
  65. if prompt_type == "positive":
  66. p.prompt = all_prompts
  67. else:
  68. p.negative_prompt = all_prompts
  69. p.seed = [p.seed + (i if different_seeds else 0) for i in range(len(all_prompts))]
  70. p.prompt_for_display = positive_prompt
  71. processed = process_images(p)
  72. grid = images.image_grid(processed.images, p.batch_size, rows=1 << ((len(prompt_matrix_parts) - 1) // 2))
  73. grid = images.draw_prompt_matrix(grid, processed.images[0].width, processed.images[0].height, prompt_matrix_parts, margin_size)
  74. processed.images.insert(0, grid)
  75. processed.index_of_first_image = 1
  76. processed.infotexts.insert(0, processed.infotexts[0])
  77. if opts.grid_save:
  78. images.save_image(processed.images[0], p.outpath_grids, "prompt_matrix", extension=opts.grid_format, prompt=original_prompt, seed=processed.seed, grid=True, p=p)
  79. return processed