// ==UserScript==
// @name Wplace Overlay Harpy
// @namespace http://tampermonkey.net/
// @version 1.0.0
// @description Overlay for Wplace
// @author hebert richers
// @match https://wplace.live/*
// @icon https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcQbGI9BVUTdE15GujzdovWGVSnLE6kAwDxJWy_GhsLGUDDEUqolAX7j6oP54g0wl8hDLyo&usqp=CAU
// @license MIT
// @grant none
// @run-at document-start
// ==/UserScript==
(async function() {
'use strict';
const OVERLAY_IMAGE_BASE64 = "";
const PIXEL_URL = "https://backend.wplace.live/s0/pixel/752/1115?x=54&y=22";
const OFFSET_X = 0; // Mover X pixels para direita (negativo = esquerda)
const OFFSET_Y = 0; // Mover Y pixels para baixo (negativo = cima)
function extractCoordinatesFromURL(pixelUrl) {
try {
const url = new URL(pixelUrl);
const pathParts = url.pathname.split('/');
const searchParams = new URLSearchParams(url.search);
return {
chunk1: pathParts[3], // ex: 752
chunk2: pathParts[4], // ex: 1115
posX: parseInt(searchParams.get('x')) || 0, // ex: 287
posY: parseInt(searchParams.get('y')) || 0 // ex: 184
};
} catch (e) {
console.error("Erro ao extrair coordenadas da URL:", e);
return { chunk1: "757", chunk2: "1162", posX: 0, posY: 0 }; // valores padrão
}
}
const coordinates = extractCoordinatesFromURL(PIXEL_URL);
const [chunk1, chunk2] = [coordinates.chunk1, coordinates.chunk2];
console.log(`Aplicando overlay no chunk ${chunk1}/${chunk2} na posição (${coordinates.posX}, ${coordinates.posY})`);
const OVERLAY_MODES = ["overlay", "original"];
let overlayMode = OVERLAY_MODES[0];
let darken = false;
const chunksString = `/${chunk1}/${chunk2}.png`
let cachedOverlayImagePromise = null;
const overlayImage = await getOverlayImage();
const overlayCanvas = new OffscreenCanvas(1000, 1000);
const overlayCtx = overlayCanvas.getContext("2d");
const finalX = coordinates.posX + OFFSET_X;
const finalY = coordinates.posY + OFFSET_Y;
overlayCtx.drawImage(overlayImage, finalX, finalY);
const overlayData = overlayCtx.getImageData(0, 0, 1000, 1000);
console.log(`✅ Imagem posicionada em (${finalX}, ${finalY}) dentro do chunk [base: (${coordinates.posX}, ${coordinates.posY}) + offset: (${OFFSET_X}, ${OFFSET_Y})]`);
fetch = new Proxy(fetch, {
apply: async (target, thisArg, argList) => {
const urlString = typeof argList[0] === "object" ? argList[0].url : argList[0];
let url;
try {
url = new URL(urlString);
} catch (e) {
throw new Error("Invalid URL provided to fetch");
}
if (url.hostname === "backend.wplace.live" && url.pathname.startsWith("/files/") && url.pathname.endsWith(chunksString)) {
if (overlayMode !== "original") {
const originalResponse = await target.apply(thisArg, argList);
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);
const resultData = ctx.getImageData(0, 0, width, height);
const d1 = originalData.data;
const d2 = overlayData.data;
const dr = resultData.data;
for (let i = 0; i < d1.length; i += 4) {
const isTransparent =
d2[i] === 0 &&
d2[i + 1] === 0 &&
d2[i + 2] === 0 &&
d2[i + 3] === 0;
const samePixel =
d1[i] === d2[i] &&
d1[i + 1] === d2[i + 1] &&
d1[i + 2] === d2[i + 2] &&
d1[i + 3] === d2[i + 3];
if (samePixel && !isTransparent) {
dr[i] = 0;
dr[i + 1] = 255;
dr[i + 2] = 0;
dr[i + 3] = 255;
} else if (!isTransparent) {
dr[i] = d2[i];
dr[i + 1] = d2[i + 1];
dr[i + 2] = d2[i + 2];
dr[i + 3] = d2[i + 3];
}
}
ctx.putImageData(resultData, 0, 0);
const mergedBlob = await canvas.convertToBlob();
return new Response(mergedBlob, {
headers: { "Content-Type": "image/png" }
});
}
}
return target.apply(thisArg, argList);
}
});
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) {
cachedOverlayImagePromise = loadImage(OVERLAY_IMAGE_BASE64)
}
return cachedOverlayImagePromise;
}
function patchUI() {
if (document.getElementById("overlay-blend-button")) {
return;
}
let blendButton = document.createElement("button");
blendButton.id = "overlay-blend-button";
blendButton.textContent = overlayMode.charAt(0).toUpperCase() + overlayMode.slice(1);
blendButton.style.backgroundColor = "#0e0e0e7f";
blendButton.style.color = "white";
blendButton.style.border = "solid";
blendButton.style.borderColor = "#1d1d1d7f";
blendButton.style.borderRadius = "4px";
blendButton.style.padding = "5px 10px";
blendButton.style.cursor = "pointer";
blendButton.style.backdropFilter = "blur(2px)";
blendButton.addEventListener("click", () => {
overlayMode = OVERLAY_MODES[(OVERLAY_MODES.indexOf(overlayMode) + 1) % OVERLAY_MODES.length];
blendButton.textContent = `${overlayMode.charAt(0).toUpperCase() + overlayMode.slice(1)}`;
});
const buttonContainer = document.querySelector("div.gap-4:nth-child(1) > div:nth-child(2)");
const leftSidebar = document.querySelector("html body div div.disable-pinch-zoom.relative.h-full.overflow-hidden.svelte-6wmtgk div.absolute.right-2.top-2.z-30 div.flex.flex-col.gap-4.items-center");
if (buttonContainer) {
buttonContainer.appendChild(blendButton);
buttonContainer.classList.remove("items-center");
buttonContainer.classList.add("items-end");
}
if (leftSidebar) {
leftSidebar.classList.add("items-end");
leftSidebar.classList.remove("items-center");
}
}
const observer = new MutationObserver(() => {
patchUI();
});
observer.observe(document.querySelector("div.gap-4:nth-child(1)"), {
childList: true,
subtree: true
});
patchUI();
})();