- // ==UserScript==
- // @name GhostCanvas
- // @namespace Definable
- // @version 0.1.2
- // @description An SubModule for the Definable ModMenu for Drawaria.Online.
- // @homepage https://drawaria.online/profile/?uid=63196790-c7da-11ec-8266-c399f90709b7
- // @author ≺ᴄᴜʙᴇ³≻
- // @match https://drawaria.online/
- // @match https://drawaria.online/test
- // @match https://drawaria.online/room/*
- // @icon 
- // @grant none
- // @license GNU GPLv3
- // ==/UserScript==
-
- //#region TypeDefinitions
- /**
- * @typedef {Object} PlayerInstance
- * @property {string} name - The name of the player.
- * @property {string} uid - The unique identifier for the player.
- * @property {string} wt - The weight of the player.
- * @property {string} roomID - The room ID the player is in.
- * @property {WebSocket} socket - The WebSocket connection for the player.
- * @property {Map<string, Function[]>} events - The events map for the player.
- * @property {boolean} isConnected - Whether the player is connected.
- * @property {(invitelink:string)=>void} connect - Connects the player to a room.
- * @property {()=>void} disconnect - Disconnects the player.
- * @property {()=>void} reconnect - Reconnects the player.
- * @property {(invitelink:string)=>void} enterRoom - Enters a room.
- * @property {()=>void} nextRoom - Moves the player to the next room.
- * @property {()=>void} leaveRoom - Leaves the current room.
- * @property {(payload:string)=>void} send - Sends a message through the WebSocket.
- * @property {(event:string,handler:Function)=>void} addEventListener - Adds an event listener.
- * @property {(event:string)=>boolean} hasEventListener - Checks if an event listener exists.
- * @property {()=>void} __invokeEvent - Invokes an event.
- */
-
- /**
- * @typedef {Object} DrawariaOnlineMessageTypes
- * @property {(message: string) => string} chatmsg - Sends a chat message.
- * @property {() => string} passturn - Passes the turn.
- * @property {(playerid: number|string) => string} pgdrawvote - Votes for a player to draw.
- * @property {() => string} pgswtichroom - Switches the room.
- * @property {() => string} playerafk - Marks the player as AFK.
- * @property {() => string} playerrated - Rates the player.
- * @property {(gestureid: number|string) => string} sendgesture - Sends a gesture.
- * @property {() => string} sendvote - Sends a vote.
- * @property {(playerid: number|string) => string} sendvotekick - Votes to kick a player.
- * @property {(wordid: number|string) => string} wordselected - Selects a word.
- * @property {Object} clientcmd - Client commands.
- * @property {(itemid: number|string, isactive: boolean) => string} clientcmd.activateitem - Activates an item.
- * @property {(itemid: number|string) => string} clientcmd.buyitem - Buys an item.
- * @property {(itemid: number|string, target: "zindex"|"shared", value: any) => string} clientcmd.canvasobj_changeattr - Changes an attribute of a canvas object.
- * @property {() => string} clientcmd.canvasobj_getobjects - Gets canvas objects.
- * @property {(itemid: number|string) => string} clientcmd.canvasobj_remove - Removes a canvas object.
- * @property {(itemid: number|string, positionX: number|string, positionY: number|string, speed: number|string) => string} clientcmd.canvasobj_setposition - Sets the position of a canvas object.
- * @property {(itemid: number|string, rotation: number|string) => string} clientcmd.canvasobj_setrotation - Sets the rotation of a canvas object.
- * @property {(value: any) => string} clientcmd.customvoting_setvote - Sets a custom vote.
- * @property {(value: any) => string} clientcmd.getfpid - Gets the FPID.
- * @property {() => string} clientcmd.getinventory - Gets the inventory.
- * @property {() => string} clientcmd.getspawnsstate - Gets the spawn state.
- * @property {(positionX: number|string, positionY: number|string) => string} clientcmd.moveavatar - Moves the avatar.
- * @property {() => string} clientcmd.setavatarprop - Sets the avatar properties.
- * @property {(flagid: number|string, isactive: boolean) => string} clientcmd.setstatusflag - Sets a status flag.
- * @property {(playerid: number|string, tokenid: number|string) => string} clientcmd.settoken - Sets a token.
- * @property {(playerid: number|string, value: any) => string} clientcmd.snapchatmessage - Sends a Snapchat message.
- * @property {() => string} clientcmd.spawnavatar - Spawns an avatar.
- * @property {() => string} clientcmd.startrollbackvoting - Starts rollback voting.
- * @property {() => string} clientcmd.trackforwardvoting - Tracks forward voting.
- * @property {(trackid: number|string) => string} clientcmd.votetrack - Votes for a track.
- * @property {(roomID: string, name?: string, uid?: string, wt?: string) => string} startplay - Starts the play.
- * @property {Object} clientnotify - Client notifications.
- * @property {(playerid: number|string) => string} clientnotify.requestcanvas - Requests a canvas.
- * @property {(playerid: number|string, base64: string) => string} clientnotify.respondcanvas - Responds with a canvas.
- * @property {(playerid: number|string, imageid: number|string) => string} clientnotify.galleryupload - Uploads to the gallery.
- * @property {(playerid: number|string, type: any) => string} clientnotify.warning - Sends a warning.
- * @property {(playerid: number|string, targetname: string, mute?: boolean) => string} clientnotify.mute - Mutes a player.
- * @property {(playerid: number|string, targetname: string, hide?: boolean) => string} clientnotify.hide - Hides a player.
- * @property {(playerid: number|string, reason: string, targetname: string) => string} clientnotify.report - Reports a player.
- * @property {Object} drawcmd - Drawing commands.
- * @property {(x1: number|string, y1: number|string, x2: number|string, y2: number|string, color: number|string, size?: number|string, ispixel?: boolean, playerid?: number|string) => string} drawcmd.line - Draws a line.
- * @property {(x1: number|string, y1: number|string, x2: number|string, y2: number|string, color: number|string, size: number|string, ispixel?: boolean, playerid?: number|string) => string} drawcmd.erase - Erases a part of the drawing.
- * @property {(x: number|string, y: number|string, color: number|string, tolerance: number|string, r: number|string, g: number|string, b: number|string, a: number|string) => string} drawcmd.flood - Flood fills an area.
- * @property {(playerid: number|string) => string} drawcmd.undo - Undoes the last action.
- * @property {() => string} drawcmd.clear - Clears the drawing.
- */
-
- /**
- * @typedef {Object} PlayerClass
- * @property {PlayerInstance[]} instances
- * @property {PlayerInstance} noConflict
- * @property {(inviteLink:string)=>URL} getSocketServerURL
- * @property {(inviteLink:string)=>string} getRoomID
- * @property {DrawariaOnlineMessageTypes} parseMessage
- */
-
- /**
- * @typedef {Object} UI
- * @property {(selectors: string, parentElement?: ParentNode) => Element|null} querySelect - Returns the first element that is a descendant of node that matches selectors.
- * @property {(selectors: string, parentElement?: ParentNode) => NodeListOf<Element>} querySelectAll - Returns all element descendants of node that match selectors.
- * @property {(tagName: string, properties?: object) => Element} createElement - Creates an element and assigns properties to it.
- * @property {(element: Element, attributes: object) => void} setAttributes - Assigns attributes to an element.
- * @property {(element: Element, styles: object) => void} setStyles - Assigns styles to an element.
- * @property {(name?: string) => HTMLDivElement} createContainer - Creates a container element.
- * @property {() => HTMLDivElement} createRow - Creates a row element.
- * @property {(name: string) => HTMLElement} createIcon - Creates an icon element.
- * @property {(type: string, properties?: object) => HTMLInputElement} createInput - Creates an input element.
- * @property {(input: HTMLInputElement, properties?: object) => HTMLLabelElement} createLabelFor - Creates a label for an input element.
- * @property {(className?: string) => HTMLElement & { show: Function, hide: Function }} createSpinner - Creates a spinner element.
- * @property {(input: HTMLInputElement, addon: HTMLLabelElement|HTMLInputElement|HTMLButtonElement|HTMLElement) => HTMLDivElement} createGroup - Creates an input group element.
- * @property {(inputs: Array<HTMLLabelElement|HTMLInputElement|HTMLButtonElement|HTMLElement>) => HTMLDivElement} createInputGroup - Creates an input group element.
- */
-
- /**
- * @typedef {Object} Other
- * @property {(message: string, styles?: string, application?: string) => void} log - Logs a message with styles.
- * @property {(size?: number) => string} uid - Generates a random UID.
- * @property {(byteArray: number[]) => string} toHexString - Converts a byte array to a hex string.
- * @property {(key: string, value: string) => void} setCookie - Sets a cookie.
- * @property {() => Array<*>&{addEventListener:(event:"delete"|"set",handler:(property:string,value:*)=>void)=>}} makeObservableArray - Creates an observable array.
- * @property {(message: string) => (Array<any> | object)} tryParseJSON - Tries to parse a JSON string.
- */
-
- /**
- * @class
- * @typedef {Object} DefinableCore
- * @property {PlayerClass} Player
- * @property {UI} UI
- * @property {Other} helper
- */
-
- /**
- * @typedef {Object} Definable
- * @property {PlayerClass} Player
- * @property {UI} UI
- * @property {()=>HTMLElement} createRow
- * @property {(submodule:Core)=>void} registerModule
- */
-
- /**
- * @typedef {Object} Position
- * @prop {number} x
- * @prop {number} y
- */
-
- /**
- * @typedef {Object} Color
- * @prop {number} r
- * @prop {number} g
- * @prop {number} b
- * @prop {number} a
- */
-
- /**
- * @typedef {Object} Volume
- * @prop {number} width
- * @prop {number} height
- */
-
- /**
- * @typedef {Position & Color} Pixel
- */
-
- /**
- * @typedef {Pixel & Volume} Area
- */
-
- /**
- * @typedef {Object} PromiseResponse
- * @prop {*} data
- * @prop {Error|string|undefined} error
- */
- //#endregion TypeDefinitions
-
- (function () {
- "use strict";
-
- /**
- * @typedef {Definable & {listOfPixels:object[],instructions:string[]&{addEventListener:(event:"delete"|"set",handler:(property:string,value:*)=>void)=>}}} GhostCanvas
- */
-
- /**
- * @param {Definable} definable
- * @param {DefinableCore} $core
- */
- function initialize(definable, $core) {
- /** @type {GhostCanvas} */
- const ghostcanvas = new $core("GhostCanvas", "ghost");
- definable.registerModule(ghostcanvas);
- ghostcanvas.listOfPixels = [];
- ghostcanvas.instructions = $core.helper.makeObservableArray();
- ghostcanvas.instructions.isRunning = false;
- const ui = $core.UI;
- const player = $core.Player.noConflict;
-
- const originalCanvas = ui.querySelect("#canvas");
- const canvas = ui.createElement("canvas", {
- className: "position-fixed",
- style: "pointer-events: none; opacity: 0.6; box-shadow: rebeccapurple 0px 0px 2px 2px inset;",
- });
- const context = getOptimizedRenderingContext(canvas);
- document.body.appendChild(canvas);
-
- /* Row 1 */ {
- const row = ghostcanvas.createRow();
-
- {
- const input = ui.createInput("checkbox");
- const label = ui.createLabelFor(input, { title: "Toggle Visibility" });
- label.appendChild(ui.createIcon("low-vision"));
- label.className = label.className.replace("secondary", "success");
- input.addEventListener("input", function () {
- canvas.classList[input.checked ? "remove" : "add"]("d-none");
- });
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Realign" });
- label.appendChild(ui.createIcon("arrows-alt"));
- input.addEventListener("click", function () {
- updatePositionAlt(originalCanvas, canvas);
- });
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("file", { accept: "image/*" });
- const label = ui.createLabelFor(input, { title: "Add Image" });
- label.appendChild(ui.createIcon("plus"));
- label.className = label.className.replace("secondary", "info");
- input.addEventListener("input", function () {
- loadFileAsImage(input.files[0]).then((response) => {
- if (response.data) {
- /** @type {HTMLImageElement} */
- const image = response.data;
- image.classList.add("transformable");
- document.body.appendChild(image);
- input.value = null;
- }
- });
- });
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Remove all images", style: "margin-left: auto;" });
- label.appendChild(ui.createIcon("trash"));
- label.className = label.className.replace("secondary", "warning");
- input.addEventListener("click", function () {
- globalThis["transformable"].target = null;
- const transformableImages = ui.querySelectAll(".transformable");
- Array.from(transformableImages).forEach((element) => element.remove());
- });
- row.appendChild(label);
- }
- }
-
- /* Row 2 */ {
- const row = ghostcanvas.createRow();
-
- // {
- // const input = ui.createInput("button");
- // const label = ui.createLabelFor(input, { title: "Save Image Position" });
- // label.appendChild(ui.createIcon("print"));
- // input.addEventListener("click", function () {
- // ui.querySelectAll(".transformable").forEach((transformable) => {
- // const placed = drawTransformedImage(context, transformable);
- // if (!placed) console.debug("%o is out of bounds", transformable);
- // });
- // });
- // row.appendChild(label);
- // }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Save Pixels" });
- label.appendChild(ui.createIcon("print"));
-
- input.addEventListener("click", function () {
- ui.querySelectAll(".transformable").forEach((transformable) => {
- const placed = drawTransformedImage(context, transformable);
- if (!placed) console.debug("%o is out of bounds", transformable);
- });
- getDefaultPixels(context, ghostcanvas);
- });
-
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("text", { title: "Pixels Buffer", readOnly: true, value: 0, id: "pixelbufferDisplay" });
-
- input.classList.add("col", "form-control-sm");
-
- row.appendChild(input);
- }
- }
-
- /* Row 3 */ {
- const row = ghostcanvas.createRow();
-
- {
- const input = ui.createInput("number", { title: "Color Tolerance", id: "comparePixelsColorTolerance", value: 16, min: 4, max: 128 });
- input.classList.add("col", "form-control-sm");
- row.appendChild(input);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Group by Y" });
- const rotatedIcon = ui.createIcon("bars");
- rotatedIcon.style.rotate = "90deg";
- label.appendChild(rotatedIcon);
- input.addEventListener("click", async function () {
- if (ghostcanvas.listOfPixels.length < 1) {
- await getDefaultPixels(context, ghostcanvas);
- }
- const sortedPixels = await sortPixelsByPosition.callAsWorker(ghostcanvas.listOfPixels, true);
- ghostcanvas.listOfPixels = sortedPixels;
- const groupedPixels = await groupPixelsByColor.callAsWorker(sortedPixels, ui.querySelect("#comparePixelsColorTolerance").value, false, areSameColor.toString(), true);
- ghostcanvas.listOfPixels = groupedPixels;
- ui.querySelect("#pixelbufferDisplay").value = groupedPixels.length;
- });
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Group by X" });
- label.appendChild(ui.createIcon("bars"));
- input.addEventListener("click", async function () {
- if (ghostcanvas.listOfPixels.length < 1) {
- await getDefaultPixels(context, ghostcanvas);
- }
- const sortedPixels = await sortPixelsByPosition.callAsWorker(ghostcanvas.listOfPixels, false);
- ghostcanvas.listOfPixels = sortedPixels;
- const groupedPixels = await groupPixelsByColor.callAsWorker(sortedPixels, ui.querySelect("#comparePixelsColorTolerance").value, true, areSameColor.toString(), true);
- ghostcanvas.listOfPixels = groupedPixels;
- ui.querySelect("#pixelbufferDisplay").value = groupedPixels.length;
- });
- row.appendChild(label);
- }
- }
-
- /* Row 4 */ {
- const row = ghostcanvas.createRow();
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Sort by X (Left to Right)" });
- label.appendChild(ui.createIcon("arrow-right"));
- input.addEventListener("click", async function () {
- if (ghostcanvas.listOfPixels.length < 1) {
- await getDefaultPixels(context, ghostcanvas);
- }
- const sortedPixels = await sortPixelsByPosition.callAsWorker(ghostcanvas.listOfPixels, true);
- ghostcanvas.listOfPixels = sortedPixels;
- ui.querySelect("#pixelbufferDisplay").value = sortedPixels.length;
- });
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Sort by Y (Top to Bottom)" });
- label.appendChild(ui.createIcon("arrow-down"));
- input.addEventListener("click", async function () {
- if (ghostcanvas.listOfPixels.length < 1) {
- await getDefaultPixels(context, ghostcanvas);
- }
- const sortedPixels = await sortPixelsByPosition.callAsWorker(ghostcanvas.listOfPixels, false);
- ghostcanvas.listOfPixels = sortedPixels;
- ui.querySelect("#pixelbufferDisplay").value = sortedPixels.length;
- });
- row.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { title: "Sort by Color" });
- label.appendChild(ui.createIcon("sort-alpha-down"));
- input.addEventListener("click", async function () {
- if (ghostcanvas.listOfPixels.length < 1) {
- await getDefaultPixels(context, ghostcanvas);
- }
- const sortedPixels = await sortPixelsByColor.callAsWorker(ghostcanvas.listOfPixels);
- ghostcanvas.listOfPixels = sortedPixels;
- ui.querySelect("#pixelbufferDisplay").value = sortedPixels.length;
- });
- row.appendChild(label);
- }
- }
-
- /* Row 5 */ {
- ghostcanvas.__contaier.appendChild(ui.querySelect("#chatbox_messages").previousElementSibling.cloneNode(false));
- const row = ghostcanvas.createRow();
-
- const parseInstructionsInput = ui.createInput("button");
- const parseInstructionsLabel = ui.createLabelFor(parseInstructionsInput, { title: "Generate Instructions", textContent: "Load" });
- row.appendChild(parseInstructionsLabel);
- parseInstructionsInput.addEventListener("click", function () {
- ghostcanvas.instructions.push(...ghostcanvas.listOfPixels.map((o) => ghostcanvas.Player.parseMessage.drawcmd.line(o.x, o.y, o.x + o.width, o.y + o.height, `rgb(${o.r},${o.g},${o.b})`, 2)));
- savedInstructionsCountInput.value = ghostcanvas.instructions.length;
- });
-
- const savedInstructionsCountInput = ui.createInput("text", { readOnly: true, title: "Total length of Instructions generated", value: 0, id: "savedInstructionsCountInput" });
- savedInstructionsCountInput.classList.add("col", "form-control-sm");
- row.appendChild(savedInstructionsCountInput);
- ghostcanvas.instructions.addEventListener("delete", (prop, val) => {
- savedInstructionsCountInput.value = ghostcanvas.instructions.length;
- });
- }
-
- /* Row 6 & 7 */ {
- const row6 = ghostcanvas.createRow();
- const row7 = ghostcanvas.createRow();
-
- const bulkSizeInput = ui.createInput("number", { title: "Bulk Execution Size", min: 1, value: 100 });
- bulkSizeInput.classList.add("col", "form-control-sm");
- row6.appendChild(bulkSizeInput);
-
- const intervalInput = ui.createInput("number", { title: "Wait interval between Executions (Milliseconds)", min: 1, value: 1000 });
- intervalInput.classList.add("col", "form-control-sm");
- row6.appendChild(intervalInput);
-
- {
- const input = ui.createInput("checkbox", { id: "gcIsRunning" });
- const label = ui.createLabelFor(input, { textContent: "Start" });
- label.classList.add("col");
- label.className = label.className.replace("secondary", "success");
- input.addEventListener("input", function () {
- ghostcanvas.instructions.isRunning = input.checked;
- execute(ghostcanvas.instructions, player.send, Number(intervalInput.value), Number(bulkSizeInput.value)).finally(() => {
- input.checked = false;
- label.classList.remove("active");
- ui.querySelect("#savedInstructionsCountInput").value = ghostcanvas.instructions.length;
- });
- });
- row7.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { textContent: "Stop" });
- label.classList.add("col");
- label.className = label.className.replace("secondary", "warning");
- input.addEventListener("click", function () {
- ui.querySelect("#gcIsRunning").checked = false;
- ui.querySelect("#gcIsRunning").parentElement.classList.remove("active");
- ghostcanvas.instructions.isRunning = false;
- });
- row7.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { textContent: "Step" });
- label.classList.add("col");
- input.addEventListener("click", function () {
- player.send(ghostcanvas.instructions.shift());
- });
- row7.appendChild(label);
- }
-
- {
- const input = ui.createInput("button");
- const label = ui.createLabelFor(input, { textContent: "Reset" });
- label.classList.add("col");
- label.className = label.className.replace("secondary", "danger");
- input.addEventListener("click", function () {
- ghostcanvas.instructions.length = 0;
- ui.querySelect("#gcIsRunning").checked = false;
- ui.querySelect("#gcIsRunning").parentElement.classList.remove("active");
- ui.querySelect("#savedInstructionsCountInput").value = 0;
- });
- row7.appendChild(label);
- }
- }
-
- updatePositionAlt(originalCanvas, canvas);
- canvas.classList.add("d-none");
- ui.querySelect("#definableStyles").textContent += "\n.transformable { position: fixed; top: 0px; left: 0px; }";
- }
-
- async function getDefaultPixels(context, ghostcanvas) {
- const [imageData, width] = getImageDataForProcessing(context);
- const allPixels = await convertImageDataToPixels.callAsWorker(imageData, width);
- const nonTransparentPixels = await filterForNonTransparentPixels.callAsWorker(allPixels, 128);
-
- ghostcanvas.listOfPixels = nonTransparentPixels;
- document.querySelector("#pixelbufferDisplay").value = nonTransparentPixels.length;
- return true;
- }
-
- /**
- * @param {HTMLCanvasElement} canvas
- * @returns {CanvasRenderingContext2D}
- */
- function getOptimizedRenderingContext(canvas) {
- if (canvas.optimizedRenderingContext) return canvas.optimizedRenderingContext;
- const context = canvas.getContext("2d", {
- alpha: true,
- willReadFrequently: true,
- });
- canvas.optimizedRenderingContext = context;
- return context;
- }
-
- /**
- * @param {File} file
- * @returns {Promise<PromiseResponse>}
- */
- function loadFileAsImage(file) {
- return new Promise((resolve, reject) => {
- if (!(FileReader && file)) {
- reject({ data: undefined, error: "Native FileReader not present." });
- } else {
- const reader = new FileReader();
- reader.onload = function () {
- const image = new Image();
- image.src = reader.result;
- image.onload = function () {
- resolve({ data: image, error: undefined });
- };
- };
- reader.readAsDataURL(file);
- }
- });
- }
-
- /**
- * @param {CanvasRenderingContext2D} context
- * @returns {[Uint8ClampedArray, number]}
- */
- function getImageDataForProcessing(context) {
- return [context.getImageData(0, 0, context.canvas.width, context.canvas.height).data, context.canvas.width];
- }
-
- /**
- * Get Pixels
- * @param {Uint8ClampedArray} imageData
- * @param {number} width
- * @returns {Array<Pixel>}
- */
- function convertImageDataToPixels(imageData, width) {
- const pixelCount = imageData.length / 4;
- const pixels = new Array(pixelCount);
- const widthFactor = 1 / width;
-
- for (let i = 0; i < pixelCount; i++) {
- const index = i * 4;
- const x = i % width;
- const y = (i * widthFactor) | 0;
- const [r, g, b, a] = imageData.slice(index, index + 4);
- pixels[i] = { x, y, r, g, b, a };
- }
-
- return pixels;
- }
-
- /**
- * Filter for NonTransparent Pixels
- * @param {Array<Pixel>} pixels
- * @param {number} [threshold=64]
- * @returns {Array<Pixel>}
- */
- function filterForNonTransparentPixels(pixels, threshold = 64) {
- return pixels.filter(({ a: alpha }) => alpha > threshold);
- }
-
- /**
- * Check if the two provided Pixels are close to similar in the rgb spectrum.
- * @param {Pixel} pixel1 - The first color object with r, g, b, and a properties.
- * @param {Pixel} pixel2 - The second color object with r, g, b, and a properties.
- * @param {number} tolerance - The maximum allowed difference for each color component.
- * @param {boolean} [ignoreAphaValue=true] - Should Alphavalues be ignored.
- * @returns {boolean} - True if the colors are considered the same, false otherwise.
- */
- function areSameColor(pixel1, pixel2, tolerance, ignoreAphaValue = true) {
- const dr = Math.abs(pixel1.r - pixel2.r);
- const dg = Math.abs(pixel1.g - pixel2.g);
- const db = Math.abs(pixel1.b - pixel2.b);
- const da_ok = !ignoreAphaValue ? Math.abs(pixel1.a - pixel2.a) <= tolerance : true;
-
- return dr <= tolerance && dg <= tolerance && db <= tolerance && da_ok;
- }
-
- /**
- * Combine Pixels to Areas by proximity and color.
- * @param {Array<Pixel>} data - Array of Pixel objects.
- * @param {number} [tolerance=32] - The maximum allowed difference for each color component.
- * @param {boolean} [groupByY=true] - If true, group by y then x; if false, group by y then x.
- * @param {string} comparisonFunctionAsString - The Function to be used for Color Comparison as a string.
- * @param {boolean} [ignoreAphaValue=true] - Should Alphavalues be ignored.
- * @returns {Array<Area>} - Array of Area objects.
- */
- function groupPixelsByColor(data, tolerance = 32, groupByY = true, comparisonFunctionAsString = undefined, ignoreAphaValue = true) {
- if (data.length === 0) return [];
-
- const comparisonFunction =
- comparisonFunctionAsString && typeof comparisonFunctionAsString === "string"
- ? new Function("return (" + comparisonFunctionAsString + ")(...arguments)")
- : areSameColor ?? new Function("return false;");
-
- const nameOfValueToIncrease = groupByY ? "width" : "height";
- const nameOfValueToBeSimilar = groupByY ? "y" : "x";
- const nameOfValueToNOTBeSimilar = !groupByY ? "y" : "x";
-
- data.sort();
-
- const lines = new Array(data.length);
- let newLineIndex = 0;
- let currentLine = {
- x: data[0].x,
- y: data[0].y,
- width: 0,
- height: 0,
- r: data[0].r,
- g: data[0].g,
- b: data[0].b,
- a: data[0].a,
- };
-
- for (let i = 1; i < data.length; i++) {
- const pixel = data[i];
- const lastPixel = data[i ? i - 1 : 0];
- if (
- pixel[nameOfValueToBeSimilar] === currentLine[nameOfValueToBeSimilar] &&
- comparisonFunction(currentLine, pixel, tolerance, ignoreAphaValue) &&
- Math.abs(lastPixel[nameOfValueToNOTBeSimilar] - pixel[nameOfValueToNOTBeSimilar]) < 2
- ) {
- currentLine[nameOfValueToIncrease]++;
- } else {
- lines[newLineIndex++] = currentLine;
- currentLine = {
- x: pixel.x,
- y: pixel.y,
- width: 1,
- height: 1,
- r: pixel.r,
- g: pixel.g,
- b: pixel.b,
- a: pixel.a,
- };
- }
- }
-
- // Push the last line
- lines[newLineIndex] = currentLine;
-
- return lines.slice(0, newLineIndex + 1);
- }
-
- /**
- * Sort Pixellike Object by their Position.
- * @param {Array<Pixel>} pixels - Array of Pixel objects.
- * @param {boolean} [sortByX=true] - If true, sort by x then y; if false, sort by y then x.
- * @returns {Array<Pixel>} - Sorted array of Pixel objects.
- */
- function sortPixelsByPosition(pixels, sortByX = true) {
- const sortFunction = sortByX
- ? function (a, b) {
- return a.x - b.x || a.y - b.y;
- }
- : function (a, b) {
- return a.y - b.y || a.x - b.x;
- };
- return pixels.sort(sortFunction);
- }
-
- /**
- * Sort Pixellike Object by their Color.
- * @param {Array<Pixel>} pixels - Array of Pixel objects.
- * @returns {Array<Pixel>}
- */
- function sortPixelsByColor(pixels) {
- return pixels.sort(function (a, b) {
- return a.r - b.r || a.g - b.g || a.b - b.b;
- });
- }
-
- /**
- * Get the Color of the Pixel formatted as rgb.
- * @param {Pixel|Area} pixel
- */
- function getPixelColorAsRGB(pixel) {
- return `rgb(${pixel.r},${pixel.g},${pixel.b})`;
- }
- /**
- * Check if two HTMLElements are overlapping over each other.
- * @param {HTMLElement} element1
- * @param {HTMLElement} element2
- */
- function areOverlapping(element1, element2) {
- const bbox1 = element1.getBoundingClientRect();
- const bbox2 = element2.getBoundingClientRect();
- return bbox1.bottom > bbox2.top || bbox1.right > bbox2.left || bbox1.top < bbox2.bottom || bbox1.left < bbox2.right;
- }
-
- /**
- * Get the position of the image relative to the canvas.
- * @param {HTMLCanvasElement} canvas
- * @param {HTMLImageElement} image
- * @returns
- */
- function getImagePositionRelativeToCanvas(canvas, image) {
- const rect = canvas.getBoundingClientRect();
- const style = window.getComputedStyle(image);
- const x = parseFloat(style.left);
- const y = parseFloat(style.top);
- return [x - rect.left, y - rect.top];
- }
-
- /**
- * Draw the image on the canvas with its transformations.
- * @param {CanvasRenderingContext2D} context
- * @param {HTMLImageElement} image
- */
- function drawTransformedImage(context, image) {
- if (!areOverlapping(context.canvas, image)) return false;
-
- // Get the computed style of the image
- const style = window.getComputedStyle(image);
- const transform = style.transform;
- const width = parseFloat(style.width);
- const height = parseFloat(style.height);
-
- // Save the current context state
- context.save();
-
- // Calculate the position of the image on the canvas
- const [x, y] = getImagePositionRelativeToCanvas(context.canvas, image);
-
- // Calculate the scaling factor
- const scaleX = context.canvas.width / context.canvas.clientWidth;
- const scaleY = context.canvas.height / context.canvas.clientHeight;
-
- // Apply the CSS transform to the canvas context
- context.setTransform(1, 0, 0, 1, 0, 0); // Reset transform
- const matrix = new DOMMatrix(transform);
-
- // Translate to the center of the image
- context.translate((x + width / 2) * scaleX, (y + height / 2) * scaleY);
- // Apply the rotation
- context.transform(matrix.a, matrix.b, matrix.c, matrix.d, matrix.e, matrix.f);
- // Translate back
- context.translate((-width / 2) * scaleX, (-height / 2) * scaleY);
-
- // Draw the image on the canvas
- context.drawImage(image, 0, 0, width * scaleX, height * scaleY);
-
- // Restore the context to its original state
- context.restore();
- return true;
- }
-
- /**
- * @param {HTMLElement} parent
- * @param {HTMLElement} child
- */
- function updatePosition(parent, child) {
- const rect = parent.getBoundingClientRect();
- child.style.top = rect.top.toFixed(0) + "px";
- child.style.left = rect.left.toFixed(0) + "px";
- child.style.width = rect.width.toFixed(0) + "px";
- child.style.height = rect.height.toFixed(0) + "px";
- child.width = 1000;
- child.height = 1000;
- }
-
- function updatePositionAlt(parent, child) {
- const rect = parent.getBoundingClientRect();
- child.style.top = rect.top.toFixed(0) + "px";
- child.style.left = rect.left.toFixed(0) + "px";
- child.width = rect.width.toFixed(0);
- child.height = rect.height.toFixed(0);
- }
-
- /**
- * @param {string[]} instructions
- * @param {Function} callback
- * @param {number} [interval=1000]
- * @param {number} [bulkSize=100]
- */
- function execute(instructions, callback, interval = 1000, bulkSize = 100) {
- return new Promise((resolve, reject) => {
- if (!instructions || !callback) {
- reject();
- return;
- }
- if (!instructions.isRunning || !instructions.length) {
- resolve();
- return;
- }
- const intervalID = setInterval(() => {
- instructions.splice(0, bulkSize).forEach((s) => {
- callback(s);
- });
- if (!instructions.length || !instructions.isRunning) {
- clearInterval(intervalID);
- instructions.isRunning = false;
- resolve();
- return;
- }
- }, interval);
- });
- }
-
- window.addEventListener("definable:init", function (event) {
- /** @type {Definable} */
- const main = event.detail.main;
- /** @type {DefinableCore} */
- const core = event.detail.core;
- initialize(main, core);
- });
- })();