options.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. import json
  2. import sys
  3. from dataclasses import dataclass
  4. import gradio as gr
  5. from modules import errors
  6. from modules.shared_cmd_options import cmd_opts
  7. class OptionInfo:
  8. def __init__(self, default=None, label="", component=None, component_args=None, onchange=None, section=None, refresh=None, comment_before='', comment_after='', infotext=None, restrict_api=False, category_id=None):
  9. self.default = default
  10. self.label = label
  11. self.component = component
  12. self.component_args = component_args
  13. self.onchange = onchange
  14. self.section = section
  15. self.category_id = category_id
  16. self.refresh = refresh
  17. self.do_not_save = False
  18. self.comment_before = comment_before
  19. """HTML text that will be added after label in UI"""
  20. self.comment_after = comment_after
  21. """HTML text that will be added before label in UI"""
  22. self.infotext = infotext
  23. self.restrict_api = restrict_api
  24. """If True, the setting will not be accessible via API"""
  25. def link(self, label, url):
  26. self.comment_before += f"[<a href='{url}' target='_blank'>{label}</a>]"
  27. return self
  28. def js(self, label, js_func):
  29. self.comment_before += f"[<a onclick='{js_func}(); return false'>{label}</a>]"
  30. return self
  31. def info(self, info):
  32. self.comment_after += f"<span class='info'>({info})</span>"
  33. return self
  34. def html(self, html):
  35. self.comment_after += html
  36. return self
  37. def needs_restart(self):
  38. self.comment_after += " <span class='info'>(requires restart)</span>"
  39. return self
  40. def needs_reload_ui(self):
  41. self.comment_after += " <span class='info'>(requires Reload UI)</span>"
  42. return self
  43. class OptionHTML(OptionInfo):
  44. def __init__(self, text):
  45. super().__init__(str(text).strip(), label='', component=lambda **kwargs: gr.HTML(elem_classes="settings-info", **kwargs))
  46. self.do_not_save = True
  47. def options_section(section_identifier, options_dict):
  48. for v in options_dict.values():
  49. if len(section_identifier) == 2:
  50. v.section = section_identifier
  51. elif len(section_identifier) == 3:
  52. v.section = section_identifier[0:2]
  53. v.category_id = section_identifier[2]
  54. return options_dict
  55. options_builtin_fields = {"data_labels", "data", "restricted_opts", "typemap"}
  56. class Options:
  57. typemap = {int: float}
  58. def __init__(self, data_labels: dict[str, OptionInfo], restricted_opts):
  59. self.data_labels = data_labels
  60. self.data = {k: v.default for k, v in self.data_labels.items() if not v.do_not_save}
  61. self.restricted_opts = restricted_opts
  62. def __setattr__(self, key, value):
  63. if key in options_builtin_fields:
  64. return super(Options, self).__setattr__(key, value)
  65. if self.data is not None:
  66. if key in self.data or key in self.data_labels:
  67. assert not cmd_opts.freeze_settings, "changing settings is disabled"
  68. info = self.data_labels.get(key, None)
  69. if info.do_not_save:
  70. return
  71. comp_args = info.component_args if info else None
  72. if isinstance(comp_args, dict) and comp_args.get('visible', True) is False:
  73. raise RuntimeError(f"not possible to set {key} because it is restricted")
  74. if cmd_opts.hide_ui_dir_config and key in self.restricted_opts:
  75. raise RuntimeError(f"not possible to set {key} because it is restricted")
  76. self.data[key] = value
  77. return
  78. return super(Options, self).__setattr__(key, value)
  79. def __getattr__(self, item):
  80. if item in options_builtin_fields:
  81. return super(Options, self).__getattribute__(item)
  82. if self.data is not None:
  83. if item in self.data:
  84. return self.data[item]
  85. if item in self.data_labels:
  86. return self.data_labels[item].default
  87. return super(Options, self).__getattribute__(item)
  88. def set(self, key, value, is_api=False, run_callbacks=True):
  89. """sets an option and calls its onchange callback, returning True if the option changed and False otherwise"""
  90. oldval = self.data.get(key, None)
  91. if oldval == value:
  92. return False
  93. option = self.data_labels[key]
  94. if option.do_not_save:
  95. return False
  96. if is_api and option.restrict_api:
  97. return False
  98. try:
  99. setattr(self, key, value)
  100. except RuntimeError:
  101. return False
  102. if run_callbacks and option.onchange is not None:
  103. try:
  104. option.onchange()
  105. except Exception as e:
  106. errors.display(e, f"changing setting {key} to {value}")
  107. setattr(self, key, oldval)
  108. return False
  109. return True
  110. def get_default(self, key):
  111. """returns the default value for the key"""
  112. data_label = self.data_labels.get(key)
  113. if data_label is None:
  114. return None
  115. return data_label.default
  116. def save(self, filename):
  117. assert not cmd_opts.freeze_settings, "saving settings is disabled"
  118. with open(filename, "w", encoding="utf8") as file:
  119. json.dump(self.data, file, indent=4, ensure_ascii=False)
  120. def same_type(self, x, y):
  121. if x is None or y is None:
  122. return True
  123. type_x = self.typemap.get(type(x), type(x))
  124. type_y = self.typemap.get(type(y), type(y))
  125. return type_x == type_y
  126. def load(self, filename):
  127. with open(filename, "r", encoding="utf8") as file:
  128. self.data = json.load(file)
  129. # 1.6.0 VAE defaults
  130. if self.data.get('sd_vae_as_default') is not None and self.data.get('sd_vae_overrides_per_model_preferences') is None:
  131. self.data['sd_vae_overrides_per_model_preferences'] = not self.data.get('sd_vae_as_default')
  132. # 1.1.1 quicksettings list migration
  133. if self.data.get('quicksettings') is not None and self.data.get('quicksettings_list') is None:
  134. self.data['quicksettings_list'] = [i.strip() for i in self.data.get('quicksettings').split(',')]
  135. # 1.4.0 ui_reorder
  136. if isinstance(self.data.get('ui_reorder'), str) and self.data.get('ui_reorder') and "ui_reorder_list" not in self.data:
  137. self.data['ui_reorder_list'] = [i.strip() for i in self.data.get('ui_reorder').split(',')]
  138. bad_settings = 0
  139. for k, v in self.data.items():
  140. info = self.data_labels.get(k, None)
  141. if info is not None and not self.same_type(info.default, v):
  142. print(f"Warning: bad setting value: {k}: {v} ({type(v).__name__}; expected {type(info.default).__name__})", file=sys.stderr)
  143. bad_settings += 1
  144. if bad_settings > 0:
  145. print(f"The program is likely to not work with bad settings.\nSettings file: {filename}\nEither fix the file, or delete it and restart.", file=sys.stderr)
  146. def onchange(self, key, func, call=True):
  147. item = self.data_labels.get(key)
  148. item.onchange = func
  149. if call:
  150. func()
  151. def dumpjson(self):
  152. d = {k: self.data.get(k, v.default) for k, v in self.data_labels.items()}
  153. d["_comments_before"] = {k: v.comment_before for k, v in self.data_labels.items() if v.comment_before is not None}
  154. d["_comments_after"] = {k: v.comment_after for k, v in self.data_labels.items() if v.comment_after is not None}
  155. item_categories = {}
  156. for item in self.data_labels.values():
  157. category = categories.mapping.get(item.category_id)
  158. category = "Uncategorized" if category is None else category.label
  159. if category not in item_categories:
  160. item_categories[category] = item.section[1]
  161. # _categories is a list of pairs: [section, category]. Each section (a setting page) will get a special heading above it with the category as text.
  162. d["_categories"] = [[v, k] for k, v in item_categories.items()] + [["Defaults", "Other"]]
  163. return json.dumps(d)
  164. def add_option(self, key, info):
  165. self.data_labels[key] = info
  166. if key not in self.data and not info.do_not_save:
  167. self.data[key] = info.default
  168. def reorder(self):
  169. """Reorder settings so that:
  170. - all items related to section always go together
  171. - all sections belonging to a category go together
  172. - sections inside a category are ordered alphabetically
  173. - categories are ordered by creation order
  174. Category is a superset of sections: for category "postprocessing" there could be multiple sections: "face restoration", "upscaling".
  175. This function also changes items' category_id so that all items belonging to a section have the same category_id.
  176. """
  177. category_ids = {}
  178. section_categories = {}
  179. settings_items = self.data_labels.items()
  180. for _, item in settings_items:
  181. if item.section not in section_categories:
  182. section_categories[item.section] = item.category_id
  183. for _, item in settings_items:
  184. item.category_id = section_categories.get(item.section)
  185. for category_id in categories.mapping:
  186. if category_id not in category_ids:
  187. category_ids[category_id] = len(category_ids)
  188. def sort_key(x):
  189. item: OptionInfo = x[1]
  190. category_order = category_ids.get(item.category_id, len(category_ids))
  191. section_order = item.section[1]
  192. return category_order, section_order
  193. self.data_labels = dict(sorted(settings_items, key=sort_key))
  194. def cast_value(self, key, value):
  195. """casts an arbitrary to the same type as this setting's value with key
  196. Example: cast_value("eta_noise_seed_delta", "12") -> returns 12 (an int rather than str)
  197. """
  198. if value is None:
  199. return None
  200. default_value = self.data_labels[key].default
  201. if default_value is None:
  202. default_value = getattr(self, key, None)
  203. if default_value is None:
  204. return None
  205. expected_type = type(default_value)
  206. if expected_type == bool and value == "False":
  207. value = False
  208. else:
  209. value = expected_type(value)
  210. return value
  211. @dataclass
  212. class OptionsCategory:
  213. id: str
  214. label: str
  215. class OptionsCategories:
  216. def __init__(self):
  217. self.mapping = {}
  218. def register_category(self, category_id, label):
  219. if category_id in self.mapping:
  220. return category_id
  221. self.mapping[category_id] = OptionsCategory(category_id, label)
  222. categories = OptionsCategories()