// ==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);
});
})();