Path: blob/master/javascript/dragdrop.js
3055 views
// allows drag-dropping files into gradio image elements, and also pasting images from clipboard12function isValidImageList(files) {3return files && files?.length === 1 && ['image/png', 'image/gif', 'image/jpeg'].includes(files[0].type);4}56function dropReplaceImage(imgWrap, files) {7if (!isValidImageList(files)) {8return;9}1011const tmpFile = files[0];1213imgWrap.querySelector('.modify-upload button + button, .touch-none + div button + button')?.click();14const callback = () => {15const fileInput = imgWrap.querySelector('input[type="file"]');16if (fileInput) {17if (files.length === 0) {18files = new DataTransfer();19files.items.add(tmpFile);20fileInput.files = files.files;21} else {22fileInput.files = files;23}24fileInput.dispatchEvent(new Event('change'));25}26};2728if (imgWrap.closest('#pnginfo_image')) {29// special treatment for PNG Info tab, wait for fetch request to finish30const oldFetch = window.fetch;31window.fetch = async(input, options) => {32const response = await oldFetch(input, options);33if ('api/predict/' === input) {34const content = await response.text();35window.fetch = oldFetch;36window.requestAnimationFrame(() => callback());37return new Response(content, {38status: response.status,39statusText: response.statusText,40headers: response.headers41});42}43return response;44};45} else {46window.requestAnimationFrame(() => callback());47}48}4950function eventHasFiles(e) {51if (!e.dataTransfer || !e.dataTransfer.files) return false;52if (e.dataTransfer.files.length > 0) return true;53if (e.dataTransfer.items.length > 0 && e.dataTransfer.items[0].kind == "file") return true;5455return false;56}5758function isURL(url) {59try {60const _ = new URL(url);61return true;62} catch {63return false;64}65}6667function dragDropTargetIsPrompt(target) {68if (target?.placeholder && target?.placeholder.indexOf("Prompt") >= 0) return true;69if (target?.parentNode?.parentNode?.className?.indexOf("prompt") > 0) return true;70return false;71}7273window.document.addEventListener('dragover', e => {74const target = e.composedPath()[0];75if (!eventHasFiles(e)) return;7677var targetImage = target.closest('[data-testid="image"]');78if (!dragDropTargetIsPrompt(target) && !targetImage) return;7980e.stopPropagation();81e.preventDefault();82e.dataTransfer.dropEffect = 'copy';83});8485window.document.addEventListener('drop', async e => {86const target = e.composedPath()[0];87const url = e.dataTransfer.getData('text/uri-list') || e.dataTransfer.getData('text/plain');88if (!eventHasFiles(e) && !isURL(url)) return;8990if (dragDropTargetIsPrompt(target)) {91e.stopPropagation();92e.preventDefault();9394const isImg2img = get_tab_index('tabs') == 1;95let prompt_image_target = isImg2img ? "img2img_prompt_image" : "txt2img_prompt_image";9697const imgParent = gradioApp().getElementById(prompt_image_target);98const files = e.dataTransfer.files;99const fileInput = imgParent.querySelector('input[type="file"]');100if (eventHasFiles(e) && fileInput) {101fileInput.files = files;102fileInput.dispatchEvent(new Event('change'));103} else if (url) {104try {105const request = await fetch(url);106if (!request.ok) {107console.error('Error fetching URL:', url, request.status);108return;109}110const data = new DataTransfer();111data.items.add(new File([await request.blob()], 'image.png'));112fileInput.files = data.files;113fileInput.dispatchEvent(new Event('change'));114} catch (error) {115console.error('Error fetching URL:', url, error);116return;117}118}119}120121var targetImage = target.closest('[data-testid="image"]');122if (targetImage) {123e.stopPropagation();124e.preventDefault();125const files = e.dataTransfer.files;126dropReplaceImage(targetImage, files);127return;128}129});130131window.addEventListener('paste', e => {132const files = e.clipboardData.files;133if (!isValidImageList(files)) {134return;135}136137const visibleImageFields = [...gradioApp().querySelectorAll('[data-testid="image"]')]138.filter(el => uiElementIsVisible(el))139.sort((a, b) => uiElementInSight(b) - uiElementInSight(a));140141142if (!visibleImageFields.length) {143return;144}145146const firstFreeImageField = visibleImageFields147.filter(el => !el.querySelector('img'))?.[0];148149dropReplaceImage(150firstFreeImageField ?151firstFreeImageField :152visibleImageFields[visibleImageFields.length - 1]153, files154);155});156157158