ui_edit_user_metadata.py 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. import datetime
  2. import html
  3. import random
  4. import gradio as gr
  5. import re
  6. from modules import ui_extra_networks_user_metadata
  7. def is_non_comma_tagset(tags):
  8. average_tag_length = sum(len(x) for x in tags.keys()) / len(tags)
  9. return average_tag_length >= 16
  10. re_word = re.compile(r"[-_\w']+")
  11. re_comma = re.compile(r" *, *")
  12. def build_tags(metadata):
  13. tags = {}
  14. for _, tags_dict in metadata.get("ss_tag_frequency", {}).items():
  15. for tag, tag_count in tags_dict.items():
  16. tag = tag.strip()
  17. tags[tag] = tags.get(tag, 0) + int(tag_count)
  18. if tags and is_non_comma_tagset(tags):
  19. new_tags = {}
  20. for text, text_count in tags.items():
  21. for word in re.findall(re_word, text):
  22. if len(word) < 3:
  23. continue
  24. new_tags[word] = new_tags.get(word, 0) + text_count
  25. tags = new_tags
  26. ordered_tags = sorted(tags.keys(), key=tags.get, reverse=True)
  27. return [(tag, tags[tag]) for tag in ordered_tags]
  28. class LoraUserMetadataEditor(ui_extra_networks_user_metadata.UserMetadataEditor):
  29. def __init__(self, ui, tabname, page):
  30. super().__init__(ui, tabname, page)
  31. self.select_sd_version = None
  32. self.taginfo = None
  33. self.edit_activation_text = None
  34. self.slider_preferred_weight = None
  35. self.edit_notes = None
  36. def save_lora_user_metadata(self, name, desc, sd_version, activation_text, preferred_weight, negative_text, notes):
  37. user_metadata = self.get_user_metadata(name)
  38. user_metadata["description"] = desc
  39. user_metadata["sd version"] = sd_version
  40. user_metadata["activation text"] = activation_text
  41. user_metadata["preferred weight"] = preferred_weight
  42. user_metadata["negative text"] = negative_text
  43. user_metadata["notes"] = notes
  44. self.write_user_metadata(name, user_metadata)
  45. def get_metadata_table(self, name):
  46. table = super().get_metadata_table(name)
  47. item = self.page.items.get(name, {})
  48. metadata = item.get("metadata") or {}
  49. keys = {
  50. 'ss_output_name': "Output name:",
  51. 'ss_sd_model_name': "Model:",
  52. 'ss_clip_skip': "Clip skip:",
  53. 'ss_network_module': "Kohya module:",
  54. }
  55. for key, label in keys.items():
  56. value = metadata.get(key, None)
  57. if value is not None and str(value) != "None":
  58. table.append((label, html.escape(value)))
  59. ss_training_started_at = metadata.get('ss_training_started_at')
  60. if ss_training_started_at:
  61. table.append(("Date trained:", datetime.datetime.utcfromtimestamp(float(ss_training_started_at)).strftime('%Y-%m-%d %H:%M')))
  62. ss_bucket_info = metadata.get("ss_bucket_info")
  63. if ss_bucket_info and "buckets" in ss_bucket_info:
  64. resolutions = {}
  65. for _, bucket in ss_bucket_info["buckets"].items():
  66. resolution = bucket["resolution"]
  67. resolution = f'{resolution[1]}x{resolution[0]}'
  68. resolutions[resolution] = resolutions.get(resolution, 0) + int(bucket["count"])
  69. resolutions_list = sorted(resolutions.keys(), key=resolutions.get, reverse=True)
  70. resolutions_text = html.escape(", ".join(resolutions_list[0:4]))
  71. if len(resolutions) > 4:
  72. resolutions_text += ", ..."
  73. resolutions_text = f"<span title='{html.escape(', '.join(resolutions_list))}'>{resolutions_text}</span>"
  74. table.append(('Resolutions:' if len(resolutions_list) > 1 else 'Resolution:', resolutions_text))
  75. image_count = 0
  76. for _, params in metadata.get("ss_dataset_dirs", {}).items():
  77. image_count += int(params.get("img_count", 0))
  78. if image_count:
  79. table.append(("Dataset size:", image_count))
  80. return table
  81. def put_values_into_components(self, name):
  82. user_metadata = self.get_user_metadata(name)
  83. values = super().put_values_into_components(name)
  84. item = self.page.items.get(name, {})
  85. metadata = item.get("metadata") or {}
  86. tags = build_tags(metadata)
  87. gradio_tags = [(tag, str(count)) for tag, count in tags[0:24]]
  88. return [
  89. *values[0:5],
  90. item.get("sd_version", "Unknown"),
  91. gr.HighlightedText.update(value=gradio_tags, visible=True if tags else False),
  92. user_metadata.get('activation text', ''),
  93. float(user_metadata.get('preferred weight', 0.0)),
  94. user_metadata.get('negative text', ''),
  95. gr.update(visible=True if tags else False),
  96. gr.update(value=self.generate_random_prompt_from_tags(tags), visible=True if tags else False),
  97. ]
  98. def generate_random_prompt(self, name):
  99. item = self.page.items.get(name, {})
  100. metadata = item.get("metadata") or {}
  101. tags = build_tags(metadata)
  102. return self.generate_random_prompt_from_tags(tags)
  103. def generate_random_prompt_from_tags(self, tags):
  104. max_count = None
  105. res = []
  106. for tag, count in tags:
  107. if not max_count:
  108. max_count = count
  109. v = random.random() * max_count
  110. if count > v:
  111. res.append(tag)
  112. return ", ".join(sorted(res))
  113. def create_extra_default_items_in_left_column(self):
  114. # this would be a lot better as gr.Radio but I can't make it work
  115. self.select_sd_version = gr.Dropdown(['SD1', 'SD2', 'SDXL', 'Unknown'], value='Unknown', label='Stable Diffusion version', interactive=True)
  116. def create_editor(self):
  117. self.create_default_editor_elems()
  118. self.taginfo = gr.HighlightedText(label="Training dataset tags")
  119. self.edit_activation_text = gr.Text(label='Activation text', info="Will be added to prompt along with Lora")
  120. self.slider_preferred_weight = gr.Slider(label='Preferred weight', info="Set to 0 to disable", minimum=0.0, maximum=2.0, step=0.01)
  121. self.edit_negative_text = gr.Text(label='Negative prompt', info="Will be added to negative prompts")
  122. with gr.Row() as row_random_prompt:
  123. with gr.Column(scale=8):
  124. random_prompt = gr.Textbox(label='Random prompt', lines=4, max_lines=4, interactive=False)
  125. with gr.Column(scale=1, min_width=120):
  126. generate_random_prompt = gr.Button('Generate', size="lg", scale=1)
  127. self.edit_notes = gr.TextArea(label='Notes', lines=4)
  128. generate_random_prompt.click(fn=self.generate_random_prompt, inputs=[self.edit_name_input], outputs=[random_prompt], show_progress=False)
  129. def select_tag(activation_text, evt: gr.SelectData):
  130. tag = evt.value[0]
  131. words = re.split(re_comma, activation_text)
  132. if tag in words:
  133. words = [x for x in words if x != tag and x.strip()]
  134. return ", ".join(words)
  135. return activation_text + ", " + tag if activation_text else tag
  136. self.taginfo.select(fn=select_tag, inputs=[self.edit_activation_text], outputs=[self.edit_activation_text], show_progress=False)
  137. self.create_default_buttons()
  138. viewed_components = [
  139. self.edit_name,
  140. self.edit_description,
  141. self.html_filedata,
  142. self.html_preview,
  143. self.edit_notes,
  144. self.select_sd_version,
  145. self.taginfo,
  146. self.edit_activation_text,
  147. self.slider_preferred_weight,
  148. self.edit_negative_text,
  149. row_random_prompt,
  150. random_prompt,
  151. ]
  152. self.button_edit\
  153. .click(fn=self.put_values_into_components, inputs=[self.edit_name_input], outputs=viewed_components)\
  154. .then(fn=lambda: gr.update(visible=True), inputs=[], outputs=[self.box])
  155. edited_components = [
  156. self.edit_description,
  157. self.select_sd_version,
  158. self.edit_activation_text,
  159. self.slider_preferred_weight,
  160. self.edit_negative_text,
  161. self.edit_notes,
  162. ]
  163. self.setup_save_handler(self.button_save, self.save_lora_user_metadata, edited_components)