ui.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. // various functions for interaction with ui.py not large enough to warrant putting them in separate files
  2. function set_theme(theme){
  3. var gradioURL = window.location.href
  4. if (!gradioURL.includes('?__theme=')) {
  5. window.location.replace(gradioURL + '?__theme=' + theme);
  6. }
  7. }
  8. function all_gallery_buttons() {
  9. var allGalleryButtons = gradioApp().querySelectorAll('[style="display: block;"].tabitem div[id$=_gallery].gradio-gallery .thumbnails > .thumbnail-item.thumbnail-small');
  10. var visibleGalleryButtons = [];
  11. allGalleryButtons.forEach(function(elem) {
  12. if (elem.parentElement.offsetParent) {
  13. visibleGalleryButtons.push(elem);
  14. }
  15. })
  16. return visibleGalleryButtons;
  17. }
  18. function selected_gallery_button() {
  19. var allCurrentButtons = gradioApp().querySelectorAll('[style="display: block;"].tabitem div[id$=_gallery].gradio-gallery .thumbnail-item.thumbnail-small.selected');
  20. var visibleCurrentButton = null;
  21. allCurrentButtons.forEach(function(elem) {
  22. if (elem.parentElement.offsetParent) {
  23. visibleCurrentButton = elem;
  24. }
  25. })
  26. return visibleCurrentButton;
  27. }
  28. function selected_gallery_index(){
  29. var buttons = all_gallery_buttons();
  30. var button = selected_gallery_button();
  31. var result = -1
  32. buttons.forEach(function(v, i){ if(v==button) { result = i } })
  33. return result
  34. }
  35. function extract_image_from_gallery(gallery){
  36. if (gallery.length == 0){
  37. return [null];
  38. }
  39. if (gallery.length == 1){
  40. return [gallery[0]];
  41. }
  42. var index = selected_gallery_index()
  43. if (index < 0 || index >= gallery.length){
  44. // Use the first image in the gallery as the default
  45. index = 0;
  46. }
  47. return [gallery[index]];
  48. }
  49. function args_to_array(args){
  50. var res = []
  51. for(var i=0;i<args.length;i++){
  52. res.push(args[i])
  53. }
  54. return res
  55. }
  56. function switch_to_txt2img(){
  57. gradioApp().querySelector('#tabs').querySelectorAll('button')[0].click();
  58. return args_to_array(arguments);
  59. }
  60. function switch_to_img2img_tab(no){
  61. gradioApp().querySelector('#tabs').querySelectorAll('button')[1].click();
  62. gradioApp().getElementById('mode_img2img').querySelectorAll('button')[no].click();
  63. }
  64. function switch_to_img2img(){
  65. switch_to_img2img_tab(0);
  66. return args_to_array(arguments);
  67. }
  68. function switch_to_sketch(){
  69. switch_to_img2img_tab(1);
  70. return args_to_array(arguments);
  71. }
  72. function switch_to_inpaint(){
  73. switch_to_img2img_tab(2);
  74. return args_to_array(arguments);
  75. }
  76. function switch_to_inpaint_sketch(){
  77. switch_to_img2img_tab(3);
  78. return args_to_array(arguments);
  79. }
  80. function switch_to_inpaint(){
  81. gradioApp().querySelector('#tabs').querySelectorAll('button')[1].click();
  82. gradioApp().getElementById('mode_img2img').querySelectorAll('button')[2].click();
  83. return args_to_array(arguments);
  84. }
  85. function switch_to_extras(){
  86. gradioApp().querySelector('#tabs').querySelectorAll('button')[2].click();
  87. return args_to_array(arguments);
  88. }
  89. function get_tab_index(tabId){
  90. var res = 0
  91. gradioApp().getElementById(tabId).querySelector('div').querySelectorAll('button').forEach(function(button, i){
  92. if(button.className.indexOf('selected') != -1)
  93. res = i
  94. })
  95. return res
  96. }
  97. function create_tab_index_args(tabId, args){
  98. var res = []
  99. for(var i=0; i<args.length; i++){
  100. res.push(args[i])
  101. }
  102. res[0] = get_tab_index(tabId)
  103. return res
  104. }
  105. function get_img2img_tab_index() {
  106. let res = args_to_array(arguments)
  107. res.splice(-2)
  108. res[0] = get_tab_index('mode_img2img')
  109. return res
  110. }
  111. function create_submit_args(args){
  112. var res = []
  113. for(var i=0;i<args.length;i++){
  114. res.push(args[i])
  115. }
  116. // As it is currently, txt2img and img2img send back the previous output args (txt2img_gallery, generation_info, html_info) whenever you generate a new image.
  117. // This can lead to uploading a huge gallery of previously generated images, which leads to an unnecessary delay between submitting and beginning to generate.
  118. // I don't know why gradio is sending outputs along with inputs, but we can prevent sending the image gallery here, which seems to be an issue for some.
  119. // If gradio at some point stops sending outputs, this may break something
  120. if(Array.isArray(res[res.length - 3])){
  121. res[res.length - 3] = null
  122. }
  123. return res
  124. }
  125. function showSubmitButtons(tabname, show){
  126. gradioApp().getElementById(tabname+'_interrupt').style.display = show ? "none" : "block"
  127. gradioApp().getElementById(tabname+'_skip').style.display = show ? "none" : "block"
  128. }
  129. function showRestoreProgressButton(tabname, show){
  130. var button = gradioApp().getElementById(tabname + "_restore_progress")
  131. if(! button) return
  132. button.style.display = show ? "flex" : "none"
  133. }
  134. function submit(){
  135. rememberGallerySelection('txt2img_gallery')
  136. showSubmitButtons('txt2img', false)
  137. var id = randomId()
  138. localStorage.setItem("txt2img_task_id", id);
  139. requestProgress(id, gradioApp().getElementById('txt2img_gallery_container'), gradioApp().getElementById('txt2img_gallery'), function(){
  140. showSubmitButtons('txt2img', true)
  141. localStorage.removeItem("txt2img_task_id")
  142. showRestoreProgressButton('txt2img', false)
  143. })
  144. var res = create_submit_args(arguments)
  145. res[0] = id
  146. return res
  147. }
  148. function submit_img2img(){
  149. rememberGallerySelection('img2img_gallery')
  150. showSubmitButtons('img2img', false)
  151. var id = randomId()
  152. localStorage.setItem("img2img_task_id", id);
  153. requestProgress(id, gradioApp().getElementById('img2img_gallery_container'), gradioApp().getElementById('img2img_gallery'), function(){
  154. showSubmitButtons('img2img', true)
  155. localStorage.removeItem("img2img_task_id")
  156. showRestoreProgressButton('img2img', false)
  157. })
  158. var res = create_submit_args(arguments)
  159. res[0] = id
  160. res[1] = get_tab_index('mode_img2img')
  161. return res
  162. }
  163. function restoreProgressTxt2img(){
  164. showRestoreProgressButton("txt2img", false)
  165. var id = localStorage.getItem("txt2img_task_id")
  166. id = localStorage.getItem("txt2img_task_id")
  167. if(id) {
  168. requestProgress(id, gradioApp().getElementById('txt2img_gallery_container'), gradioApp().getElementById('txt2img_gallery'), function(){
  169. showSubmitButtons('txt2img', true)
  170. }, null, 0)
  171. }
  172. return id
  173. }
  174. function restoreProgressImg2img(){
  175. showRestoreProgressButton("img2img", false)
  176. var id = localStorage.getItem("img2img_task_id")
  177. if(id) {
  178. requestProgress(id, gradioApp().getElementById('img2img_gallery_container'), gradioApp().getElementById('img2img_gallery'), function(){
  179. showSubmitButtons('img2img', true)
  180. }, null, 0)
  181. }
  182. return id
  183. }
  184. onUiLoaded(function () {
  185. showRestoreProgressButton('txt2img', localStorage.getItem("txt2img_task_id"))
  186. showRestoreProgressButton('img2img', localStorage.getItem("img2img_task_id"))
  187. });
  188. function modelmerger(){
  189. var id = randomId()
  190. requestProgress(id, gradioApp().getElementById('modelmerger_results_panel'), null, function(){})
  191. var res = create_submit_args(arguments)
  192. res[0] = id
  193. return res
  194. }
  195. function ask_for_style_name(_, prompt_text, negative_prompt_text) {
  196. var name_ = prompt('Style name:')
  197. return [name_, prompt_text, negative_prompt_text]
  198. }
  199. function confirm_clear_prompt(prompt, negative_prompt) {
  200. if(confirm("Delete prompt?")) {
  201. prompt = ""
  202. negative_prompt = ""
  203. }
  204. return [prompt, negative_prompt]
  205. }
  206. promptTokecountUpdateFuncs = {}
  207. function recalculatePromptTokens(name){
  208. if(promptTokecountUpdateFuncs[name]){
  209. promptTokecountUpdateFuncs[name]()
  210. }
  211. }
  212. function recalculate_prompts_txt2img(){
  213. recalculatePromptTokens('txt2img_prompt')
  214. recalculatePromptTokens('txt2img_neg_prompt')
  215. return args_to_array(arguments);
  216. }
  217. function recalculate_prompts_img2img(){
  218. recalculatePromptTokens('img2img_prompt')
  219. recalculatePromptTokens('img2img_neg_prompt')
  220. return args_to_array(arguments);
  221. }
  222. var opts = {}
  223. onUiUpdate(function(){
  224. if(Object.keys(opts).length != 0) return;
  225. var json_elem = gradioApp().getElementById('settings_json')
  226. if(json_elem == null) return;
  227. var textarea = json_elem.querySelector('textarea')
  228. var jsdata = textarea.value
  229. opts = JSON.parse(jsdata)
  230. executeCallbacks(optionsChangedCallbacks);
  231. Object.defineProperty(textarea, 'value', {
  232. set: function(newValue) {
  233. var valueProp = Object.getOwnPropertyDescriptor(HTMLTextAreaElement.prototype, 'value');
  234. var oldValue = valueProp.get.call(textarea);
  235. valueProp.set.call(textarea, newValue);
  236. if (oldValue != newValue) {
  237. opts = JSON.parse(textarea.value)
  238. }
  239. executeCallbacks(optionsChangedCallbacks);
  240. },
  241. get: function() {
  242. var valueProp = Object.getOwnPropertyDescriptor(HTMLTextAreaElement.prototype, 'value');
  243. return valueProp.get.call(textarea);
  244. }
  245. });
  246. json_elem.parentElement.style.display="none"
  247. function registerTextarea(id, id_counter, id_button){
  248. var prompt = gradioApp().getElementById(id)
  249. var counter = gradioApp().getElementById(id_counter)
  250. var textarea = gradioApp().querySelector("#" + id + " > label > textarea");
  251. if(counter.parentElement == prompt.parentElement){
  252. return
  253. }
  254. prompt.parentElement.insertBefore(counter, prompt)
  255. prompt.parentElement.style.position = "relative"
  256. promptTokecountUpdateFuncs[id] = function(){ update_token_counter(id_button); }
  257. textarea.addEventListener("input", promptTokecountUpdateFuncs[id]);
  258. }
  259. registerTextarea('txt2img_prompt', 'txt2img_token_counter', 'txt2img_token_button')
  260. registerTextarea('txt2img_neg_prompt', 'txt2img_negative_token_counter', 'txt2img_negative_token_button')
  261. registerTextarea('img2img_prompt', 'img2img_token_counter', 'img2img_token_button')
  262. registerTextarea('img2img_neg_prompt', 'img2img_negative_token_counter', 'img2img_negative_token_button')
  263. var show_all_pages = gradioApp().getElementById('settings_show_all_pages')
  264. var settings_tabs = gradioApp().querySelector('#settings div')
  265. if(show_all_pages && settings_tabs){
  266. settings_tabs.appendChild(show_all_pages)
  267. show_all_pages.onclick = function(){
  268. gradioApp().querySelectorAll('#settings > div').forEach(function(elem){
  269. elem.style.display = "block";
  270. })
  271. }
  272. }
  273. })
  274. onOptionsChanged(function(){
  275. var elem = gradioApp().getElementById('sd_checkpoint_hash')
  276. var sd_checkpoint_hash = opts.sd_checkpoint_hash || ""
  277. var shorthash = sd_checkpoint_hash.substring(0,10)
  278. if(elem && elem.textContent != shorthash){
  279. elem.textContent = shorthash
  280. elem.title = sd_checkpoint_hash
  281. elem.href = "https://google.com/search?q=" + sd_checkpoint_hash
  282. }
  283. })
  284. let txt2img_textarea, img2img_textarea = undefined;
  285. let wait_time = 800
  286. let token_timeouts = {};
  287. function update_txt2img_tokens(...args) {
  288. update_token_counter("txt2img_token_button")
  289. if (args.length == 2)
  290. return args[0]
  291. return args;
  292. }
  293. function update_img2img_tokens(...args) {
  294. update_token_counter("img2img_token_button")
  295. if (args.length == 2)
  296. return args[0]
  297. return args;
  298. }
  299. function update_token_counter(button_id) {
  300. if (token_timeouts[button_id])
  301. clearTimeout(token_timeouts[button_id]);
  302. token_timeouts[button_id] = setTimeout(() => gradioApp().getElementById(button_id)?.click(), wait_time);
  303. }
  304. function restart_reload(){
  305. document.body.innerHTML='<h1 style="font-family:monospace;margin-top:20%;color:lightgray;text-align:center;">Reloading...</h1>';
  306. setTimeout(function(){location.reload()},2000)
  307. return []
  308. }
  309. // Simulate an `input` DOM event for Gradio Textbox component. Needed after you edit its contents in javascript, otherwise your edits
  310. // will only visible on web page and not sent to python.
  311. function updateInput(target){
  312. let e = new Event("input", { bubbles: true })
  313. Object.defineProperty(e, "target", {value: target})
  314. target.dispatchEvent(e);
  315. }
  316. var desiredCheckpointName = null;
  317. function selectCheckpoint(name){
  318. desiredCheckpointName = name;
  319. gradioApp().getElementById('change_checkpoint').click()
  320. }
  321. function currentImg2imgSourceResolution(_, _, scaleBy){
  322. var img = gradioApp().querySelector('#mode_img2img > div[style="display: block;"] img')
  323. return img ? [img.naturalWidth, img.naturalHeight, scaleBy] : [0, 0, scaleBy]
  324. }
  325. function updateImg2imgResizeToTextAfterChangingImage(){
  326. // At the time this is called from gradio, the image has no yet been replaced.
  327. // There may be a better solution, but this is simple and straightforward so I'm going with it.
  328. setTimeout(function() {
  329. gradioApp().getElementById('img2img_update_resize_to').click()
  330. }, 500);
  331. return []
  332. }