extraNetworks.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. function toggleCss(key, css, enable) {
  2. var style = document.getElementById(key);
  3. if (enable && !style) {
  4. style = document.createElement('style');
  5. style.id = key;
  6. style.type = 'text/css';
  7. document.head.appendChild(style);
  8. }
  9. if (style && !enable) {
  10. document.head.removeChild(style);
  11. }
  12. if (style) {
  13. style.innerHTML == '';
  14. style.appendChild(document.createTextNode(css));
  15. }
  16. }
  17. function setupExtraNetworksForTab(tabname) {
  18. gradioApp().querySelector('#' + tabname + '_extra_tabs').classList.add('extra-networks');
  19. var tabs = gradioApp().querySelector('#' + tabname + '_extra_tabs > div');
  20. var searchDiv = gradioApp().getElementById(tabname + '_extra_search');
  21. var search = searchDiv.querySelector('textarea');
  22. var sort = gradioApp().getElementById(tabname + '_extra_sort');
  23. var sortOrder = gradioApp().getElementById(tabname + '_extra_sortorder');
  24. var refresh = gradioApp().getElementById(tabname + '_extra_refresh');
  25. var showDirsDiv = gradioApp().getElementById(tabname + '_extra_show_dirs');
  26. var showDirs = gradioApp().querySelector('#' + tabname + '_extra_show_dirs input');
  27. sort.dataset.sortkey = 'sortDefault';
  28. tabs.appendChild(searchDiv);
  29. tabs.appendChild(sort);
  30. tabs.appendChild(sortOrder);
  31. tabs.appendChild(refresh);
  32. tabs.appendChild(showDirsDiv);
  33. var applyFilter = function() {
  34. var searchTerm = search.value.toLowerCase();
  35. gradioApp().querySelectorAll('#' + tabname + '_extra_tabs div.card').forEach(function(elem) {
  36. var searchOnly = elem.querySelector('.search_only');
  37. var text = elem.querySelector('.name').textContent.toLowerCase() + " " + elem.querySelector('.search_term').textContent.toLowerCase();
  38. var visible = text.indexOf(searchTerm) != -1;
  39. if (searchOnly && searchTerm.length < 4) {
  40. visible = false;
  41. }
  42. elem.style.display = visible ? "" : "none";
  43. });
  44. };
  45. var applySort = function() {
  46. var reverse = sortOrder.classList.contains("sortReverse");
  47. var sortKey = sort.querySelector("input").value.toLowerCase().replace("sort", "").replaceAll(" ", "_").replace(/_+$/, "").trim();
  48. sortKey = sortKey ? "sort" + sortKey.charAt(0).toUpperCase() + sortKey.slice(1) : "";
  49. var sortKeyStore = sortKey ? sortKey + (reverse ? "Reverse" : "") : "";
  50. if (!sortKey || sortKeyStore == sort.dataset.sortkey) {
  51. return;
  52. }
  53. sort.dataset.sortkey = sortKeyStore;
  54. var cards = gradioApp().querySelectorAll('#' + tabname + '_extra_tabs div.card');
  55. cards.forEach(function(card) {
  56. card.originalParentElement = card.parentElement;
  57. });
  58. var sortedCards = Array.from(cards);
  59. sortedCards.sort(function(cardA, cardB) {
  60. var a = cardA.dataset[sortKey];
  61. var b = cardB.dataset[sortKey];
  62. if (!isNaN(a) && !isNaN(b)) {
  63. return parseInt(a) - parseInt(b);
  64. }
  65. return (a < b ? -1 : (a > b ? 1 : 0));
  66. });
  67. if (reverse) {
  68. sortedCards.reverse();
  69. }
  70. cards.forEach(function(card) {
  71. card.remove();
  72. });
  73. sortedCards.forEach(function(card) {
  74. card.originalParentElement.appendChild(card);
  75. });
  76. };
  77. search.addEventListener("input", applyFilter);
  78. applyFilter();
  79. ["change", "blur", "click"].forEach(function(evt) {
  80. sort.querySelector("input").addEventListener(evt, applySort);
  81. });
  82. sortOrder.addEventListener("click", function() {
  83. sortOrder.classList.toggle("sortReverse");
  84. applySort();
  85. });
  86. extraNetworksApplyFilter[tabname] = applyFilter;
  87. var showDirsUpdate = function() {
  88. var css = '#' + tabname + '_extra_tabs .extra-network-subdirs { display: none; }';
  89. toggleCss(tabname + '_extra_show_dirs_style', css, !showDirs.checked);
  90. localSet('extra-networks-show-dirs', showDirs.checked ? 1 : 0);
  91. };
  92. showDirs.checked = localGet('extra-networks-show-dirs', 1) == 1;
  93. showDirs.addEventListener("change", showDirsUpdate);
  94. showDirsUpdate();
  95. }
  96. function applyExtraNetworkFilter(tabname) {
  97. setTimeout(extraNetworksApplyFilter[tabname], 1);
  98. }
  99. var extraNetworksApplyFilter = {};
  100. var activePromptTextarea = {};
  101. function setupExtraNetworks() {
  102. setupExtraNetworksForTab('txt2img');
  103. setupExtraNetworksForTab('img2img');
  104. function registerPrompt(tabname, id) {
  105. var textarea = gradioApp().querySelector("#" + id + " > label > textarea");
  106. if (!activePromptTextarea[tabname]) {
  107. activePromptTextarea[tabname] = textarea;
  108. }
  109. textarea.addEventListener("focus", function() {
  110. activePromptTextarea[tabname] = textarea;
  111. });
  112. }
  113. registerPrompt('txt2img', 'txt2img_prompt');
  114. registerPrompt('txt2img', 'txt2img_neg_prompt');
  115. registerPrompt('img2img', 'img2img_prompt');
  116. registerPrompt('img2img', 'img2img_neg_prompt');
  117. }
  118. onUiLoaded(setupExtraNetworks);
  119. var re_extranet = /<([^:]+:[^:]+):[\d.]+>(.*)/;
  120. var re_extranet_g = /\s+<([^:]+:[^:]+):[\d.]+>/g;
  121. function tryToRemoveExtraNetworkFromPrompt(textarea, text) {
  122. var m = text.match(re_extranet);
  123. var replaced = false;
  124. var newTextareaText;
  125. if (m) {
  126. var extraTextAfterNet = m[2];
  127. var partToSearch = m[1];
  128. var foundAtPosition = -1;
  129. newTextareaText = textarea.value.replaceAll(re_extranet_g, function(found, net, pos) {
  130. m = found.match(re_extranet);
  131. if (m[1] == partToSearch) {
  132. replaced = true;
  133. foundAtPosition = pos;
  134. return "";
  135. }
  136. return found;
  137. });
  138. if (foundAtPosition >= 0 && newTextareaText.substr(foundAtPosition, extraTextAfterNet.length) == extraTextAfterNet) {
  139. newTextareaText = newTextareaText.substr(0, foundAtPosition) + newTextareaText.substr(foundAtPosition + extraTextAfterNet.length);
  140. }
  141. } else {
  142. newTextareaText = textarea.value.replaceAll(new RegExp(text, "g"), function(found) {
  143. if (found == text) {
  144. replaced = true;
  145. return "";
  146. }
  147. return found;
  148. });
  149. }
  150. if (replaced) {
  151. textarea.value = newTextareaText;
  152. return true;
  153. }
  154. return false;
  155. }
  156. function cardClicked(tabname, textToAdd, allowNegativePrompt) {
  157. var textarea = allowNegativePrompt ? activePromptTextarea[tabname] : gradioApp().querySelector("#" + tabname + "_prompt > label > textarea");
  158. if (!tryToRemoveExtraNetworkFromPrompt(textarea, textToAdd)) {
  159. textarea.value = textarea.value + opts.extra_networks_add_text_separator + textToAdd;
  160. }
  161. updateInput(textarea);
  162. }
  163. function saveCardPreview(event, tabname, filename) {
  164. var textarea = gradioApp().querySelector("#" + tabname + '_preview_filename > label > textarea');
  165. var button = gradioApp().getElementById(tabname + '_save_preview');
  166. textarea.value = filename;
  167. updateInput(textarea);
  168. button.click();
  169. event.stopPropagation();
  170. event.preventDefault();
  171. }
  172. function extraNetworksSearchButton(tabs_id, event) {
  173. var searchTextarea = gradioApp().querySelector("#" + tabs_id + ' > label > textarea');
  174. var button = event.target;
  175. var text = button.classList.contains("search-all") ? "" : button.textContent.trim();
  176. searchTextarea.value = text;
  177. updateInput(searchTextarea);
  178. }
  179. var globalPopup = null;
  180. var globalPopupInner = null;
  181. function closePopup() {
  182. if (!globalPopup) return;
  183. globalPopup.style.display = "none";
  184. }
  185. function popup(contents) {
  186. if (!globalPopup) {
  187. globalPopup = document.createElement('div');
  188. globalPopup.onclick = closePopup;
  189. globalPopup.classList.add('global-popup');
  190. var close = document.createElement('div');
  191. close.classList.add('global-popup-close');
  192. close.onclick = closePopup;
  193. close.title = "Close";
  194. globalPopup.appendChild(close);
  195. globalPopupInner = document.createElement('div');
  196. globalPopupInner.onclick = function(event) {
  197. event.stopPropagation(); return false;
  198. };
  199. globalPopupInner.classList.add('global-popup-inner');
  200. globalPopup.appendChild(globalPopupInner);
  201. gradioApp().querySelector('.main').appendChild(globalPopup);
  202. }
  203. globalPopupInner.innerHTML = '';
  204. globalPopupInner.appendChild(contents);
  205. globalPopup.style.display = "flex";
  206. }
  207. function extraNetworksShowMetadata(text) {
  208. var elem = document.createElement('pre');
  209. elem.classList.add('popup-metadata');
  210. elem.textContent = text;
  211. popup(elem);
  212. }
  213. function requestGet(url, data, handler, errorHandler) {
  214. var xhr = new XMLHttpRequest();
  215. var args = Object.keys(data).map(function(k) {
  216. return encodeURIComponent(k) + '=' + encodeURIComponent(data[k]);
  217. }).join('&');
  218. xhr.open("GET", url + "?" + args, true);
  219. xhr.onreadystatechange = function() {
  220. if (xhr.readyState === 4) {
  221. if (xhr.status === 200) {
  222. try {
  223. var js = JSON.parse(xhr.responseText);
  224. handler(js);
  225. } catch (error) {
  226. console.error(error);
  227. errorHandler();
  228. }
  229. } else {
  230. errorHandler();
  231. }
  232. }
  233. };
  234. var js = JSON.stringify(data);
  235. xhr.send(js);
  236. }
  237. function extraNetworksRequestMetadata(event, extraPage, cardName) {
  238. var showError = function() {
  239. extraNetworksShowMetadata("there was an error getting metadata");
  240. };
  241. requestGet("./sd_extra_networks/metadata", {page: extraPage, item: cardName}, function(data) {
  242. if (data && data.metadata) {
  243. extraNetworksShowMetadata(data.metadata);
  244. } else {
  245. showError();
  246. }
  247. }, showError);
  248. event.stopPropagation();
  249. }
  250. var extraPageUserMetadataEditors = {};
  251. function extraNetworksEditUserMetadata(event, tabname, extraPage, cardName) {
  252. var id = tabname + '_' + extraPage + '_edit_user_metadata';
  253. var editor = extraPageUserMetadataEditors[id];
  254. if (!editor) {
  255. editor = {};
  256. editor.page = gradioApp().getElementById(id);
  257. editor.nameTextarea = gradioApp().querySelector("#" + id + "_name" + ' textarea');
  258. editor.button = gradioApp().querySelector("#" + id + "_button");
  259. extraPageUserMetadataEditors[id] = editor;
  260. }
  261. editor.nameTextarea.value = cardName;
  262. updateInput(editor.nameTextarea);
  263. editor.button.click();
  264. popup(editor.page);
  265. event.stopPropagation();
  266. }
  267. function extraNetworksRefreshSingleCard(page, tabname, name) {
  268. requestGet("./sd_extra_networks/get-single-card", {page: page, tabname: tabname, name: name}, function(data) {
  269. if (data && data.html) {
  270. var card = gradioApp().querySelector('.card[data-name=' + JSON.stringify(name) + ']'); // likely using the wrong stringify function
  271. var newDiv = document.createElement('DIV');
  272. newDiv.innerHTML = data.html;
  273. var newCard = newDiv.firstElementChild;
  274. newCard.style = '';
  275. card.parentElement.insertBefore(newCard, card);
  276. card.parentElement.removeChild(card);
  277. }
  278. });
  279. }