// ==UserScript== // @name Wplace Images overlay // @namespace http://tampermonkey.net/ // @version 1.6.0 // @description Overlay for Wplace // @author nopeee // @match https://wplace.live/* // @license MIT // @grant none // @run-at document-start // @homepageURL https://notepad.pw/markdown/zc3u92d1b // @downloadURL none // ==/UserScript== (async function () { 'use strict'; const originalFetch = fetch; // Store the original fetch function // Function to determine if a function is a Proxy function isProxy(fn) { return typeof fn === 'object' && fn !== null && fn.constructor === Proxy; } var OVERLAY_IMAGE_BASE64 = ""; let PIXEL_URL = "https://backend.wplace.live/s0/pixel/1053/734?x=516&y=409"; var coordinates = null var [chunk1, chunk2] = [null, null] var chunksString = null; let waitingForNewOrigin = false; let cachedOverlayImagePromise = null; let cachedResponses = new Map(); let chunkColorsMap = new Map(); let chosenColor = null function createSelectAndOptionsAndCallback(label, forwhat, optionsList, callback, defaultOption = undefined) { const l = document.createElement("label") l.innerText = label l.setAttribute("for", forwhat + "-select") const a = document.createElement("select") a.classList.add("btn") a.name = forwhat a.id = forwhat + "-select" optionsList.forEach((elementName) => { let b = document.createElement("option") b.value = elementName b.innerText = elementName a.appendChild(b) }) a.onchange = (event) => { console.log("value changed to " + event.target.value + " for dropdown " + forwhat, event) callback(event) } if (defaultOption) a.value = optionsList.find((el) => el.toLowerCase() == defaultOption) let enclosingDiv = document.createElement("div") enclosingDiv.appendChild(l) enclosingDiv.appendChild(a) return enclosingDiv } let chunkStrings = [] function extractCoordinatesFromURL(pixelUrl) { cachedResponses.clear() chunkColorsMap.clear() try { const url = new URL(pixelUrl); const pathParts = url.pathname.split('/'); const searchParams = new URLSearchParams(url.search); coordinates = { startChunkX: Number(pathParts[3]), // ex: 752 startChunkY: Number(pathParts[4]), // ex: 1115 posX: parseInt(searchParams.get('x')) || 0, // ex: 287 posY: parseInt(searchParams.get('y')) || 0 // ex: 184 }; } catch (e) { alert("Bad origin pixel url.") console.error("Error extracting coordinates from URL:", e); coordinates = { startChunkX: "757", startChunkY: "1162", posX: 0, posY: 0 }; // default values } console.log(`Applying overlay to chunk ${coordinates.startChunkX}/${coordinates.startChunkY} at position (${coordinates.posX}, ${coordinates.posY})`); getOverlayImage().then((img) => { const [baseChunkX, baseChunkY] = [coordinates.startChunkX, coordinates.startChunkY]; const [baseCoordX, baseCoordY] = [coordinates.posX, coordinates.posY]; const imgWidth = img.width const imgHeight = img.height console.log([baseChunkX, baseChunkY], [baseCoordX, baseCoordY], imgWidth, imgHeight) // There are 2048x2048 chunks (including chunk 0) // There are 1000x1000 pixels per chunk (including pixels inx 0) // Every possible chunk we need is either down, or to the right. // The simplest way is to get the lower right coords and deduce what other chunks we crossed const [endCoordsX, endCoordsY] = [baseCoordX + imgWidth, baseCoordY + imgHeight] coordinates.endChunkX = baseChunkX + Math.floor(endCoordsX / 1000) coordinates.endChunkY = baseChunkY + Math.floor(endCoordsY / 1000) console.log([endCoordsX, endCoordsY], coordinates.endChunkX, coordinates.endChunkY) console.log(`x will range ${coordinates.startChunkX}-${coordinates.endChunkX}`) console.log(`y will range ${coordinates.startChunkY}-${coordinates.endChunkY}`) chunkStrings = [] let x = coordinates.startChunkX while (x <= coordinates.endChunkX) { console.log(`in while x=${x}`) let y = coordinates.startChunkY while (y <= coordinates.endChunkY) { console.log(`in while y=${y}` + `adding /${x}/${y}`) chunkStrings = chunkStrings.concat(`/${x}/${y}`) y++ } x++ } console.log("Current origin affects " + chunkStrings.length + "chunks") }) } extractCoordinatesFromURL(PIXEL_URL) const OVERLAY_MODES = ["overlay", "original", "onlyNonBlank"]; let overlayMode = OVERLAY_MODES[0]; let darken = false; let overlayData = null async function resetAndLoadNewImage() { const overlayImage = await getOverlayImage(); const overlayCanvas = new OffscreenCanvas(1000, 1000); const overlayCtx = overlayCanvas.getContext("2d", { willReadFrequently: true }); // console.log(`✅ Image positioned at (${finalX}, ${finalY}) within chunk [base: (${coordinates.posX}, ${coordinates.posY}) + offset: (${OFFSET_X}, ${OFFSET_Y})]`); fetch = new Proxy(originalFetch, { apply: async (target, thisArg, argList) => { // Check if the original fetch is already proxied const targetFetch = isProxy(target) ? target : originalFetch; const urlString = typeof argList[0] === "object" ? argList[0].url : argList[0]; const options = typeof argList[0] === "object" ? argList[0] : argList[1]; let url; try { url = new URL(urlString); } catch (e) { throw new Error("Invalid URL provided to fetch"); } const matchingChunkString = chunkStrings.find((el) => url.pathname.replace(".png", "").endsWith(el)) if (url.hostname === "backend.wplace.live" && url.pathname.startsWith("/s0/pixel") && matchingChunkString != null) { if (options && options.method && options.method.toUpperCase() === "POST") { console.log("POST request detected 4 chunk " + matchingChunkString + ", clearing cached chunk"); cachedResponses.delete(matchingChunkString) } } else if (url.hostname === "backend.wplace.live" && url.pathname.startsWith("/files/") && matchingChunkString != null) { if (overlayMode !== "hidden") { const currentChunkX = matchingChunkString.split("/")[1] const currentChunkY = matchingChunkString.split("/")[2] console.log("Current chunk is probably " + currentChunkX + " " + currentChunkY) const originalResponse = await target.apply(thisArg, argList); const currentLastModified = originalResponse.headers.get("last-modified") if (overlayMode != "colorselect" && cachedResponses.has(matchingChunkString)) { const [cachedLastModified, savedMergedBlob] = cachedResponses.get(matchingChunkString); if (currentLastModified === cachedLastModified) { console.log("Using cached merged blob"); return new Response(savedMergedBlob, { headers: { "Content-Type": "image/png" } }); } } else console.log("No matching ChunkString") let localColorsMap = new Map() // chunkColorsMap.set(matchingChunkString, new Map()) const originalBlob = await originalResponse.blob(); const originalImage = await blobToImage(originalBlob); const width = originalImage.width; const height = originalImage.height; const canvas = new OffscreenCanvas(width, height); const ctx = canvas.getContext("2d"); ctx.drawImage(originalImage, 0, 0, width, height); const originalData = ctx.getImageData(0, 0, width, height); let [sx, sy, sw, sh, dx, dy, dw, dh] = [0, 0, 1000, 1000, coordinates.posX, coordinates.posY, 1000, 1000] // https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-drawimage if (currentChunkX - coordinates.startChunkX != 0) { sx = (1000 - coordinates.posX + (currentChunkX - coordinates.startChunkX - 1) * 1000) dx = 0 console.log("X chunk is not same as base. altering to " + sx) } if (currentChunkY - coordinates.startChunkY != 0) { sy = (1000 - coordinates.posY + (currentChunkY - coordinates.startChunkY - 1) * 1000) dy = 0 console.log("X chunk is not same as base. altering to " + sx) } console.log(`params are`, [sx, sy, sw, sh, dx, dy, dw, dh]) overlayCtx.clearRect(0, 0, overlayCanvas.width, overlayCanvas.height); overlayCtx.drawImage(overlayImage, sx, sy, sw, sh, dx, dy, dw, dh); overlayData = overlayCtx.getImageData(0, 0, 1000, 1000); const resultData = ctx.getImageData(0, 0, width, height); const serverD = originalData.data; const overlayD = overlayData.data; const resultD = resultData.data; for (let i = 0; i < serverD.length; i += 4) { const isTransparent = overlayD[i] === 0 && overlayD[i + 1] === 0 && overlayD[i + 2] === 0 && overlayD[i + 3] === 0; const samePixel = serverD[i] === overlayD[i] && serverD[i + 1] === overlayD[i + 1] && serverD[i + 2] === overlayD[i + 2] && serverD[i + 3] === overlayD[i + 3]; if (!samePixel && !isTransparent) { const key = overlayD.slice(i, i + 4).join(", ") if (localColorsMap.has(key)) { localColorsMap.set(key, localColorsMap.get(key) + 1) } else localColorsMap.set(key, 1) } if (overlayMode == "onlynonblank") { if (!(serverD[i] === 0 && serverD[i + 1] === 0 && serverD[i + 2] === 0 && serverD[i + 3] === 0)) { resultD[i] = serverD[i]; resultD[i + 1] = serverD[i + 1]; resultD[i + 2] = serverD[i + 2]; resultD[i + 3] = serverD[i + 3]; } else { resultD[i] = overlayD[i]; resultD[i + 1] = overlayD[i + 1]; resultD[i + 2] = overlayD[i + 2]; resultD[i + 3] = overlayD[i + 3]; }// og pixel not transparent } else if (overlayMode == "findwrong") { if (!samePixel) { resultD[i] = 255 resultD[i + 1] = 0 resultD[i + 2] = 0 resultD[i + 3] = 255 } else { resultD[i] = serverD[i]; resultD[i + 1] = serverD[i + 1]; resultD[i + 2] = serverD[i + 2]; resultD[i + 3] = serverD[i + 3]; } } else if (overlayMode == "colorselect") { const curColor = overlayD.slice(i, i + 4) if (chosenColor != null && !samePixel && curColor[0] == chosenColor[0] && curColor[1] == chosenColor[1] && curColor[2] == chosenColor[2] && curColor[3] == chosenColor[3]) { resultD[i] = 255; resultD[i + 1] = 255; resultD[i + 2] = 255; resultD[i + 3] = 255; } else { resultD[i] = 0; resultD[i + 1] = 0; resultD[i + 2] = 0; resultD[i + 3] = 255; } } else { if (samePixel && !isTransparent) { resultD[i] = 0; resultD[i + 1] = 255; resultD[i + 2] = 0; resultD[i + 3] = 255; } else if (!isTransparent) { resultD[i] = overlayD[i]; resultD[i + 1] = overlayD[i + 1]; resultD[i + 2] = overlayD[i + 2]; resultD[i + 3] = overlayD[i + 3]; } } } chunkColorsMap.set(matchingChunkString, localColorsMap) ctx.putImageData(resultData, 0, 0); const mergedBlob = await canvas.convertToBlob(); if (overlayMode != "colorselect") { cachedResponses.set(matchingChunkString, [currentLastModified, mergedBlob]); // Cache the MD5 and mergedBlob console.log(`stored ${matchingChunkString} with date ${currentLastModified}`) } return new Response(mergedBlob, { headers: { "Content-Type": "image/png" } }); } // https://backend.wplace.live/s0/pixel/1036/704?x=469&y=393 } else if (url.hostname === "backend.wplace.live" && url.pathname.startsWith("/s0/pixel") && url.searchParams.has("x") && url.searchParams.has("y")) { if (waitingForNewOrigin) { console.log("Setting new origin to:", url.href); PIXEL_URL = url.href; extractCoordinatesFromURL(PIXEL_URL); resetAndLoadNewImage(); waitingForNewOrigin = false; if (document.getElementById("overlay-origin-button")) { document.getElementById("overlay-origin-button").textContent = "Set Origin..."; } } } return targetFetch.apply(thisArg, argList); } }); } await resetAndLoadNewImage(); function blobToImage(blob) { return new Promise((resolve, reject) => { const img = new Image(); img.onload = () => resolve(img); img.onerror = reject; img.src = URL.createObjectURL(blob); }); } function loadImage(src) { return new Promise((resolve, reject) => { const img = new Image(); img.crossOrigin = "anonymous"; img.onload = () => resolve(img); img.onerror = reject; img.src = src; }); } function getOverlayImage() { if (!cachedOverlayImagePromise) { console.log("Loaded new image") cachedOverlayImagePromise = loadImage(OVERLAY_IMAGE_BASE64) cachedResponses.clear() } return cachedOverlayImagePromise; } function patchColorPick() { const colorSelector = document.querySelector("[id^=color]")?.parentElement?.parentElement if (colorSelector?.parentElement?.previousElementSibling?.children[1] != null) { // const specialColorDiv = document.createElement("div") // specialColorDiv.style.display = "none" console.log("Hooking color selector") const colorInfoText = document.createElement("h2") colorInfoText.innerText = "Hover a color for info" colorInfoText.id = "colorInfoText" colorSelector.parentElement.previousElementSibling.children[1].appendChild(colorInfoText) colorSelector.parentElement.parentElement.onmouseover = (event) => { try { if (event.target?.className?.includes("grid-cols") == true) return } catch (error) { console.log(error,event) } if (event.target.tagName != "BUTTON") { let targetText = "Hover or click a color for info" console.log(event) if (chosenColor) { console.log("Hover, chosencolor" + chosenColor) let key = chosenColor.join(", "); let qty = 0; [...chunkColorsMap.values()].forEach((colorMap) => { if (colorMap.has(key)) qty += colorMap.get(key); } ) let colorName = chosenColor[3] == 0 ? "Transparent" : document.querySelector(`[style='background: rgb(${chosenColor.slice(0, 3).join(", ")});']`).ariaLabel targetText = colorName + ": " + qty + " to place" } document.querySelector("#colorInfoText").innerText = targetText } else { if (!chunkColorsMap) return; const bgc = event.target.style?.background let hoverColor; if (!bgc) { // transparent probably hoverColor = "0, 0, 0, 0" } else hoverColor = bgc.replace(")", "").split("(").slice(1)[0] + ", 255" // "249, 221, 59" +, 255 let qty = 0; [...chunkColorsMap.values()].forEach((colorMap) => { if (colorMap.has(hoverColor)) qty += colorMap.get(hoverColor); } ) if (qty != 0) { document.querySelector("#colorInfoText").innerText = event.target.ariaLabel + ": " + qty + " to place" console.log("Hovered on a mapped color with " + qty + " targets") } else document.querySelector("#colorInfoText").innerText = event.target.ariaLabel + " is unused" } } colorSelector.onclick = (event) => { if (!chunkColorsMap) return; const bgc = event.target.style?.background if (!bgc) { // transparent probably chosenColor = "0, 0, 0, 0".split(", ") } else chosenColor = (bgc.replace(")", "").split("(").slice(1)[0] + ", 255").split(", ") // "249, 221, 59" +, 255 console.log("CLick: set color to " + chosenColor, event) } // const whatColorText = document.createElement("p") // specialColorDiv.appendChild(whatColorText) // whatColorText = "No color selected" colorSelector.parentElement.nextElementSibling.children[1] return } } function createOverlayButton() { if (document.getElementById("overlayDiv")) { return; } const baseButton = document.querySelector("button.btn[title='Leaderboard']") if (!baseButton) { return } const deployButton = baseButton.cloneNode(true) deployButton.title = "Overlay actions" deployButton.children[0].children[0].setAttribute("d", "M0,0 L80,50 L0,100 Z") deployButton.children[0].setAttribute("viewBox", "-50 0 200 100") deployButton.classList.add("deployed") let overlayDiv = document.createElement("div") overlayDiv.id = "overlayDiv" overlayDiv.style.backgroundColor = "white" overlayDiv.style.borderRadius = "20px" overlayDiv.style.padding = "5px" // overlayDiv.style.display = "block" overlayDiv.style.position = "fixed" overlayDiv.style.top = "40px" overlayDiv.style.left = "40px" overlayDiv.style.zIndex = "1" overlayDiv.appendChild(deployButton) let otherButtonsDiv = document.createElement("div") otherButtonsDiv.id = "otherButtonsDiv" otherButtonsDiv.style.flexDirection = "column" overlayDiv.appendChild(otherButtonsDiv) //deployButton.classList.add("deployed") deployButton.onclick = () => { if (deployButton.parentElement.classList.contains("dragged")) { deployButton.parentElement.classList.remove("dragged") return } if (deployButton.classList.contains("deployed")) { deployButton.classList.remove("deployed") otherButtonsDiv.classList.add("hidden") } else { deployButton.classList.add("deployed") otherButtonsDiv.classList.remove("hidden") } } draggable(overlayDiv); document.body.prepend(overlayDiv) let overlayModePart = createSelectAndOptionsAndCallback("Choose overlay mode: ", "overlayMode", ["Overlay", "OnlyNonBlank", "ColorSelect", "FindWrong", "Hidden"], (event) => { overlayMode = event.target.value.toLowerCase(); overlayData = null; cachedResponses.clear() resetAndLoadNewImage(); console.log(event.target.value) }, overlayMode ) otherButtonsDiv.appendChild(overlayModePart) const setOriginButton = baseButton.cloneNode(false) setOriginButton.textContent = "Set Origin..." setOriginButton.classList.remove("btn-square") setOriginButton.id = "overlay-origin-button"; setOriginButton.addEventListener("click", () => { waitingForNewOrigin = true; console.log("Click on new origin..."); setOriginButton.textContent = "Click on new origin... (10s)" setTimeout(() => { setOriginButton.textContent = "Set Origin..." waitingForNewOrigin = false; }, 10000) }); otherButtonsDiv.appendChild(setOriginButton) let parentFileInputDiv = document.createElement("div") parentFileInputDiv.style.display = "flex" parentFileInputDiv.style.alignItems = "center" let fileInputLabel = document.createElement("label") fileInputLabel.setAttribute("for", "fileInput") fileInputLabel.textContent = "Load a file: " let fileInput = document.createElement("input"); fileInput.classList.add("btn") fileInput.id = "fileInput" fileInput.type = "file"; fileInput.accept = "image/*"; fileInput.style.marginTop = "10px"; fileInput.style.display = "block"; parentFileInputDiv.appendChild(fileInputLabel) parentFileInputDiv.appendChild(fileInput) otherButtonsDiv.appendChild(parentFileInputDiv) // Create the file input for selecting an image fileInput.addEventListener("change", async (event) => { const file = event.target.files[0]; if (file) { const base64 = await convertToBase64(file); console.log("Selected image in Base64:", base64); OVERLAY_IMAGE_BASE64 = base64; cachedOverlayImagePromise = null; overlayData = null; chunkColorsMap.clear() resetAndLoadNewImage() } }); let saveToLocalStorageBtn = baseButton.cloneNode(false) saveToLocalStorageBtn.textContent = "Save to localStorage" saveToLocalStorageBtn.title = "Will store name, location, and image data locally for easy reloading" saveToLocalStorageBtn.classList.remove("btn-square") saveToLocalStorageBtn.id = "saveToLocalStorageBtn"; saveToLocalStorageBtn.addEventListener("click", () => { let fileName; if (fileInput.files.length == 1) { fileName = fileInput.files[0].name } else fileName = prompt("Name currently loaded image") if (fileName == "") return const fileData = OVERLAY_IMAGE_BASE64 const originLocation = PIXEL_URL let prevData = localStorage.getItem("overlayData") if (prevData == null) { prevData = [] } else prevData = JSON.parse(prevData) if (prevData.findIndex((el) => el[0] == fileName) != -1) { console.log("Updated entry " + fileName) prevData.splice(prevData.findIndex((el) => el[0] == fileName), 1) } prevData.push([fileName, originLocation, fileData]) localStorage.setItem("overlayData", JSON.stringify(prevData)) updateLocalStorageEntries() }); otherButtonsDiv.appendChild(saveToLocalStorageBtn) let loadOrDeleteEntryContainer = document.createElement("div") loadOrDeleteEntryContainer.style.display = "flex" loadOrDeleteEntryContainer.style.alignItems = "center" let curData = localStorage.getItem("overlayData") if (curData == null) { curData = [] } else curData = JSON.parse(curData) let currentlySelectedlocalStorageElement = "" let localStorageEntrySelector = createSelectAndOptionsAndCallback("", "localStorageEntry", [""].concat(curData.map((el) => el[0])), (event) => { currentlySelectedlocalStorageElement = event.target.value } ) let loadSelectedEntry = baseButton.cloneNode(false) loadSelectedEntry.textContent = "Load that entry" loadSelectedEntry.title = "Will load name, location, and image data from localStorage" loadSelectedEntry.classList.remove("btn-square") loadSelectedEntry.id = "loadFromLocalStorageBtn"; loadSelectedEntry.addEventListener("click", () => { if (currentlySelectedlocalStorageElement != "") { let prevData = localStorage.getItem("overlayData") if (prevData == null) { prevData = [] } else prevData = JSON.parse(prevData) const targetIndex = prevData.findIndex((el) => el[0] == currentlySelectedlocalStorageElement) if (targetIndex != -1) { OVERLAY_IMAGE_BASE64 = prevData[targetIndex][2] PIXEL_URL = prevData[targetIndex][1] extractCoordinatesFromURL(PIXEL_URL); console.log("Done !") cachedOverlayImagePromise = null; overlayData = null; resetAndLoadNewImage() } } }); let deleteSelectedEntry = baseButton.cloneNode(false) deleteSelectedEntry.textContent = "Delete that entry" deleteSelectedEntry.title = "Will remove selected entry from localStorage" deleteSelectedEntry.classList.remove("btn-square") deleteSelectedEntry.id = "removeFromLocalStorageBtn"; deleteSelectedEntry.addEventListener("click", () => { if (currentlySelectedlocalStorageElement != "") { let prevData = localStorage.getItem("overlayData") if (prevData == null) { prevData = [] } else prevData = JSON.parse(prevData) if (prevData.findIndex((el) => el[0] == currentlySelectedlocalStorageElement) != -1) { console.log("Removed entry " + currentlySelectedlocalStorageElement) prevData.splice(prevData.findIndex((el) => el[0] == currentlySelectedlocalStorageElement), 1) } else alert("Could not delete this as does not exist in localSotage: " + currentlySelectedlocalStorageElement) localStorage.setItem("overlayData", JSON.stringify(prevData)) updateLocalStorageEntries() } }); loadOrDeleteEntryContainer.appendChild(loadSelectedEntry) loadOrDeleteEntryContainer.appendChild(deleteSelectedEntry) loadOrDeleteEntryContainer.appendChild(localStorageEntrySelector) otherButtonsDiv.appendChild(loadOrDeleteEntryContainer) } function updateLocalStorageEntries() { // Get the existing selector element const localStorageEntrySelector = document.getElementById("localStorageEntry-select"); // Retrieve current localStorage data let curData = localStorage.getItem("overlayData") if (curData == null) { curData = [] } else { curData = JSON.parse(curData) } // Clear existing options localStorageEntrySelector.innerHTML = ''; // Add default empty option let defaultOption = document.createElement("option") defaultOption.value = "" defaultOption.innerText = "" localStorageEntrySelector.appendChild(defaultOption) // Populate with new entries curData.forEach((el) => { let option = document.createElement("option") option.value = el[0] option.innerText = el[0] localStorageEntrySelector.appendChild(option) }) } // Function to convert a file to Base64 function convertToBase64(file) { return new Promise((resolve, reject) => { const reader = new FileReader(); reader.onloadend = () => { resolve(reader.result); // This will be the Base64 string }; reader.onerror = reject; reader.readAsDataURL(file); // Read the file as a data URL }); } const observer = new MutationObserver(() => { patchColorPick(); createOverlayButton() }); observer.observe(document.querySelector("div.gap-4:nth-child(1)"), { childList: true, subtree: true }); function addGlobalStyle(css) { var head, style; head = document.getElementsByTagName('head')[0]; if (!head) { return; } style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = css; head.appendChild(style); } addGlobalStyle('button.deployed > svg > path{ transform: rotate(90deg); transform-box: fill-box; transform-origin: center;}'); addGlobalStyle("#otherButtonsDiv.hidden { display: none;}") addGlobalStyle("#otherButtonsDiv { display: flex;}") patchColorPick(); createOverlayButton(); function draggable(el) { el.addEventListener('mousedown', function (e) { var offsetX = e.clientX - parseInt(window.getComputedStyle(this).left); var offsetY = e.clientY - parseInt(window.getComputedStyle(this).top); var startX = e.clientX; var startY = e.clientY; function mouseMoveHandler(e) { el.style.top = (e.clientY - offsetY) + 'px'; el.style.left = (e.clientX - offsetX) + 'px'; if (el.classList.contains('dragged')) return var distanceX = Math.abs(e.clientX - startX); var distanceY = Math.abs(e.clientY - startY); if (distanceX > 5 || distanceY > 5) { el.classList.add('dragged'); // Add the dragged class } } function reset() { window.removeEventListener('mousemove', mouseMoveHandler); window.removeEventListener('mouseup', reset); } window.addEventListener('mousemove', mouseMoveHandler); window.addEventListener('mouseup', reset); }); } })();