// ==UserScript==
// @name Neko's Scripts
// @namespace http://tampermonkey.net/
// @version 0.11.0
// @description Script for OWOP
// @author Neko
// @match https://ourworldofpixels.com/*
// @exclude https://ourworldofpixels.com/api/*
// @run-at document-start
// @icon https://www.google.com/s2/favicons?sz=64&domain=ourworldofpixels.com
// @grant none
// ==/UserScript==
'use strict';
/*global OWOP*/
{
let k = EventTarget.prototype.addEventListener;
EventTarget._eventlists = [];
EventTarget.prototype.addEventListener = function (r, i, e) {
if (EventTarget._eventlists) EventTarget._eventlists.push(i);
return k.bind(this)(...arguments)
};
let l = EventTarget.prototype.removeEventListener;
EventTarget.prototype.removeEventListener = function () {
return l.bind(this)(...arguments);
};
}
const NS = {};
function install() {
"use strict";
class Point {
constructor(x, y) {
this.x = x;
this.y = y;
}
static distance(p1, p2) {
if (p1 instanceof Point && p2 instanceof Point) return Math.sqrt((p1.x - p2.x) ** 2 + (p1.y - p2.y) ** 2);
}
}
class Color {
static compare(c1, c2) {
return (c1[0] == c2[0] && c1[1] == c2[1] && c1[2] == c2[2]);
}
}
class Pixel extends Point {
constructor(x, y, c, o = false) {
super(x, y);
this.c = c;
this.o = o;
}
static compare(p1, p2) {
return (p1.x == p2.x && p1.y == p2.y) && Color.compare(p1.c, p2.c);
}
}
class Action {
constructor(p1, p2) {
this.x = p1.x;
this.y = p1.y;
this.before_color = p1.c;
this.after_color = p2.c;
}
undo() {
return this.before_color;
}
redo() {
return this.after_color;
}
}
class PixelManager {
constructor() {
this.undoStack = [];
this.redoStack = [];
this.actionStack = {};
this.record = false;
this.queue = {};
this.on = true;
this.extra = {};
this.extra.placeData = [];
this.disabled = false;
let p1 = new Point(0, 0);
for (let y = -47; y < 47; y++) {
for (let x = -47; x < 47; x++) {
let p2 = new Point(x, y);
let d = Point.distance(p1, p2);
// d = Math.random();
this.extra.placeData.push([d, p2]);
}
}
this.extra.placeData.sort((a, b) => {
return a[0] - b[0];
});
setInterval(() => {
if (!this.disabled) this.placePixel();
}, 20);
}
undo() {
if (!this.undoStack.length) return;
let action = this.undoStack.pop();
for (let e in action) {
let e2 = action[e];
if (!this.queue[`${e2.x},${e2.y}`] && (delete action[e], true)) continue;
this.setPixel(e2.x, e2.y, e2.undo());
// console.log(e2.x, e2.y, e2.undo());
}
if (!Object.keys(action).length) {
this.undo();
return;
}
this.redoStack.push(action);
}
redo() {
if (!this.redoStack.length) return;
let action = this.redoStack.pop();
for (let e in action) {
let e2 = action[e];
if (!this.queue[`${e2.x},${e2.y}`] && (delete action[e], true)) continue;
this.setPixel(e2.x, e2.y, e2.redo());
// console.log(e2.x, e2.y, e2.redo());
}
if (!Object.keys(action).length) {
this.redo();
return;
}
this.undoStack.push(action);
}
startHistory() {
this.record = true;
}
endHistory() {
if (!this.record) return;
this.record = false;
if (Object.keys(this.actionStack).length) this.undoStack.push(this.actionStack);
this.actionStack = {};
this.redoStack = [];
}
enable() {
this.on = true;
}
disable() {
this.on = false;
}
clearQueue() {
this.queue = {};
}
unsetPixel(x, y) {
let p = new Point(x, y);
this.deletePixels(p);
return true;
}
deletePixels() {
for (let i = 0; i < arguments.length; i++) {
if (Array.isArray(arguments[i])) this.deletePixels(arguments[i]);
else if (arguments[i] instanceof Point) delete this.queue[`${arguments[i].x},${arguments[i].y}`];
}
}
setPixel(x, y, c, placeOnce = false) { // make checks for all variables coming in to make sure nothing is incorrectly set and c 4th element is either undefined or 255 otherwise drop the set
if (this.disabled) {
OWOP.world.setPixel(x, y, c);
return;
}
if (!Number.isInteger(x) || !Number.isInteger(y)) return false;
if (!Array.isArray(c) || c.length < 3 || c.length > 4) return false;
if (c.length == 4) c.pop();
if (c.find(e => !Number.isInteger(e) || e < 0 || e > 255) !== undefined) return false;
let p = new Pixel(x, y, c);
if (placeOnce) p.o = true;
let xchunk = Math.floor(p.x / 16);
let ychunk = Math.floor(p.y / 16);
if (OWOP?.misc?._world?.protectedChunks[`${xchunk},${ychunk}`]) return false;
if (this.record) {
let stackE = this.actionStack[`${x},${y}`];
if (!(stackE instanceof Action)) {
let bp = new Pixel(x, y, this.getPixel(x, y, 1));
if (bp.c !== p.c) this.actionStack[`${x},${y}`] = new Action(bp, p);
} else if (stackE.after_color !== c) {
stackE.after_color = c;
}
}
this.addPixels(p);
return true;
}
getPixel(x, y, a = 1) {
if (!Number.isInteger(x) || !Number.isInteger(y)) return console.error("There is no inputs in \"getPixel\" on PixelManager instance.");
// if (!Object.keys(OWOP.world).includes("_getPixel")) return undefined;
if (a && this.queue[`${x},${y}`]) return this.queue[`${x},${y}`].c;
try {
OWOP.world.getPixel;
} catch (e) {
return undefined;
}
return OWOP.world.getPixel(x, y);
}
addPixels() {
for (let i = 0; i < arguments.length; i++) {
if (Array.isArray(arguments[i])) this.addPixels(arguments[i]);
else if (arguments[i] instanceof Pixel) this.queue[`${arguments[i].x},${arguments[i].y}`] = arguments[i];
}
}
placePixel() {
let totalPlaced = 0;
for (let i = 0; i < this.extra.placeData.length; i++) {
let e = this.extra.placeData[i][1];
let tX = OWOP.mouse.tileX;
let tY = OWOP.mouse.tileY;
let pixel = this.queue[`${tX + e.x},${tY + e.y}`];
if (!pixel) continue;
let xchunk = Math.floor(pixel.x / 16);
let ychunk = Math.floor(pixel.y / 16);
if (OWOP?.misc?._world?.protectedChunks[`${xchunk},${ychunk}`]) continue;
let xcc = Math.floor(tX / 16) * 16;
let ycc = Math.floor(tY / 16) * 16;
if (pixel.x < (xcc - 31) || pixel.y < (ycc - 31) || pixel.x > (xcc + 46) || pixel.y > (ycc + 46)) continue;
let c = this.getPixel(pixel.x, pixel.y, 0);
if (!c) continue;
if (!Color.compare(pixel.c, c)) {
if (!OWOP.world.setPixel(pixel.x, pixel.y, pixel.c) || (++totalPlaced === 5, false)) return;
} else if (pixel.o && this.deletePixels(pixel)) continue;
}
}
}
const modulo = (i, m) => {
return i - m * Math.floor(i / m);
}
const line = (x1, y1, x2, y2, m, e, plot) => {
if (x1 === undefined || y1 === undefined || x2 === undefined || y2 === undefined) return console.error();
var dx = Math.abs(x2 - x1), sx = x1 < x2 ? 1 : -1;
var dy = -Math.abs(y2 - y1), sy = y1 < y2 ? 1 : -1;
var err = dx + dy, e2;
if (e?.type == "mousemove") {
if (x1 == x2 && y1 == y2) return;
e2 = 2 * err;
if (e2 >= dy) { err += dy; x1 += sx; }
if (e2 <= dx) { err += dx; y1 += sy; }
}
var i = 0;
while (true) {
plot(x1, y1, i);
i++;
if (x1 == x2 && y1 == y2) break;
e2 = 2 * err;
if (e2 >= dy) { err += dy; x1 += sx; }
if (e2 <= dx) { err += dx; y1 += sy; }
}
return [i];
}
NS.modulo = modulo;
NS.line = line;
NS.Point = Point;
NS.Color = Color;
NS.Pixel = Pixel;
NS.PixelManager = PixelManager;
const PM = new PixelManager();
NS.PM = PM;
NS.localStorage = localStorage.NS;
if (!NS.localStorage) {
localStorage.NS = "";
}
OWOP.OPM = false;
if (OWOP.misc) OWOP.OPM = true;
if (!OWOP.OPM) OWOP.tool = OWOP.tools;
if (localStorage.options) {
let o = JSON.parse(localStorage.options);
if (o?.enableSounds) OWOP.options.enableSounds = o.enableSounds;
}
const windows = {};
NS.windows = windows;
(function () {
var camera = OWOP.camera;
var renderer = OWOP.renderer;
var GUIWindow = OWOP.windowSys.class.window;
const mkHTML = OWOP.util.mkHTML;
const isSame = (a, b) => a && b && a[0] === b[0] && a[1] === b[1] && a[2] === b[2];
var drawText = (t, e, n, r, o) => {
t.strokeStyle = "#000000";
t.fillStyle = "#FFFFFF";
t.lineWidth = 2.5;
t.globalAlpha = 0.5;
o && (n -= t.measureText(e).width >> 1);
t.strokeText(e, n, r);
t.globalAlpha = 1;
t.fillText(e, n, r);
};
var setColor = (cursor, color) => {
if (!color) return;
if (cursor === 1) {
OWOP.player.selectedColor = color;
} else if (cursor === 2) {
OWOP.player.rightSelectedColor = color;
localStorage.setItem("rSC", JSON.stringify(OWOP.player.rightSelectedColor));
}
};
// var C = OWOP.require('util/color').colorUtils;
if (!localStorage["rSC"]) localStorage.setItem("rSC", JSON.stringify([255, 255, 255]));
OWOP.player.rightSelectedColor = JSON.parse(localStorage.getItem("rSC"));
OWOP.tool.addToolObject(new OWOP.tool.class('Cursor', OWOP.cursors.cursor, OWOP.fx.player.NONE, OWOP.RANK.USER, tool => {
// render protected chunks
tool.setFxRenderer((fx, ctx, time) => {
let defaultFx = OWOP.fx.player.RECT_SELECT_ALIGNED(1);
if (tool.extra.state.chunkize) defaultFx = OWOP.fx.player.RECT_SELECT_ALIGNED(16);
defaultFx(fx, ctx, time);
return;
if (!fx.extra.isLocalPlayer) return 1;
var x = fx.extra.player.x;
var y = fx.extra.player.y;
var fxx = (Math.floor(x / 16) - camera.x) * camera.zoom;
var fxy = (Math.floor(y / 16) - camera.y) * camera.zoom;
var oldlinew = ctx.lineWidth;
ctx.lineWidth = 1;
if (tool.extra.end) {
var s = tool.extra.start;
var e = tool.extra.end;
var x = (s[0] - camera.x) * camera.zoom + 0.5;
var y = (s[1] - camera.y) * camera.zoom + 0.5;
var w = e[0] - s[0];
var h = e[1] - s[1];
ctx.beginPath();
ctx.rect(x, y, w * camera.zoom, h * camera.zoom);
ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
ctx.setLineDash([3, 4]);
ctx.strokeStyle = "#000000";
ctx.stroke();
ctx.globalAlpha = 0.25 + Math.sin(time / 500) / 4;
ctx.fillStyle = renderer.patterns.unloaded;
ctx.fill();
ctx.setLineDash([]);
var oldfont = ctx.font;
ctx.font = "16px sans-serif";
var txt = `${!tool.extra.clicking ? "Right click to screenshot " : ""}(${Math.abs(w)}x${Math.abs(h)})`;
var txtx = window.innerWidth >> 1;
var txty = window.innerHeight >> 1;
txtx = Math.max(x, Math.min(txtx, x + w * camera.zoom));
txty = Math.max(y, Math.min(txty, y + h * camera.zoom));
drawText(ctx, txt, txtx, txty, true);
ctx.font = oldfont;
ctx.lineWidth = oldlinew;
return 0;
} else {
ctx.beginPath();
ctx.moveTo(0, fxy + 0.5);
ctx.lineTo(window.innerWidth, fxy + 0.5);
ctx.moveTo(fxx + 0.5, 0);
ctx.lineTo(fxx + 0.5, window.innerHeight);
//ctx.lineWidth = 1;
ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
ctx.setLineDash([3]);
ctx.strokeStyle = "#000000";
ctx.stroke();
ctx.setLineDash([]);
ctx.lineWidth = oldlinew;
return 1;
}
});
// cursor functionality
tool.extra.state = {
scalar: "1",
rainbow: false,
chunkize: false,
perfect: false
};
tool.extra.lastX;
tool.extra.lastY;
tool.extra.last1PX;
tool.extra.last1PY;
tool.extra.last2PX;
tool.extra.last2PY;
tool.extra.start;
tool.extra.c = 0;
tool.setEvent('mousedown mousemove', (mouse, event) => {
if (mouse.buttons !== 2 && mouse.buttons !== 1) return 3;
if (tool.extra.lastX == mouse.tileX && tool.extra.lastY == mouse.tileY) return 3;
if (event?.ctrlKey) return setColor(mouse.buttons, PM.getPixel(mouse.tileX, mouse.tileY));
let c = mouse.buttons === 1 ? OWOP.player.selectedColor : OWOP.player.rightSelectedColor;
if (isNaN(tool.extra.lastX) || isNaN(tool.extra.lastY)) {
tool.extra.lastX = mouse.tileX;
tool.extra.lastY = mouse.tileY;
tool.extra.last1PX = mouse.tileX;
tool.extra.last1PY = mouse.tileY;
tool.extra.last2PX = mouse.tileX;
tool.extra.last2PY = mouse.tileY;
tool.extra.start = true;
}
PM.startHistory();
line(tool.extra.lastX, tool.extra.lastY, mouse.tileX, mouse.tileY, undefined, event, (x, y) => {
let tempx = x;
let tempy = y;
if (tool.extra.state.perfect) {
let place = false;
// check to place
if (tool.extra.start) {
tool.extra.start = false;
place = true;
} else {
if (tool.extra.last1PX == x && tool.extra.last1PY == y) {
tool.extra.last2PX = tool.extra.last1PX;
tool.extra.last2PY = tool.extra.last1PY;
tool.extra.last1PX = x;
tool.extra.last1PY = y;
return;
}
}
if (!place) {
for (let x1 = -1; x1 < 2; x1++) {
for (let y1 = -1; y1 < 2; y1++) {
if (x1 == 0 && y1 == 0) continue;
if (tool.extra.last2PX === (x + x1) && tool.extra.last2PY === (y + y1)) {
tool.extra.last1PX = x;
tool.extra.last1PY = y;
return;
}
}
}
}
tempx = tool.extra.last1PX;
tempy = tool.extra.last1PY;
}
let test = false;
if (tool.extra.state.rainbow) {
let pixel;
if ((pixel = PM.getPixel(tempx, tempy), !pixel)) return;
c = mouse.buttons === 1 ? hue(tempx - tempy, 8) : hue(tool.extra.c++, 8);
if (!Color.compare(pixel, c)) test = true;
} else {
test = true;
}
if (test) {
let size = Number(tool.extra.state.scalar);
if (tool.extra.state.chunkize) size = 16;
let offset = Math.floor(size / 2);
for (let x1 = 0; x1 < size; x1++) {
for (let y1 = 0; y1 < size; y1++) {
if (tool.extra.state.chunkize) {
tempx = Math.floor(tempx / 16) * 16;
tempy = Math.floor(tempy / 16) * 16;
offset = 0;
}
PM.setPixel(tempx + x1 - offset, tempy + y1 - offset, c);
//[Math.round(OWOP.mouse.worldX/16)-0.5, Math.round(OWOP.mouse.worldY/16)-0.5]
}
}
}
if (tool.extra.state.perfect) {
tool.extra.last2PX = tool.extra.last1PX;
tool.extra.last2PY = tool.extra.last1PY;
tool.extra.last1PX = x;
tool.extra.last1PY = y;
}
});
tool.extra.lastX = mouse.tileX;
tool.extra.lastY = mouse.tileY;
return 3;
});
tool.setEvent('mouseup deselect', () => {
PM.endHistory();
tool.extra.lastX = undefined;
tool.extra.lastY = undefined;
tool.extra.last1PX = undefined;
tool.extra.last1PY = undefined;
tool.extra.last2PX = undefined;
tool.extra.last2PY = undefined;
});
// change color positions
tool.setEvent('keydown', keys => {
if ((keys["87"] && keys["83"]) || !keys["16"]) return;
if (keys["87"]) { // w
let i1 = OWOP.player.paletteIndex;
let i2 = modulo(i1 - 1, OWOP.player.palette.length);
if (i2 == OWOP.player.palette.length - 1) {
OWOP.player.palette.push(OWOP.player.palette.shift());
} else {
[OWOP.player.palette[i1], OWOP.player.palette[i2]] = [OWOP.player.palette[i2], OWOP.player.palette[i1]];
}
OWOP.player.paletteIndex = i2;
}
if (keys["83"]) { // s
let i1 = OWOP.player.paletteIndex;
let i2 = modulo(i1 + 1, OWOP.player.palette.length);
if (i2 === 0) {
OWOP.player.palette.unshift(OWOP.player.palette.pop());
} else {
[OWOP.player.palette[i1], OWOP.player.palette[i2]] = [OWOP.player.palette[i2], OWOP.player.palette[i1]];
}
OWOP.player.paletteIndex = i2;
}
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Pipette', OWOP.cursors.pipette, OWOP.fx.player.NONE, OWOP.RANK.NONE, tool => {
tool.setEvent('mousedown mousemove', mouse => {
var c = PM.getPixel(mouse.tileX, mouse.tileY);
if (!c) return mouse.buttons;
switch (mouse.buttons) {
case 1:
OWOP.player.selectedColor = c;
break;
case 2:
OWOP.player.rightSelectedColor = c;
localStorage.setItem("rSC", JSON.stringify(OWOP.player.rightSelectedColor));
break;
}
return mouse.buttons;
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Export', OWOP.cursors.select, OWOP.fx.player.NONE, OWOP.RANK.NONE, tool => {
tool.extra.state = {
type: "export",
rainbow: false,
chunkize: false
};
tool.setFxRenderer((fx, ctx, time) => {
if (!fx.extra.isLocalPlayer) return 1;
var x = fx.extra.player.x;
var y = fx.extra.player.y;
var fxx = (Math.floor(x / 16) - camera.x) * camera.zoom;
var fxy = (Math.floor(y / 16) - camera.y) * camera.zoom;
var oldlinew = ctx.lineWidth;
ctx.lineWidth = 1;
if (tool.extra.end) {
var s = tool.extra.start;
var e = tool.extra.end;
var x = s[0];
var y = s[1];
var w = e[0];
var h = e[1];
if (s[0] > e[0]) [w, x] = [x, w];
if (s[1] > e[1]) [h, y] = [y, h];
if (tool.extra.state.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
w = Math.floor(w / 16) * 16 + 16;
h = Math.floor(h / 16) * 16 + 16;
}
w = w - x;
h = h - y;
x = (x - camera.x) * camera.zoom + 0.5;
y = (y - camera.y) * camera.zoom + 0.5;
ctx.beginPath();
ctx.rect(x, y, w * camera.zoom, h * camera.zoom);
ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
ctx.setLineDash([3, 4]);
ctx.strokeStyle = "#000000";
ctx.stroke();
ctx.globalAlpha = 0.25 + Math.sin(time / 500) / 4;
ctx.fillStyle = renderer.patterns.unloaded;
ctx.fill();
ctx.setLineDash([]);
var oldfont = ctx.font;
ctx.font = "16px sans-serif";
var txt = `${!tool.extra.clicking ? "Right click " : ""}(${Math.abs(w)}x${Math.abs(h)})`;
var txtx = window.innerWidth >> 1;
var txty = window.innerHeight >> 1;
txtx = Math.max(x, Math.min(txtx, x + w * camera.zoom));
txty = Math.max(y, Math.min(txty, y + h * camera.zoom));
drawText(ctx, txt, txtx, txty, true);
ctx.font = oldfont;
ctx.lineWidth = oldlinew;
return 0;
} else {
var x = fx.extra.player.x;
var y = fx.extra.player.y;
var fxx = Math.floor(x / 16);
var fxy = Math.floor(y / 16);
if (tool.extra.state.chunkize) {
fxx = Math.floor(fxx / 16) * 16;
fxy = Math.floor(fxy / 16) * 16;
}
fxx -= camera.x;
fxy -= camera.y;
fxx *= camera.zoom;
fxy *= camera.zoom;
ctx.beginPath();
ctx.moveTo(0, fxy + 0.5);
ctx.lineTo(window.innerWidth, fxy + 0.5);
ctx.moveTo(fxx + 0.5, 0);
ctx.lineTo(fxx + 0.5, window.innerHeight);
//ctx.lineWidth = 1;
ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
ctx.setLineDash([3]);
ctx.strokeStyle = "#000000";
ctx.stroke();
ctx.setLineDash([]);
ctx.lineWidth = oldlinew;
return 1;
}
});
tool.extra.start = undefined;
tool.extra.end = undefined;
tool.extra.clicking = false;
tool.setEvent('mousedown', (mouse, event) => {
var s = tool.extra.start;
var e = tool.extra.end;
const isInside = () => {
var x = s[0];
var y = s[1];
var w = e[0];
var h = e[1];
if (tool.extra.state.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
w = Math.floor(w / 16) * 16 + 16;
h = Math.floor(h / 16) * 16 + 16;
}
return mouse.tileX >= x && mouse.tileX < w && mouse.tileY >= y && mouse.tileY < h;
}
if (mouse.buttons === 1 && !tool.extra.end) {
tool.extra.start = [mouse.tileX, mouse.tileY];
tool.extra.clicking = true;
tool.setEvent('mousemove', (mouse, event) => {
if (tool.extra.start && mouse.buttons === 1) {
tool.extra.end = [mouse.tileX, mouse.tileY];
return 1;
}
});
const finish = () => {
tool.setEvent('mousemove mouseup deselect', null);
tool.extra.clicking = false;
var s = tool.extra.start;
var e = tool.extra.end;
var tmp = undefined;
if (e) {
if (s[0] === e[0] || s[1] === e[1]) {
tool.extra.start = undefined;
tool.extra.end = undefined;
}
if (s[0] > e[0]) {
tmp = e[0];
e[0] = s[0];
s[0] = tmp;
}
if (s[1] > e[1]) {
tmp = e[1];
e[1] = s[1];
s[1] = tmp;
}
}
renderer.render(renderer.rendertype.FX);
}
tool.setEvent('deselect', finish);
tool.setEvent('mouseup', (mouse, event) => {
if (!(mouse.buttons & 1)) {
finish();
}
});
} else if (mouse.buttons === 1 && tool.extra.end) {
if (isInside()) {
var offx = mouse.tileX;
var offy = mouse.tileY;
tool.setEvent('mousemove', (mouse, event) => {
var dx = mouse.tileX - offx;
var dy = mouse.tileY - offy;
tool.extra.start = [s[0] + dx, s[1] + dy];
tool.extra.end = [e[0] + dx, e[1] + dy];
});
const end = () => {
tool.setEvent('mouseup deselect mousemove', null);
}
tool.setEvent('deselect', end);
tool.setEvent('mouseup', (mouse, event) => {
if (!(mouse.buttons & 1)) {
end();
}
});
} else {
tool.extra.start = undefined;
tool.extra.end = undefined;
}
} else if (mouse.buttons === 2 && tool.extra.end && isInside()) {
tool.extra.start = undefined;
tool.extra.end = undefined;
var x = s[0];
var y = s[1];
var w = e[0];
var h = e[1];
if (tool.extra.state.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
w = Math.floor(w / 16) * 16 + 16;
h = Math.floor(h / 16) * 16 + 16;
}
w -= x;
h -= y;
let warn = false;
switch (tool.extra.state.type) {
case "export": {
((x, y, w, h, onblob) => {
var c = document.createElement('canvas');
c.width = w;
c.height = h;
var ctx = c.getContext('2d');
var d = ctx.createImageData(w, h);
for (var i = y; i < y + h; i++) {
for (var j = x; j < x + w; j++) {
let pix;
let tempPix = PM.queue[`${j},${i}`];
if (!tempPix) {
if ((pix = PM.getPixel(j, i), !pix)) {
warn = true;
pix = [255, 255, 255];
}
} else {
pix = tempPix.c;
}
d.data[4 * ((i - y) * w + (j - x))] = pix[0];
d.data[4 * ((i - y) * w + (j - x)) + 1] = pix[1];
d.data[4 * ((i - y) * w + (j - x)) + 2] = pix[2];
d.data[4 * ((i - y) * w + (j - x)) + 3] = 255;
}
}
ctx.putImageData(d, 0, 0);
c.toBlob(onblob);
})(x, y, w, h, b => {
var url = URL.createObjectURL(b);
var img = new Image();
img.onload = () => {
if (OWOP.windowSys.windows['Resulting image']) OWOP.windowSys.delWindow(OWOP.windowSys.windows['Resulting image']);
OWOP.windowSys.addWindow(new GUIWindow("Resulting image", {
centerOnce: true,
closeable: true
}, win => {
var props = ['width', 'height'];
if (img.width > img.height) {
props.reverse();
}
var r = img[props[0]] / img[props[1]];
var shownSize = img[props[1]] >= 128 ? 256 : 128;
img[props[0]] = r * shownSize;
img[props[1]] = shownSize;
//win.container.classList.add('centeredChilds');
//setTooltip(img, "Right click to copy/save!");
var p1 = document.createElement("p");
img.style = "display:block; margin-left: auto; margin-right: auto; padding-bottom:15px;";
p1.appendChild(img);
//p1.appendChild(document.createElement("br"));
var closeButton = mkHTML("button", {
innerHTML: "CLOSE",
style: "width: 100%; height: 30px; margin: auto; padding-left: 10%;",
onclick: () => {
img.remove();
URL.revokeObjectURL(url);
win.getWindow().close();
}
});
var saveButton = mkHTML("button", {
innerHTML: "SAVE",
style: "width: 100%; height: 30px; margin: auto; padding-left: 10%;"
});
saveButton.onclick = () => {
var a = document.createElement('a');
a.download = `${Base64.fromNumber(Date.now())} OWOP_${OWOP.world.name} at ${s[0]} ${s[1]}.png`;
a.href = img.src;
a.click();
}
var scalar = document.createElement("input");
scalar.id = "scalar";
scalar.type = "range";
scalar.style = "width: 100%; margin: auto;";
scalar.value = "1";
scalar.min = "1";
scalar.max = "10";
//<p id="scalar-num" style="margin: auto;top: -8px; right: 12px; user-select: none; color: white;">1</p>
scalar.oninput = () => {
// scalarNum.textContent = scalar.value;
}
p1.appendChild(saveButton);
p1.appendChild(closeButton);
p1.appendChild(scalar);
var image = win.addObj(p1);
}));
}
img.src = url;
});
} break;
case "color": {
let test = false;
let totalAdded = 0;
let limit = 50;
for (var i = x; i < x + w; i++) {
for (var j = y; j < y + h; j++) {
if (totalAdded >= limit) continue;
var pix = PM.getPixel(i, j);
if (!pix) continue;
for (let k = 0; k < OWOP.player.palette.length; k++) {
var c = OWOP.player.palette[k];
if (isSame(c, pix)) {
test = true;
break;
}
}
if (test) {
test = false;
continue;
}
OWOP.player.palette.push(pix);
totalAdded++;
}
}
OWOP.player.paletteIndex = OWOP.player.palette.length - 1;
if (totalAdded >= limit) OWOP.chat.local(`total colors added limit has been reached (${limit} added)`);
} break;
case "adder": {
for (var i = x; i < x + w; i++) {
for (var j = y; j < y + h; j++) {
var pix = PM.getPixel(i, j);
if (pix && !PM.queue[`${i},${j}`]) PM.setPixel(i, j, pix);
}
}
} break;
case "filler": {
var pix = OWOP.player.selectedColor;
PM.startHistory();
for (var i = x; i < x + w; i++) {
for (var j = y; j < y + h; j++) {
PM.setPixel(i, j, pix);
}
}
PM.endHistory();
} break;
case "clearer": {
for (var i = x; i < x + w; i++) {
for (var j = y; j < y + h; j++) {
PM.unsetPixel(i, j);
}
}
} break;
}
if (warn) console.warn("Well something happened, you probably tried getting an area outside of loaded chunks.");
}
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Fill', OWOP.cursors.fill, OWOP.fx.player.NONE, OWOP.RANK.USER, tool => {
tool.extra.state = {
rainbow: false,
checkered: false,
dither: false,
dither2: false,
dither3: false,
dither4: false,
dither5: false,
dither6: false
}
tool.extra.usedQueue = {};
tool.extra.queue = {};
tool.extra.fillingColor = undefined;
tool.extra.button = 0;
tool.extra.checkered = 0;
const isFillColor = (x, y) => isSame(PM.getPixel(x, y), tool.extra.fillingColor) && (!tool.extra.usedQueue[`${x},${y}`]) && (tool.extra.queue[`${x},${y}`] = { x: x, y: y }, true);
function tick() {
var selClr = tool.extra.button === 1 ? OWOP.player.selectedColor : OWOP.player.rightSelectedColor;
for (var current in tool.extra.queue) {
current = tool.extra.queue[current];
var x = current.x;
var y = current.y;
if (tool.extra.state.rainbow) selClr = hue(x - y, 8);
var thisClr = PM.getPixel(x, y);
if (isSame(thisClr, tool.extra.fillingColor) && !isSame(thisClr, selClr)) {
if (tool.extra.state.checkered) {
let pattern = [
[1, 0],
[0, 1]
];
if (pattern[modulo(x, 2)][modulo(y, 2)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither) {
let pattern = [
[1, 0, 1, 0],
[0, 1, 0, 0],
[1, 0, 1, 0],
[0, 0, 0, 1]
];
if (pattern[modulo(x, 4)][modulo(y, 4)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither2) {
let pattern = [
[0, 0],
[0, 1],
[0, 0],
[1, 0]
];
if (pattern[modulo(x, 4)][modulo(y, 2)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither3) {
let pattern = [
[1, 0, 0, 0, 1, 0, 1, 0],
[0, 1, 0, 1, 0, 1, 0, 0],
[1, 0, 1, 0, 0, 0, 1, 0],
[0, 0, 0, 1, 0, 1, 0, 1],
[1, 0, 1, 0, 1, 0, 0, 0],
[0, 1, 0, 0, 0, 1, 0, 1],
[0, 0, 1, 0, 1, 0, 1, 0],
[0, 1, 0, 1, 0, 0, 0, 1]
];
if (pattern[modulo(x, 8)][modulo(y, 8)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither4) {
let pattern = [
[0, 1, 0, 0],
[1, 1, 0, 0],
[0, 0, 1, 1],
[0, 0, 1, 0]
];
if (pattern[modulo(x, 4)][modulo(y, 4)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither5) {
let pattern = [
[0, 1, 0, 0, 0, 0, 1, 0],
[1, 1, 0, 0, 0, 0, 1, 1],
[0, 0, 1, 1, 1, 1, 0, 0],
[0, 0, 1, 0, 0, 1, 0, 0],
[0, 0, 1, 0, 0, 1, 0, 0],
[0, 0, 1, 1, 1, 1, 0, 0],
[1, 1, 0, 0, 0, 0, 1, 1],
[0, 1, 0, 0, 0, 0, 1, 0]
];
if (pattern[modulo(x, 8)][modulo(y, 8)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither6) {
let pattern = [
[0, 1, 1, 0, 0],
[1, 1, 1, 1, 0],
[1, 0, 1, 0, 0],
[1, 0, 1, 1, 0],
[0, 0, 0, 0, 0]
];
if (pattern[modulo(x, 5)][modulo(y, 5)]) PM.setPixel(x, y, selClr);
} else if (tool.extra.state.dither7) {
let pattern = [
[1, 1, 1, 0, 1, 1, 1, 0, 1, 0],
[0, 0, 0, 0, 1, 0, 1, 0, 0, 0],
[1, 0, 1, 0, 1, 1, 1, 0, 1, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 1, 0],
[1, 0, 1, 1, 1, 0, 1, 0, 1, 1],
[0, 0, 1, 0, 1, 0, 0, 0, 0, 0],
[1, 0, 1, 1, 1, 0, 1, 1, 1, 0],
[0, 0, 0, 0, 0, 0, 1, 0, 1, 0],
[1, 1, 1, 0, 1, 0, 1, 1, 1, 0],
[1, 0, 1, 0, 0, 0, 0, 0, 0, 0]
];
if (pattern[modulo(x, 10)][modulo(y, 10)]) PM.setPixel(x, y, selClr);
} else {
PM.setPixel(x, y, selClr);
}
var t = isFillColor(x, y - 1);
var b = isFillColor(x, y + 1);
var l = isFillColor(x - 1, y);
var r = isFillColor(x + 1, y);
t && l && isFillColor(x - 1, y - 1);
t && r && isFillColor(x + 1, y - 1);
b && l && isFillColor(x - 1, y + 1);
b && r && isFillColor(x + 1, y + 1);
}
delete tool.extra.queue[`${x},${y}`];
tool.extra.usedQueue[`${x},${y}`] = true;
}
}
tool.setFxRenderer((fx, ctx, time) => {
ctx.globalAlpha = 0.8;
ctx.strokeStyle = rgb(...(tool.extra.button === 1 ? OWOP.player.selectedColor : OWOP.player.rightSelectedColor));
var z = OWOP.camera.zoom;
if (!tool.extra.fillingColor || !fx.extra.isLocalPlayer) return OWOP.fx.player.RECT_SELECT_ALIGNED(1)(fx, ctx, time);
ctx.beginPath();
for (var current in tool.extra.queue) {
current = tool.extra.queue[current];
if (tool.extra.state.rainbow) ctx.strokeStyle = rgb(...hue(current.x - current.y, 8));
let x = current.x
let y = current.y;
// if (tool.extra.state.checkered) {
// if ((x + y) - 2 * Math.floor((x + y) / 2) == tool.extra.checkered) ctx.rect((x - OWOP.camera.x) * z, (y - OWOP.camera.y) * z, z, z);
// } else {
ctx.rect((x - OWOP.camera.x) * z, (y - OWOP.camera.y) * z, z, z);
// }
}
ctx.stroke();
});
tool.setEvent("mousedown", (mouse, event) => {
if (event.which !== 1 && event.which !== 3) return;
tool.extra.button = event.which;
tool.extra.fillingColor = PM.getPixel(mouse.tileX, mouse.tileY);
tool.extra.queue[`${mouse.tileX},${mouse.tileY}`] = { x: mouse.tileX, y: mouse.tileY };
tool.extra.checkered = (mouse.tileX + mouse.tileY) - 2 * Math.floor((mouse.tileX + mouse.tileY) / 2);
PM.startHistory();
tool.setEvent("tick", tick);
});
tool.setEvent("mouseup deselect", mouse => {
PM.endHistory();
tool.extra.usedQueue = {};
tool.extra.queue = {};
tool.extra.fillingColor = undefined;
tool.extra.button = 0;
tool.extra.checkered = 0;
tool.setEvent("tick", null);
return mouse && 1 & mouse.buttons;
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Line', OWOP.cursors.wand, OWOP.fx.player.NONE, OWOP.RANK.USER, tool => {
tool.extra.state = {
rainbow: false,
gradient: false
};
tool.extra.start = undefined;
tool.extra.end = undefined;
tool.extra.lineLength = 0;
tool.extra.c = 0;
var defaultFx = OWOP.fx.player.RECT_SELECT_ALIGNED(1);
tool.setFxRenderer((fx, ctx, time) => {
ctx.globalAlpha = 0.8;
ctx.strokeStyle = rgb(...(tool.extra.button === 1 ? OWOP.player.selectedColor : OWOP.player.rightSelectedColor));
if (tool.extra.state.rainbow) ctx.strokeStyle = rgb(...hue(~~(time / 100), 8));
if ((!tool.extra.start || !tool.extra.end) && (defaultFx(fx, ctx, time), true)) return;
tool.extra.lineLength = line(tool.extra.start[0], tool.extra.start[1], tool.extra.end[0], tool.extra.end[1], undefined, undefined, (x, y, i) => {
ctx.beginPath();
if (tool.extra.state.rainbow) ctx.strokeStyle = rgb(...hue(~~(time / 100) + i, 8));
ctx.rect((x - camera.x) * camera.zoom, (y - camera.y) * camera.zoom, camera.zoom, camera.zoom);
ctx.stroke();
})[0];
});
tool.setEvent('mousedown mouseup', (mouse, event) => {
if (event.which !== 1 && event.which !== 3) return;
tool.extra.button = event.which;
if (event.type === "mousedown" && !tool.extra.start) return tool.extra.start = [mouse.tileX, mouse.tileY];
if (!tool.extra.start) return;
tool.extra.end = [mouse.tileX, mouse.tileY];
if (event.type === "mouseup" && tool.extra.start[0] === tool.extra.end[0] && tool.extra.start[1] === tool.extra.end[1]) return;
PM.startHistory();
let sc = PM.getPixel(tool.extra.start[0], tool.extra.start[1]);
line(tool.extra.start[0], tool.extra.start[1], tool.extra.end[0], tool.extra.end[1], undefined, undefined, (x, y, i) => {
let c = event.which === 1 ? OWOP.player.selectedColor : OWOP.player.rightSelectedColor;
if (tool.extra.state.gradient) {
let divisor = (tool.extra.lineLength - 1);
let r = sc[0] - ((sc[0] - c[0]) / divisor) * i;
let g = sc[1] - ((sc[1] - c[1]) / divisor) * i;
let b = sc[2] - ((sc[2] - c[2]) / divisor) * i;
c = [~~r, ~~g, ~~b];
if (i == 0) c = sc;
} else if (tool.extra.state.rainbow) c = event.which === 1 ? hue(x - y, 8) : hue(tool.extra.c++, 8);
PM.setPixel(x, y, c);
});
PM.endHistory();
tool.extra.start = undefined;
tool.extra.end = undefined;
});
tool.setEvent('mousemove', mouse => {
if (tool.extra.start) tool.extra.end = [mouse.tileX, mouse.tileY];
});
tool.setEvent('deselect', () => {
PM.endHistory();
tool.extra.start = undefined;
tool.extra.end = undefined;
tool.extra.c = 0;
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Paste', OWOP.cursors.paste, OWOP.fx.player.NONE, OWOP.RANK.USER, tool => {
let c = document.createElement('canvas');
c.width = 0;
c.height = 0;
let p1 = c.getContext('2d');
tool.setFxRenderer((fx, ctx, time) => {
let p9 = OWOP.camera.zoom;
let pp = OWOP.mouse.tileX;
let pD = OWOP.mouse.tileY;
if (NS.chunkize) {
pp = Math.floor(pp / 16) * 16;
pD = Math.floor(pD / 16) * 16;
}
pp -= OWOP.camera.x;
pD -= OWOP.camera.y;
// if (p2.length) {
// ctx.globalAlpha = 0.8;
//for (let pS = 0; pS < p2.length; pS++) {
// ctx.strokeStyle = C.toHTML(p2[pS][2]);
// ctx.strokeRect((p2[pS][0] - OWOP.camera.x) * p9, (p2[pS][1] - OWOP.camera.y) * p9, p9, p9);
//}
// return 0;
// }
if (fx.extra.isLocalPlayer && c.width && c.height) {
ctx.globalAlpha = 0.5 + Math.sin(time / 500) / 4;
ctx.strokeStyle = '#000000';
ctx.scale(p9, p9);
ctx.drawImage(c, pp, pD);
ctx.scale(1 / p9, 1 / p9);
ctx.globalAlpha = 0.8;
ctx.strokeRect(pp * p9, pD * p9, c.width * p9, c.height * p9);
return 0;
}
});
tool.setEvent('select', () => {
if (tool.extra.k) {
c.width = tool.extra.k.width;
c.height = tool.extra.k.height;
p1.drawImage(tool.extra.k, 0, 0);
delete tool.extra.k;
return;
}
let p6 = document.createElement('input');
p6.type = 'file';
p6.accept = 'image/*';
p6.addEventListener('change', () => {
if (!p6.files || !p6.files[0]) return;
let p7 = new FileReader();
p7.addEventListener('load', () => {
let p8 = new Image();
p8.src = p7.result;
p8.addEventListener('load', () => {
c.width = p8.width;
c.height = p8.height;
p1.drawImage(p8, 0, 0);
});
});
p7.readAsDataURL(p6.files[0]);
});
p6.click();
});
tool.setEvent('mousedown', (mouse, event) => {
if (!(mouse.buttons & 1)) return;
let x = mouse.tileX;
let y = mouse.tileY;
let data = p1.getImageData(0, 0, c.width, c.height).data;
let fix = (p6, p7, p8) => Math.floor(p6 * (1 - p8) + p7 * p8);
if (NS.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
}
PM.startHistory();
for (let j = 0; j < c.height; j++) {
for (let i = 0; i < c.width; i++) {
let pD = (j * c.width + i) * 4;
let color = PM.getPixel(i + x, j + y);
if (!color) continue;
let pH = data[pD + 3] / 255;
color = [fix(color[0], data[pD + 0], pH), fix(color[1], data[pD + 1], pH), fix(color[2], data[pD + 2], pH)];
// use this when color is checked against being alpha color cause this is stupid
// var pix = PM.getPixel(i, j);
// if (!PM.queue[`${i},${j}`]) PM.setPixel(i, j, pix);
PM.setPixel(i + x, j + y, color);
}
}
PM.endHistory();
});
tool.setEvent('mousemove', (mouse, event) => {
if (!OWOP.OPM) return;
if (mouse.buttons !== 0) {
((x, y, startX, startY) => {
OWOP.require("canvas_renderer").moveCameraBy((startX - x) / 16, (startY - y) / 16);
})(mouse.worldX, mouse.worldY, mouse.mouseDownWorldX, mouse.mouseDownWorldY);
return mouse.buttons;
}
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Copy', OWOP.cursors.copy, OWOP.fx.player.NONE, OWOP.RANK.USER, tool => {
tool.setFxRenderer((fx, ctx, time) => {
if (!fx.extra.isLocalPlayer) return 1;
var x = fx.extra.player.x;
var y = fx.extra.player.y;
var fxx = (Math.floor(x / 16) - camera.x) * camera.zoom;
var fxy = (Math.floor(y / 16) - camera.y) * camera.zoom;
var oldlinew = ctx.lineWidth;
ctx.lineWidth = 1;
if (tool.extra.end) {
var s = tool.extra.start;
var e = tool.extra.end;
var x = s[0];
var y = s[1];
var w = e[0];
var h = e[1];
if (s[0] > e[0]) [w, x] = [x, w];
if (s[1] > e[1]) [h, y] = [y, h];
if (NS.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
w = Math.floor(w / 16) * 16 + 16;
h = Math.floor(h / 16) * 16 + 16;
}
w = w - x;
h = h - y;
x = (x - camera.x) * camera.zoom + 0.5;
y = (y - camera.y) * camera.zoom + 0.5;
ctx.beginPath();
ctx.rect(x, y, w * camera.zoom, h * camera.zoom);
ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
ctx.setLineDash([3, 4]);
ctx.strokeStyle = "#000000";
ctx.stroke();
ctx.globalAlpha = 0.25 + Math.sin(time / 500) / 4;
ctx.fillStyle = renderer.patterns.unloaded;
ctx.fill();
ctx.setLineDash([]);
var oldfont = ctx.font;
ctx.font = "16px sans-serif";
var txt = `${!tool.extra.clicking ? "Right click to copy area " : ""}(${Math.abs(w)}x${Math.abs(h)})`;
var txtx = window.innerWidth >> 1;
var txty = window.innerHeight >> 1;
txtx = Math.max(x, Math.min(txtx, x + w * camera.zoom));
txty = Math.max(y, Math.min(txty, y + h * camera.zoom));
drawText(ctx, txt, txtx, txty, true);
ctx.font = oldfont;
ctx.lineWidth = oldlinew;
return 0;
} else {
ctx.beginPath();
ctx.moveTo(0, fxy + 0.5);
ctx.lineTo(window.innerWidth, fxy + 0.5);
ctx.moveTo(fxx + 0.5, 0);
ctx.lineTo(fxx + 0.5, window.innerHeight);
//ctx.lineWidth = 1;
ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
ctx.setLineDash([3]);
ctx.strokeStyle = "#000000";
ctx.stroke();
ctx.setLineDash([]);
ctx.lineWidth = oldlinew;
return 1;
}
});
tool.extra.start = undefined;
tool.extra.end = undefined;
tool.extra.clicking = false;
tool.setEvent('mousedown', (mouse, event) => {
var s = tool.extra.start;
var e = tool.extra.end;
const isInside = () => {
var x = s[0];
var y = s[1];
var w = e[0];
var h = e[1];
if (NS.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
w = Math.floor(w / 16) * 16 + 16;
h = Math.floor(h / 16) * 16 + 16;
}
return mouse.tileX >= x && mouse.tileX < w && mouse.tileY >= y && mouse.tileY < h;
}
if (mouse.buttons === 1 && !tool.extra.end) {
tool.extra.start = [mouse.tileX, mouse.tileY];
tool.extra.clicking = true;
tool.setEvent('mousemove', (mouse, event) => {
if (tool.extra.start && mouse.buttons === 1) {
tool.extra.end = [mouse.tileX, mouse.tileY];
return 1;
}
});
const finish = () => {
tool.setEvent('mousemove mouseup deselect', null);
tool.extra.clicking = false;
var s = tool.extra.start;
var e = tool.extra.end;
if (e) {
if (s[0] === e[0] || s[1] === e[1]) {
tool.extra.start = undefined;
tool.extra.end = undefined;
}
if (s[0] > e[0]) {
var tmp = e[0];
e[0] = s[0];
s[0] = tmp;
}
if (s[1] > e[1]) {
var tmp = e[1];
e[1] = s[1];
s[1] = tmp;
}
}
renderer.render(renderer.rendertype.FX);
}
tool.setEvent('deselect', finish);
tool.setEvent('mouseup', (mouse, event) => {
if (!(mouse.buttons & 1)) {
finish();
}
});
} else if (mouse.buttons === 1 && tool.extra.end) {
if (isInside()) {
var offx = mouse.tileX;
var offy = mouse.tileY;
tool.setEvent('mousemove', (mouse, event) => {
var dx = mouse.tileX - offx;
var dy = mouse.tileY - offy;
tool.extra.start = [s[0] + dx, s[1] + dy];
tool.extra.end = [e[0] + dx, e[1] + dy];
});
const end = () => {
tool.setEvent('mouseup deselect mousemove', null);
}
tool.setEvent('deselect', end);
tool.setEvent('mouseup', (mouse, event) => {
if (!(mouse.buttons & 1)) {
end();
}
});
} else {
tool.extra.start = undefined;
tool.extra.end = undefined;
}
} else if (mouse.buttons === 2 && tool.extra.end && isInside()) {
tool.extra.start = undefined;
tool.extra.end = undefined;
var x = s[0];
var y = s[1];
var w = e[0];
var h = e[1];
if (NS.chunkize) {
x = Math.floor(x / 16) * 16;
y = Math.floor(y / 16) * 16;
w = Math.floor(w / 16) * 16 + 16;
h = Math.floor(h / 16) * 16 + 16;
}
w -= x;
h -= y;
((x, y, w, h, onblob) => {
var c = document.createElement('canvas');
c.width = w;
c.height = h;
var ctx = c.getContext('2d');
var d = ctx.createImageData(w, h);
for (var i = y; i < y + h; i++) {
for (var j = x; j < x + w; j++) {
let pix;
if ((pix = PM.getPixel(j, i), !pix)) {
console.warn("Well something happened, you probably tried getting an area outside of loaded chunks.");
pix = [255, 255, 255];
}
d.data[4 * ((i - y) * w + (j - x))] = pix[0];
d.data[4 * ((i - y) * w + (j - x)) + 1] = pix[1];
d.data[4 * ((i - y) * w + (j - x)) + 2] = pix[2];
d.data[4 * ((i - y) * w + (j - x)) + 3] = 255;
}
}
ctx.putImageData(d, 0, 0);
c.toBlob(onblob);
})(x, y, w, h, b => {
var url = URL.createObjectURL(b);
var img = new Image();
img.onload = () => {
OWOP.tool.allTools.paste.extra.k = img;
OWOP.player.tool = "paste";
}
img.src = url;
});
}
});
}));
OWOP.tool.addToolObject(new OWOP.tool.class('Write', OWOP.cursors.write, OWOP.fx.player.NONE, OWOP.RANK.USER, tool => {
tool.text = "211111211111122111112111111211111111111111211111111121111111111222211111121111112222111211111121112111112111111221111121111112211111111111111112111111211111111111112111111211111111112222222110001110000111100011100001110000011000001110000110121011000001222210110111011012222101110110111011100011100001111000111000011110000110000011012101101210110101011011101101210110000012222222101110110111011011101101110110111111011111101111110111011110111222210110110111012222100100110011011011101101110110111011011101101111111101111012101101110110101011101011101110111110112222222100000110000111012111101210110000011000001101000110000012210122111210110001121012222101010110101011012101100001110101011000011110001122101221012101110101110101012110112110101121101122222222101110110111011011101101110110111111011111101110110111011110111101110110110111011111101110110110011011101101111210110111011101111110122101221011101210101210101011101011211011211011112222222101210110000111100011100001110000011012222110001110121011000001110001110111011000001101210110111011100011101222211001011012101100001122101221100011211011211010111011101221012210000012222222111211111111122111112111111211111111112222211111211121111111111211111211121111111111111211111121112111112111222221111111112111111111222111222111112221112221111121112111221112211111112222222111111111211111111111111111111111111111111111111112222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222100011001210001100011010110001100011000110001100012222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222101011101211101111011010110111101111110110101101012222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222101012101210001100011000110001100012210110001100012222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222101011101110111111011110111101101012210110101111012222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222100011000110001100012210110001100012210110001100012222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222111111111111111111112211111111111112211111111111112222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222";
tool.extra.position = 0;
tool.extra.start = undefined;
tool.extra.end = undefined;
tool.extra.textArea = undefined;
tool.setFxRenderer((fx, ctx, time) => {
var camera = OWOP.camera;
if (!fx.extra.isLocalPlayer) return 1;
var x = fx.extra.player.x;
var y = fx.extra.player.y;
var fxx = (Math.floor(x / 16) - camera.x) * camera.zoom;
var fxy = (Math.floor(y / 16) - camera.y) * camera.zoom;
var oldlinew = ctx.lineWidth;
ctx.lineWidth = 5;
var s = tool.extra.start || [OWOP.mouse.tileX, OWOP.mouse.tileY];
var e = tool.extra.end || [OWOP.mouse.tileX + 1, OWOP.mouse.tileY + 7];
var x = (s[0] - camera.x) * camera.zoom + 0.5;
var y = (s[1] - camera.y) * camera.zoom + 0.5;
var w = e[0] - s[0];
var h = e[1] - s[1];
ctx.beginPath();
ctx.rect(x, y, w * camera.zoom, h * camera.zoom);
// ctx.globalAlpha = 1;
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
let arr = new Array(Math.floor(Math.random() * 15));
for (let e in arr) {
arr[e] = Math.floor(Math.random() * 15);
}
ctx.setLineDash(arr);
ctx.strokeStyle = "#000000";
ctx.stroke();
if (Math.floor(time / 750) % 2) {
ctx.beginPath();
ctx.moveTo(x + 0.30 * camera.zoom, y + 0.5 * camera.zoom);
ctx.lineTo(x + 0.30 * camera.zoom, y + 6.5 * camera.zoom);
ctx.stroke();
}
// if (tool.extra.textArea) console.log(tool.extra.textArea.value);
// if () {
// }
// for (let p9 = 0; p9 < 182; p9++) {
// for (let pp = 0; pp < 14; pp++) {
// let pD = (p9 * 14 + pp);
// let color = [p8[pD + 0], p8[pD + 1], p8[pD + 2], p8[pD + 3]];
// if (p8[pD+3] === 0) arr.push(2);
// else if (p8[pD + 0] == 0) arr.push(1);
// else arr.push(0);
// }
// }
// ctx.setLineDash([3, 4]);
// ctx.strokeStyle = "#000000";
// ctx.stroke();
// ctx.globalAlpha = 0.25 + Math.sin(time / 500) / 4;
// ctx.fillStyle = renderer.patterns.unloaded;
// ctx.fill();
// ctx.setLineDash([]);
// var oldfont = ctx.font;
// ctx.font = "16px sans-serif";
// var txt = `${!tool.extra.clicking ? "Right click to screenshot " : ""}(${Math.abs(w)}x${Math.abs(h)})`;
// var txtx = window.innerWidth >> 1;
// var txty = window.innerHeight >> 1;
// txtx = Math.max(x, Math.min(txtx, x + w * camera.zoom));
// txty = Math.max(y, Math.min(txty, y + h * camera.zoom));
// drawText(ctx, txt, txtx, txty, true);
// ctx.font = oldfont;
ctx.lineWidth = oldlinew;
return 0;
});
tool.setEvent('mousedown', (mouse, event) => {
if (mouse.buttons === 1) {
tool.extra.start = [mouse.tileX, mouse.tileY];
tool.extra.end = [mouse.tileX + 1, mouse.tileY + 7];
tool.extra.textArea = windows["Text Input"].elements[0];
let _map = { "\n": "\n", " ": 26, a: 0, b: 1, c: 2, d: 3, e: 4, f: 5, g: 6, h: 7, i: 8, j: 9, k: 10, l: 11, m: 12, n: 13, o: 14, p: 15, q: 16, r: 17, s: 18, t: 19, u: 20, v: 21, w: 22, x: 23, y: 24, z: 25 };
tool.setEvent('keydown', (keys, event) => {
let t = tool.extra.textArea.value;
let row = 0;
let p6 = 0;
PM.startHistory();
for (let p5 = 0; p5 < t.length; p5++, p6++) {
let offsetx = _map[t[p5].toLowerCase()];
if (offsetx === "\n") {
row++;
p6 = -1;
continue;
}
if (!offsetx) {
// if (!isNaN(t[p5]))
}
for (let p9 = 0; p9 < 7; p9++) {
for (let pp = 0; pp < 7; pp++) {
let pD = (p9 * 189 + pp + offsetx * 7);
// console.log(tool.text[pD]);
// let color = [p8[pD + 0], p8[pD + 1], p8[pD + 2], p8[pD + 3]];
let c = [OWOP.player.selectedColor, undefined, undefined];
let pos = [...tool.extra.start];
pos[0] = pos[0] + pp + p6 * 7;
pos[1] = pos[1] + p9 + row * 7;
let color = c[tool.text[pD]];
if (color) PM.setPixel(...pos, color);
}
}
}
PM.endHistory();
return true;
});
tool.setEvent('keyup', () => true);
}
});
tool.setEvent('deselect', () => {
PM.endHistory();
// let t = tool.extra.textArea.value;
// for (let p5 = 0; p5 < t.length; p5++) {
// for (let p9 = 0; p9 < 7; p9++) {
// for (let pp = 0; pp < 7; pp++) {
// let pos = [...tool.extra.start];
// pos[0] = pos[0] + pp + p5 * 7;
// pos[1] = pos[1] + p9;
// //PM.setPixel(...pos, [255, 255, 255], true);
// }
// }
// }
tool.extra.position = 0;
tool.extra.start = undefined;
tool.extra.end = undefined;
tool.extra.textArea = undefined;
tool.setEvent('keyup', null);
tool.setEvent('keydown', null);
});
}));
OWOP.tool.updateToolbar();
})();
if (document.domain && !NS.OPM) {
let r = 0;
for (let e in OWOP.tool.allTools) {
e = OWOP.tool.allTools[e];
if (e.rankRequired < 2) r++;
}
document.getElementById("toole-container").style.maxWidth = 40 * Math.ceil(r / 8) + "px";
}
if (document.domain && NS.OPM) {
let element1 = document.createElement("span");
element1.className = "top-bar";
element1.style.float = "right";
element1.style.width = "34px";
element1.style.height = "17px";
element1.style.background = "#FFFFFF";
if (localStorage.rSC) {
let arr = JSON.parse(localStorage.rSC);
element1.style.background = `#${arr[0].toString(16).padStart(2, "0")}${arr[1].toString(16).padStart(2, "0")}${arr[2].toString(16).padStart(2, "0")}`;
}
let element2 = document.createElement("span");
element2.className = "top-bar";
element2.style.float = "right";
element2.textContent = "Right Color:";
let element3 = document.createElement("span");
element3.className = "top-bar";
element3.style.float = "right";
element3.textContent = "Your ID: null";
OWOP.elements.topBar.appendChild(element1);
OWOP.elements.topBar.appendChild(element2);
OWOP.elements.topBar.appendChild(element3);
document.getElementById("playercount-display").style.marginRight = "45px";
setInterval(() => {
let arr = OWOP.player.rightSelectedColor;
element1.style.background = rgb(...arr);
element3.textContent = `Your ID:${OWOP.player.id}`;
}, 10);
// teleport detector
OWOP.playerList = {};
function tick() {
let players = OWOP.require("main").playerList;
let playersFixed = {};
playersFixed[OWOP.player.id] = {
id: OWOP.player.id,
x: OWOP.mouse.tileX,
y: OWOP.mouse.tileY
};
for (let player in players) {
let n = players[player].childNodes;
playersFixed[n[0].innerHTML] = {
id: ~~n[0].innerHTML,
x: ~~n[1].innerHTML,
y: ~~n[2].innerHTML
}
}
players = playersFixed;
// check if the local copy has a disconnected player
for (let p1 in OWOP.playerList) {
let test = false;
for (let p2 in players) {
if (p1 === p2) {
test = true;
break;
}
}
if (!test) {
delete OWOP.playerList[p1];
//OWOP.chat.local(`${p1} has left.`);
}
}
// check if the main copy has new players
for (let p1 in players) {
let test = false;
for (let p2 in OWOP.playerList) {
if (p1 === p2) {
test = true;
break;
}
}
if (!test) {
let p = players[p1];
OWOP.playerList[p.id] = {
id: p.id,
x: p.x,
y: p.y
}
//OWOP.chat.local(`${p1} has joined.`);
}
}
// check for a teleport
var banlist = [];
for (let player in players) {
let p1 = OWOP.playerList[player];
let p2 = players[player];
if (Math.sqrt((p1.x - p2.x) ** 2 + (p1.y - p2.y) ** 2) > 2000) {
//console.log("someone teleported", p2.id, p2.x, p2.y);
p1.tp = !isNaN(p1.tp) ? p1.tp + 1 : 1;
//if (!p1.ban && p1.tp < 10) OWOP.chat.local(`${player} Teleported from ${p1.x} ${p1.y} to ${p2.x} ${p2.y}`);
}
p1.x = p2.x;
p1.y = p2.y;
}
return players;
}
setInterval(tick, 10);
setInterval(() => {
for (let player in OWOP.playerList) {
let p1 = OWOP.playerList[player];
if (p1.tp >= 10) p1.ban = true;
p1.tp = 0;
}
}, 10 * 1000);
// it gets in the way of reading chat, im not trying to be mean to arc.
{ let e = document.querySelector("div[id='arc-widget-container']"); e ? e.parentElement.removeChild(e) : void 0; }
if (NS.et) {
clearInterval(OWOP.misc.tickInterval);
OWOP.misc.tickIntervalNS = setInterval(O, 1e3 / OWOP.options.tickSpeed);
function O() {
// its already on the normal interval thing, ill delete it when i understand it more
var t = ++OWOP.misc.tick;
var e = Math.max(Math.min(OWOP.options.movementSpeed, 64), 0);
var n = 0;
var r = 0;
(NS.keysdown[38] || (NS.keysdown[87] && !NS.keysdown[16])) && (r -= e);
(NS.keysdown[37] || (NS.keysdown[65] && !NS.keysdown[16])) && (n -= e);
(NS.keysdown[40] || (NS.keysdown[83] && !NS.keysdown[16])) && (r += e);
(NS.keysdown[39] || (NS.keysdown[68] && !NS.keysdown[16])) && (n += e);
if (0 !== n || 0 !== r) {
OWOP.require("canvas_renderer").moveCameraBy(n, r);
A(null, "mousemove", OWOP.mouse.x, OWOP.mouse.y);
}
OWOP.eventSys.emit(OWOP.events.tick, t);
if (null !== OWOP.player.tool && null !== OWOP.misc.world) OWOP.player.tool.call("tick");
}
function A(t, e, n, r) {
OWOP.mouse.x = n;
OWOP.mouse.y = r;
let o = 0;
if (null !== OWOP.misc.world) OWOP.mouse.validTile = OWOP.misc.world.validMousePos(OWOP.mouse.tileX, OWOP.mouse.tileY);
if (null !== OWOP.player.tool) o = OWOP.player.tool.call(e, [OWOP.mouse, t]);
if (F(OWOP.mouse.tileX, OWOP.mouse.tileY)) 1;//f.updateClientFx();
return o;
}
function F(t, e) {
return (OWOP.misc.lastXYDisplay[0] !== t || OWOP.misc.lastXYDisplay[1] !== e) && (OWOP.misc.lastXYDisplay = [t, e],
OWOP.options.hexCoords && (t = (t < 0 ? "-" : "") + "0x" + Math.abs(t).toString(16),
e = (e < 0 ? "-" : "") + "0x" + Math.abs(e).toString(16)),
OWOP.elements.xyDisplay.innerHTML = "X: " + t + ", Y: " + e,
!0)
}
}
}
// window setup
{
// palette saver
(function () {
let windowName = "Palette Saver";
let options = {
closeable: false
}
let paletteJson = {}
function windowFunc(thisWindow) {
var divwindow = document.createElement("div");
divwindow.style = "width: 300px; overflow-y: scroll; overflow-x: scroll; max-height: 165px;"
divwindow.innerHTML = `<input id="pName" type="text" style="max-width: 100px; border: 0px;" placeholder="Name"></input>
<button id="addPalette" >Save Current Palette</button> <table id="paletteTable" style="overflow-x: hidden; overflow-y: scroll;"></table>`;
thisWindow.addObj(divwindow);
}
var windowClass = OWOP.windowSys.addWindow(new OWOP.windowSys.class.window(windowName, options, windowFunc))
.move(window.innerWidth / 3, window.innerHeight / 3);
windows[windowName] = windowClass;
NS.windows[windowName].frame.style.visibility = "hidden";
var pName = document.getElementById("pName");
pName.oninput = () => {
if (pName.value.length > 25) pName.style.backgroundColor = "rgb(255 148 129)";
else pName.style.backgroundColor = "rgb(255, 255, 255)";
}
document.getElementById("addPalette").onclick = () => {
if (pName.value.length > 25) return alert("Your max name length is 25 characters.");
if (pName.value.length == 0) return alert("Invalid Name");
if (!localStorage.getItem("paletteJson")) {
paletteJson[pName.value] = OWOP.player.palette;
localStorage.setItem("paletteJson", JSON.stringify(paletteJson));
} else {
paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
if (paletteJson[pName.value]) {
pName.value = "";
return alert("You already have a palette with this name.");
}
paletteJson[pName.value] = OWOP.player.palette;
localStorage.setItem("paletteJson", JSON.stringify(paletteJson));
}
var divPalette = document.createElement("tr");
let pN = pName.value;
divPalette.id = `im-busy${pN}`;
divPalette.innerHTML = `<td id="palette-${pN}" style="cursor: pointer; padding: 5px; border: 1px solid white; border-radius: 5px; color: white;">${pN}</td> <td id="useT1-${pN}"><button id="useB1-${pN}">Use</button></td> <td id="useT2-${pN}"><button id="useB2-${pN}">Replace</button></td> <td id="useT3-${pN}"><button id="useB3-${pN}">Delete</button></td>`;
document.getElementById("paletteTable").appendChild(divPalette);
document.getElementById(`useB1-${pN}`).onclick = () => {
let paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
OWOP.player.palette.splice(0);
OWOP.player.palette.push(...paletteJson[pN]);
OWOP.player.paletteIndex = OWOP.player.paletteIndex;
}
document.getElementById(`useB2-${pN}`).onclick = () => {
if (!confirm(`Are you sure you want to REPLACE the palette ${pN}?`)) return;
let paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
paletteJson[`${pN}`] = OWOP.player.palette;
localStorage.setItem('paletteJson', JSON.stringify(paletteJson));
}
document.getElementById(`useB3-${pN}`).onclick = () => {
if (!confirm(`Are you sure you want to DELETE the palette ${pN}?`)) return;
let paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
document.getElementById(`palette-${pN}`).outerHTML = '';
document.getElementById(`im-busy${pN}`).outerHTML = '';
delete paletteJson[pN];
localStorage.setItem('paletteJson', JSON.stringify(paletteJson));
}
pName.style.backgroundColor = "rgb(255 255 255)";
pName.value = "";
}
if (localStorage.getItem("paletteJson")) {
var gettedJson = JSON.parse(localStorage.getItem("paletteJson"));
var obj = Object.keys(gettedJson);
for (var i = 0; i < obj.length; i++) {
let pN = obj[i];
var divPalette = document.createElement("tr");
divPalette.id = `im-busy${pN}`;
divPalette.innerHTML = `<td id="palette-${pN}" style="cursor: pointer; padding: 5px; border: 1px solid white; border-radius: 5px; color: white;">${pN}</td> <td id="useT1-${pN}"><button id="useB1-${pN}">Use</button></td> <td id="useT2-${pN}"><button id="useB2-${pN}">Replace</button></td> <td id="useT3-${pN}"><button id="useB3-${pN}">Delete</button></td>`;
document.getElementById("paletteTable").appendChild(divPalette);
document.getElementById(`useB1-${pN}`).onclick = () => {
let paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
OWOP.player.palette.splice(0);
OWOP.player.palette.push(...paletteJson[`${pN}`]);
OWOP.player.paletteIndex = OWOP.player.paletteIndex;
}
document.getElementById(`useB2-${pN}`).onclick = () => {
if (!confirm(`Are you sure you want to REPLACE the palette ${pN}?`)) return;
let paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
paletteJson[`${pN}`] = OWOP.player.palette;
localStorage.setItem('paletteJson', JSON.stringify(paletteJson));
}
document.getElementById(`useB3-${pN}`).onclick = () => {
if (!confirm(`Are you sure you want to DELETE the palette ${pN}?`)) return;
let paletteJson = JSON.parse(localStorage.getItem("paletteJson"));
document.getElementById(`palette-${pN}`).outerHTML = '';
document.getElementById(`im-busy${pN}`).outerHTML = '';
delete paletteJson[pN];
localStorage.setItem('paletteJson', JSON.stringify(paletteJson));
}
}
}
})();
// text input
(function () {
let windowName = "Text Input";
let options = {
closeable: false
}
function windowFunc(thisWindow) {
let textArea = document.createElement('textarea');
textArea.id = 'text-tool-input';
textArea.width = '500px';
textArea.hight = '500px';
textArea.value = "";
// let newTextArea = document.createElement('div');
// newTextArea.id = 'text-tool-input';
// newTextArea.width = '500px';
// newTextArea.height = '500px';
// newTextArea.innerHTML = "hi";
// newTextArea.contentEditable = "true";
textArea.onkeydown = () => {
}
thisWindow.addObj(textArea);
}
var windowClass = OWOP.windowSys.addWindow(new OWOP.windowSys.class.window(windowName, options, windowFunc)
.move(window.innerWidth / 3, window.innerHeight / 3));
windows[windowName] = windowClass;
NS.windows[windowName].frame.style.visibility = "hidden";
})();
// assets
(function () {
let windowName = "Assets";
let options = {
closeable: false
}
var mkHTML = OWOP.util.mkHTML;
let G = r => document.getElementById(r);
function windowFunc(thisWindow) {
thisWindow.frame.style.width = "500px";
let innerFrame = document.createElement("div");
let realAssetsCont = mkHTML("div", {
id: "real-assets-cont"
});
let p = mkHTML("p");
p.style["margin-block"] = "auto";
p.style["display"] = "flex";
p.style["justify-content"] = "space-evenly";
let button1 = mkHTML("button", {
id: "NSoptions",
innerHTML: "Add"
});
let button2 = mkHTML("button", {
id: "NSoptions",
innerHTML: "Paste"
});
let button3 = mkHTML("button", {
id: "NSoptions",
innerHTML: "Delete"
});
let button4 = mkHTML("button", {
id: "NSoptions",
innerHTML: "Reload"
});
button1.onclick = async () => {
OWOP.sounds.play(OWOP.sounds.click);
let pE = localStorage.MB_Assets;
if (!pE) pE = [];
else pE = JSON.parse(pE);
var _imgTotal = 0;
{
var _lsTotal = 0,
_xLen, _x;
for (_x in localStorage) {
if (!localStorage.hasOwnProperty(_x)) {
continue;
}
_xLen = ((localStorage[_x].length + _x.length) * 2);
_lsTotal += _xLen;
};
//console.log("Total = " + (_lsTotal / 1024).toFixed(2) + " KB");
if ((_lsTotal / 1024) > 3000) return OWOP.chat.local(`Storage limit reached (3MB), remove images to add more.`);
_imgTotal = _lsTotal;
}
{
var _imageData = await X('image/*');
var _lsTotal = 0, _x;
_x = JSON.stringify(_imageData);
_lsTotal = _x.length * 2;
//console.log("Total = " + (_lsTotal / 1024).toFixed(2) + " KB");
if ((_lsTotal / 1024) > 500) {
if (((_lsTotal + _imgTotal) / 1024) > 3000) return OWOP.chat.local(`Image being added is more than Storage limit (3KB)`);
if (!confirm(`Are you sure you want to add a image with ${(_lsTotal / 1024).toFixed(2)} KB`)) return;
}
}
pE.push(_imageData);
localStorage.MB_Assets = JSON.stringify(pE);
J();
};
button2.onclick = () => {
OWOP.sounds.play(OWOP.sounds.click);
var img = new Image();
img.onload = () => {
OWOP.tool.allTools.paste.extra.k = img;
OWOP.player.tool = "move";
OWOP.player.tool = "paste";
}
img.src = NS.selectedAsset;
};
button3.onclick = () => {
OWOP.sounds.play(OWOP.sounds.click);
if (!NS.selectedAssetIndex) return;
if (confirm("Do you want to delete the selected asset?")) NS.assets.splice(NS.selectedAssetIndex, 1);
else return;
localStorage.MB_Assets = JSON.stringify(NS.assets);
J();
};
button4.onclick = () => {
OWOP.sounds.play(OWOP.sounds.click);
J();
};
let J = () => {
NS.assets = localStorage.MB_Assets;
if (!NS.assets) NS.assets = [];
else NS.assets = JSON.parse(NS.assets);
let y = G("real-assets-cont");
y.innerHTML = '';
for (let p0 in NS.assets) {
let p1 = new Image();
p1.onload = () => {
p1.style.width = '48px';
p1.style.height = '48px';
p1.style.border = 'solid 1px';
p1.onclick = () => {
for (let p4 in G('real-assets-cont').children) {
if (typeof G('real-assets-cont').children[p4] !== 'object') break;
G('real-assets-cont').children[p4].style.border = 'solid 1px';
}
if (NS.selectedImg) {
NS.selectedImg.style.width = '48px';
NS.selectedImg.style.height = '48px';
}
NS.selectedAsset = NS.assets[p0];
NS.selectedAssetIndex = p0;
NS.selectedImg = p1;
p1.style.width = '40px';
p1.style.height = '40px';
p1.style.border = 'solid 5px black';
};
p1.oncontextmenu = p3 => {
p3.preventDefault();
NS.assets.splice(p0, 1);
localStorage.MB_Assets = JSON.stringify(NS.assets);
J();
};
y.append(p1);
};
p1.src = NS.assets[p0];
}
};
let X = (r = '*') => new Promise(Q => {
let c = document.createElement('input');
c.type = 'file';
c.accept = r;
c.onchange = () => {
let N = new FileReader();
N.onloadend = () => {
Q(N.result);
};
N.readAsDataURL(c.files[0]);
};
c.onclick = () => void 0;
c.click();
});
button2.addEventListener("click", function () {
OWOP.sounds.play(OWOP.sounds.click)
});
p.appendChild(button1);
p.appendChild(button2);
p.appendChild(button3);
p.appendChild(button4);
innerFrame.appendChild(p);
innerFrame.appendChild(realAssetsCont);
thisWindow.addObj(innerFrame);
}
var windowClass = OWOP.windowSys.addWindow(new OWOP.windowSys.class.window(windowName, options, windowFunc)
.move(window.innerWidth / 3, window.innerHeight / 3));
windows[windowName] = windowClass;
windowClass.frame.style.visibility = "hidden";
})();
// patterns
(function () {
return;
let windowName = "Patterns";
let options = {
closeable: false
}
function windowFunc(thisWindow) {
let content = `
<span style="display: flex;">
<style>
#div1, #div2, #div3 {
float: left;
width: 75px;
height: 75px;
margin: 10px;
border: 1px solid black;
}
</style>
<div style="display: flex;margin: 0px;border-radius: 5px;background-color: #7e635c;box-shadow: inset 3px 2px 0px 0px #4d313b;align-content: space-around;">
<span style="border: 10px #0000 solid;display: flex;flex-direction: column;">
<span style="display: flex;">
<div id="div1" ondrop="NS.drop(event)" ondragover="NS.allowDrop(event)">
<img src="https://opm.dimden.dev/client/img/gui.png" draggable="true" ondragstart="NS.drag(event)" id="drag1" width="75" height="75">
</div>
<div style="display: flex;flex-direction: column;justify-content: center;">
<button class="tablinks1">
Filter Type:\nOR
</button>
<button class="tablinks1">
Flip
</button>
</div>
</span>
<span style="display: flex;">
<div id="div2" ondrop="NS.drop(event)" ondragover="NS.allowDrop(event)">
<img src="https://opm.dimden.dev/client/img/gui.png" draggable="true" ondragstart="NS.drag(event)" id="drag2" width="75" height="75">
</div>
<div style="display: flex;flex-direction: column;justify-content: center;">
<button class="tablinks1">
Filter Type:\nOR
</button>
<button class="tablinks1">
Flip
</button>
</div>
</span>
<span style="display: flex;">
<div id="div3" ondrop="NS.drop(event)" ondragover="NS.allowDrop(event)"></div>
<div style="display: flex;flex-direction: column;justify-content: center;">
<button class="tablinks1">
Filter Type:\nOR
</button>
<button class="tablinks1">
Flip
</button>
</div>
</span>
</span>
</div>
<div style="width: 5px;"></div>
<div style="display: flex;margin: 0px;border-radius: 5px;background-color: #7e635c;box-shadow: inset 3px 2px 0px 0px #4d313b;align-content: space-around;">
<span style="border: 10px #0000 solid;">
<div>something 2
</div>
</span>
</div>
</span>
`;
let container = thisWindow.container;
container.style = "margin: 0px -5px -5px -5px;";
container.className = "optionsDiv";
container.innerHTML = content;
function allowDrop(ev) {
ev.preventDefault();
}
function drag(ev) {
ev.dataTransfer.setData("text", ev.target.id);
}
function drop(ev) {
ev.preventDefault();
var data = ev.dataTransfer.getData("text");
ev.target.appendChild(document.getElementById(data));
}
NS.allowDrop = allowDrop;
NS.drag = drag;
NS.drop = drop;
return;
const root = [...container.childNodes].find(e => e.nodeType !== Node.TEXT_NODE);
let mkHTML = OWOP.util.mkHTML;
let display = root.querySelector("#display");
for (let w in OWOP.windowSys.windows) {
w = OWOP.windowSys.windows[w];
if (w.title === "Options" || w.title === "Resulting image") continue;
if (w.title !== "Tools") {
w.move(window.innerWidth / 3, window.innerHeight / 3);
w.frame.style.visibility = "hidden";
let b = w.frame.querySelectorAll(".windowCloseButton");
if (b.length) w.frame.removeChild(b[0]);
}
let v = w.frame.style;
let current = mkHTML("p");
current.className = "tabp";
current.appendChild(mkHTML("p", { innerHTML: w.title }));
let button = mkHTML("button");
current.appendChild(button);
let isVisible = v.visibility === "visible" || v.visibility === "" ? true : false;
button.id = button.innerHTML = isVisible ? "on" : "off";
button.onclick = function () {
isVisible = !isVisible;
button.id = button.innerHTML = isVisible ? "on" : "off";
v.visibility = isVisible ? "visible" : "hidden";
}
display.appendChild(current);
}
let options = root.querySelector("#options");
if (window) window.openCity = function (evt, cityName, s, button) {
var i, tabcontent, tablinks;
tabcontent = document.getElementsByClassName("tabcontent" + s);
for (i = 0; i < tabcontent.length; i++) {
tabcontent[i].style.display = "none";
}
tablinks = document.getElementsByClassName(button.className);
for (i = 0; i < tablinks.length; i++) {
tablinks[i].id = "";
}
button.id = "on";
document.getElementById(cityName).style.display = "block";
}
NS.optionbutton = function (button) {
let parent1 = button.parentElement;
let parent2 = parent1.parentElement;
let s = button.id === "on" ? false : true;
button.innerHTML = button.id = s ? "on" : "off";
if (parent2.className === "tabcontentright") OWOP.tool.allTools[`${parent2.id}`].extra.state[`${parent1.id}`] = s;
}
}
var windowClass = OWOP.windowSys.addWindow(new OWOP.windowSys.class.window(windowName, options, windowFunc)
.move(window.innerWidth / 3, window.innerHeight / 3));
windows[windowName] = windowClass;
windowClass.frame.style.visibility = "hidden";
})();
function makeOptionsWindow() {
if (OWOP.windowSys.windows['Options']) OWOP.windowSys.delWindow(OWOP.windowSys.windows['Options']);
let windowName = "Options";
let options = {
closeable: false
}
function windowFunc(thisWindow) {
let content = `
<span>
<span id="optionsMinimize" style="display: flex;">
<style>
p {
margin-block: auto;
text-align: center;
color: white;
font-family: Arial;
}
.tabcontentleft, .tabcontentright {
display: none;
margin-block: auto;
text-align: center;
}
.tabp {
display: flex;
justify-content: space-between;
margin: 1px 0px;
}
.tabButton, .optionButton {
border-style: none !important;
border-image: none !important;
border: initial;
border-radius: 6px;
padding: 5px 8px;
}
.optionButton {
padding: 5px 12px;
}
button.on {
background: #9a937b;
}
</style>
<div style="display: flex;margin: 0px;border-radius: 5px;background-color: #7e635c;box-shadow: inset 3px 2px 0px 0px #4d313b;align-content: space-around;">
<span style="border: 10px #0000 solid;">
<div class="tab">
<button class="tabButton olt on" onclick="NS.switchTabs(event, 'display', 'olt', this)">
Window Display
</button>
<button class="tabButton olt" onclick="NS.switchTabs(event, 'options', 'olt', this)">
Options
</button>
<div id="display" class="tabcontentolt" style="display: block;"></div>
<div id="options" class="tabcontentolt" style="display: none;"></div>
</div>
</span>
</div>
<div style="width: 5px;"></div>
<div style="display: flex;margin: 0px;border-radius: 5px;background-color: #7e635c;box-shadow: inset 3px 2px 0px 0px #4d313b;align-content: space-around;">
<span style="border: 10px #0000 solid;">
<div class="tab">
<div style="align-content: center;">
<button class="tabButton ort on" onclick="NS.switchTabs(event, 'cursor', 'ort', this)">
Cursor
</button>
<button class="tabButton ort" onclick="NS.switchTabs(event, 'line', 'ort', this)">
Line
</button>
<button class="tabButton ort" onclick="NS.switchTabs(event, 'fill', 'ort', this)">
Bucket
</button>
<button class="tabButton ort" onclick="NS.switchTabs(event, 'export', 'ort', this)">
Select
</button>
</div>
<div id="cursor" class="tabcontentort" style="display: block;">
<div id="scalar" class="tabp">
<p>Scale</p>
<input type="range" style="margin: 0px 15px;" value="1" min="1" max="16" oninput="OWOP.tool.allTools.cursor.extra.state.scalar = this.value;document.getElementById('cursorspan').innerHTML = this.value;"></input>
<span style="padding: 4px 0px 5px 0px;" id="cursorspan">1</span>
</div>
<div id="chunkize" class="tabp">
<p>Chunkize</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="rainbow" class="tabp">
<p>Rainbow</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="perfect" class="tabp">
<p>Pixel Perfect</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
</div>
<div id="line" class="tabcontentort" style="display: none;">
<div id="rainbow" class="tabp">
<p>Rainbow</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="gradient" class="tabp">
<p>Gradient</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
</div>
<div id="fill" class="tabcontentort" style="display: none;">
<div id="rainbow" class="tabp">
<p>Rainbow</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="checkered" class="tabp">
<p>Checkered</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<!--
<div id="dither" class="tabp">
<p>Pattern</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
-->
<div id="dither2" class="tabp">
<p>Pattern 1</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="dither3" class="tabp">
<p>Pattern 2</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<!--
<div id="dither4" class="tabp">
<p>Pattern 4</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
-->
<div id="dither5" class="tabp">
<p>Pattern 3</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="dither6" class="tabp">
<p>Pattern 4</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<!--
<div id="dither7" class="tabp">
<p>Pattern 7</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
-->
</div>
<div id="export" class="tabcontentort" style="display: none;">
<div class="tabp">
<p>Type</p>
<select style="padding: 3px 0px;background: #aba389;" class="exportselect" oninput="OWOP.tool.allTools.export.extra.state.type=this.value">
<option value="export">Export</option>
<option value="color">Palette Color Adder</option>
<!--<option value="adder">Queue Adder</option>-->
<option value="filler">Queue Filler</option>
<!--<option value="clearer">Queue Clearer</option>-->
</select>
</div>
<div id="chunkize" class="tabp">
<p>Chunkize</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
<div id="rainbow" class="tabp">
<p>Rainbow</p>
<button class="optionButton" onclick="NS.optionbutton(this)">off</button>
</div>
</div>
</div>
</span>
</div>
</span>
<span id="optionsMaximize" style="display: none;">
<div style="display: flex;margin: 0px;border-radius: 5px;background-color: #7e635c;box-shadow: inset 3px 2px 0px 0px #4d313b;align-content: space-around;">
<span style="border: 10px #0000 solid;">
<button class="optionButton" onclick="NS.minimizeOptions(false)">Maximize</button>
</span>
</div>
</span>
</span>
`;
let container = thisWindow.container;
container.style = "margin: 0px -5px -5px -5px;";
container.className = "optionsDiv";
container.innerHTML = content;
const root = [...container.childNodes].find(e => e.nodeType !== Node.TEXT_NODE);
let mkHTML = OWOP.util.mkHTML;
let display = root.querySelector("#display");
for (let w in OWOP.windowSys.windows) {
w = OWOP.windowSys.windows[w];
if (w.title === "Options" || w.title === "Resulting image") continue;
if (w.title !== "Tools") {
w.move(window.innerWidth / 3, window.innerHeight / 3);
w.frame.style.visibility = "hidden";
let b = w.frame.querySelectorAll(".windowCloseButton");
if (b.length) w.frame.removeChild(b[0]);
}
let v = w.frame.style;
let current = mkHTML("p");
current.className = "tabp";
current.appendChild(mkHTML("p", { innerHTML: w.title }));
let button = mkHTML("button");
button.classList.add("optionButton");
current.appendChild(button);
v.visibility === "visible" || v.visibility === "" ? button.classList.toggle("on") : void 0;
button.innerHTML = button.classList.contains("on") ? "on" : "off";
button.onclick = function () {
button.classList.toggle("on");
let s = button.classList.contains("on");
button.innerHTML = s ? "on" : "off";
v.visibility = s ? "visible" : "hidden";
}
display.appendChild(current);
}
let options = root.querySelector("#options");
function optionmaker(name, inputType, checked, onclick) {
let current = mkHTML("div");
current.className = "tabp";
current.appendChild(mkHTML("p", { innerHTML: name }));
let button = mkHTML("button");
button.classList.add("optionButton");
current.appendChild(button);
if (inputType === "button") button.innerHTML = checked ? "on" : "off";
else if (inputType === "select") button.innerHTML = " ";
button.onclick = function () {
if (inputType === "button") {
button.classList.toggle("on");
button.innerHTML = button.classList.contains("on") ? "on" : "off";
}
onclick();
}
if (checked) button.classList.toggle("on");
return current;
}
options.appendChild(optionmaker("Disable PM", "button", NS.PM.disabled, () => NS.PM.disabled = !NS.PM.disabled));
options.appendChild(optionmaker("Clear PM", "select", void 0, () => NS.PM.clearQueue()));
//options.appendChild(optionmaker("Chunkize", "button", NS.chunkize, () => NS.chunkize = !NS.chunkize));
options.appendChild(optionmaker("Mute", "button", !OWOP.options.enableSounds, () => { OWOP.options.enableSounds = !OWOP.options.enableSounds; localStorage.setItem("options", JSON.stringify({ enableSounds: OWOP.options.enableSounds })) }));
options.appendChild(optionmaker("Undo", "select", void 0, () => PM.undo()));
options.appendChild(optionmaker("Redo", "select", void 0, () => PM.redo()));
options.appendChild(optionmaker("Minimize Options", "select", void 0, () => NS.minimizeOptions(true)));
NS.switchTabs = function (evt, cityName, s, button) {
var i, tabcontent, tablinks;
tabcontent = document.getElementsByClassName("tabcontent" + s);
for (i = 0; i < tabcontent.length; i++) {
tabcontent[i].style.display = "none";
}
tablinks = document.getElementsByClassName(s);
for (i = 0; i < tablinks.length; i++) {
tablinks[i].classList.remove("on");
}
button.classList.add("on");
document.getElementById(cityName).style.display = "block";
}
NS.optionbutton = function (button) {
let parent1 = button.parentElement;
let parent2 = parent1.parentElement;
button.classList.toggle("on");
let s = button.classList.contains("on");
button.innerHTML = s ? "on" : "off";
if (parent2.className === "tabcontentort") OWOP.tool.allTools[`${parent2.id}`].extra.state[`${parent1.id}`] = s;
}
NS.minimizeOptions = function (minimize) {
let max = document.getElementById("optionsMaximize");
let min = document.getElementById("optionsMinimize");
if (minimize) {
max.style = "display: flex;";
min.style = "display: none;";
} else {
max.style = "display: none;";
min.style = "display: flex;";
}
}
}
var windowClass = OWOP.windowSys.addWindow(new OWOP.windowSys.class.window(windowName, options, windowFunc)
.move(OWOP.windowSys.windows['Tools'].realw + 15, 32));
windows[windowName] = windowClass;
windowClass.frame.style.visibility = "visible";
}
makeOptionsWindow();
{
function x() {
if ((NS.OPM && OPM?.user?.installed?.length) && ((!OPM?.user?.installed?.includes("player-list") || OWOP?.windowSys?.windows?.['Players']) && (!OPM?.user?.installed?.includes("coords-saver") || OWOP?.windowSys?.windows?.['Coordinates Saver']))) makeOptionsWindow();
else setTimeout(x, 1000);
}
x();
}
}
function hue(d, b = 1) {
let a = 256 / b; // 1 2 4 8 16 32 64 128 256
//let b = mul; // 256 128 64 32 16 8 4 2 1
d = Math.floor(d);
// d = d % (b * 6); m_{F}\left(a,b\right)=a-b\operatorname{floor}\left(\frac{a}{b}\right)
d = (Math.abs(Math.floor(d / (b * 6)) * ((b * 6))) + (d % (b * 6))) % (b * 6);
// d = d - (b * 6) * ~~(d/(b * 6));
let nD = Math.floor(Math.abs(d / b));
let output;
if (nD < 1) {
output = [255, 0, (d % b) * a];
} else if (nD < 2) {
output = [255 - ((d % b) * a), 0, 255];
} else if (nD < 3) {
output = [0, (d % b) * a, 255];
} else if (nD < 4) {
output = [0, 255, 255 - ((d % b) * a)];
} else if (nD < 5) {
output = [(d % b) * a, 255, 0];
} else if (nD < 6) {
output = [255, 255 - ((d % b) * a), 0];
}
// console.log(d);
// console.log(nD);
// console.log(output);
return output;
}
var clamp = v => Math.round(Math.max(Math.min(v, 255), 0));
var degToRad = d => d * (Math.PI / 180);
var radToDeg = r => r / (Math.PI / 180);
class RGBRotate {
constructor(degrees) {
this.matrix = [[1, 0, 0], [0, 1, 0], [0, 0, 1]];
this.set_hue_rotation(degrees);
}
set_hue_rotation(degrees) {
let cosA = Math.cos(degToRad(degrees));
let sinA = Math.sin(degToRad(degrees));
this.matrix[0][0] = cosA + (1 - cosA) / 3;
this.matrix[0][1] = 1 / 3 * (1 - cosA) - Math.sqrt(1 / 3) * sinA;
this.matrix[0][2] = 1 / 3 * (1 - cosA) + Math.sqrt(1 / 3) * sinA;
this.matrix[1][0] = 1 / 3 * (1 - cosA) + Math.sqrt(1 / 3) * sinA;
this.matrix[1][1] = cosA + 1 / 3 * (1 - cosA);
this.matrix[1][2] = 1 / 3 * (1 - cosA) - Math.sqrt(1 / 3) * sinA;
this.matrix[2][0] = 1 / 3 * (1 - cosA) - Math.sqrt(1 / 3) * sinA;
this.matrix[2][1] = 1 / 3 * (1 - cosA) + Math.sqrt(1 / 3) * sinA;
this.matrix[2][2] = cosA + 1 / 3 * (1 - cosA);
}
apply(r, g, b) {
let rx = r * this.matrix[0][0] + g * this.matrix[0][1] + b * this.matrix[0][2];
let gx = r * this.matrix[1][0] + g * this.matrix[1][1] + b * this.matrix[1][2];
let bx = r * this.matrix[2][0] + g * this.matrix[2][1] + b * this.matrix[2][2];
return [clamp(rx), clamp(gx), clamp(bx)];
}
}
function rgb(r, g, b) {
return "#" + [r, g, b].map(v => {
return ('0' +
Math.min(Math.max(parseInt(v), 0), 255)
.toString(16)
).slice(-2);
}).join('');
}
const Base64 = {
// sourced from https://stackoverflow.com/questions/6213227/fastest-way-to-convert-a-number-to-radix-64-in-javascript
// coded by https://stackoverflow.com/users/520997/reb-cabin
_Rixits:
// 0 8 16 24 32 40 48 56 63
// v v v v v v v v v
"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+/",
// You have the freedom, here, to choose the glyphs you want for
// representing your base-64 numbers. The ASCII encoding guys usually
// choose a set of glyphs beginning with ABCD..., but, looking at
// your update #2, I deduce that you want glyphs beginning with
// 0123..., which is a fine choice and aligns the first ten numbers
// in base 64 with the first ten numbers in decimal.
// This cannot handle negative numbers and only works on the
// integer part, discarding the fractional part.
// Doing better means deciding on whether you're just representing
// the subset of javascript numbers of twos-complement 32-bit integers
// or going with base-64 representations for the bit pattern of the
// underlying IEEE floating-point number, or representing the mantissae
// and exponents separately, or some other possibility. For now, bail
fromBigInt: function (bigint) {
if (typeof bigint !== "bigint")
throw "The input is not valid";
var rixit; // like 'digit', only in some non-decimal radix
var residual = bigint;
var result = '';
while (true) {
rixit = residual % 64n;
result = this._Rixits.charAt(Number(rixit)) + result;
residual = residual / 64n;
if (residual == 0n)
break;
}
return result;
},
fromNumber: function (number) {
if (isNaN(Number(number)) || number === null ||
number === Number.POSITIVE_INFINITY)
throw "The input is not valid";
if (number < 0)
throw "Can't represent negative numbers now";
var rixit; // like 'digit', only in some non-decimal radix
var residual = Math.floor(number);
var result = '';
while (true) {
rixit = residual % 64;
result = this._Rixits.charAt(rixit) + result;
residual = Math.floor(residual / 64);
if (residual == 0)
break;
}
return result;
},
toNumber: function (rixits) {
var result = 0;
rixits = rixits.split('');
for (var e = 0; e < rixits.length; e++) {
result = (result * 64) + this._Rixits.indexOf(rixits[e]);
}
return result;
},
toBigInt: function (rixits) {
var result = 0n;
rixits = rixits.split('');
for (var e = 0; e < rixits.length; e++) {
result = (result * 64n) + BigInt(this._Rixits.indexOf(rixits[e]));
}
return result;
}
}
const Base256 = {
// sourced from https://stackoverflow.com/questions/6213227/fastest-way-to-convert-a-number-to-radix-64-in-javascript
// coded by https://stackoverflow.com/users/520997/reb-cabin
// modified by NekoNoka
_Rixits:
// 0 8 16 24 32 40 48 56 63
// v v v v v v v v v
"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyzÀÁÂÃÄÅÇÈÉÊËÌÍÎÏÐÑÒÓÔÕÖÙÚÛÜÝÞßàáâãäåçèéêëìíîïðñòóôõöùúûüýþÿĀāĂ㥹ĆćĈĉĊċČčĎďĐđĒēĔĕĖėĘęĚěĜĝĞğĠġĢģĤĥĦħĨĩĪīĬĭĮįİIJijĴĵĶķĸĹĺĻļĽľĿŀŁłŃńŅņŇňʼnŊŋŌōŎŏŐőŔŕŖŗŘřŚśŜŝŞşŠšŢţŤťŦŧŨũŪūŬŭŮůŰűŲųŴŵŶŷŸŹźŻżŽžǞǟǠǡǦǧǨǩǪǫǬǭ",
// You have the freedom, here, to choose the glyphs you want for
// representing your base-64 numbers. The ASCII encoding guys usually
// choose a set of glyphs beginning with ABCD..., but, looking at
// your update #2, I deduce that you want glyphs beginning with
// 0123..., which is a fine choice and aligns the first ten numbers
// in base 64 with the first ten numbers in decimal.
// This cannot handle negative numbers and only works on the
// integer part, discarding the fractional part.
// Doing better means deciding on whether you're just representing
// the subset of javascript numbers of twos-complement 32-bit integers
// or going with base-64 representations for the bit pattern of the
// underlying IEEE floating-point number, or representing the mantissae
// and exponents separately, or some other possibility. For now, bail
fromBigInt: function (bigint) {
if (typeof bigint !== "bigint")
throw "The input is not valid";
if (bigint < 0)
throw "Can't represent negative numbers now";
var rixit; // like 'digit', only in some non-decimal radix
var residual = bigint;
var result = '';
var l = BigInt(this._Rixits.length);
while (true) {
rixit = residual % l;
result = this._Rixits.charAt(Number(rixit)) + result;
residual = residual / l;
if (residual == 0n)
break;
}
return result;
},
fromNumber: function (number) {
if (isNaN(Number(number)) || number === null ||
number === Number.POSITIVE_INFINITY)
throw "The input is not valid";
if (number < 0)
throw "Can't represent negative numbers now";
var rixit; // like 'digit', only in some non-decimal radix
var residual = Math.floor(number);
var result = '';
while (true) {
rixit = residual % 256;
// console.log("rixit : " + rixit);
// console.log("result before : " + result);
result = this._Rixits.charAt(rixit) + result;
// console.log("result after : " + result);
// console.log("residual before : " + residual);
residual = Math.floor(residual / 256);
// console.log("residual after : " + residual);
if (residual == 0)
break;
}
return result;
},
toNumber: function (rixits) {
var result = 0;
// console.log("rixits : " + rixits);
// console.log("rixits.split('') : " + rixits.split(''));
rixits = rixits.split('');
for (var e = 0; e < rixits.length; e++) {
// console.log("_Rixits.indexOf(" + rixits[e] + ") : " +
// this._Rixits.indexOf(rixits[e]));
// console.log("result before : " + result);
result = (result * 256) + this._Rixits.indexOf(rixits[e]);
// console.log("result after : " + result);
}
return result;
},
toBigInt: function (rixits) {
var result = 0n;
rixits = rixits.split('');
for (var e = 0; e < rixits.length; e++) {
result = (result * 256n) + BigInt(this._Rixits.indexOf(rixits[e]));
}
return result;
}
}
{
setInterval(() => {
let k = document.getElementById("chat-messages").children;
for (let i = 0; i < k.length; i++) {
let t = k[i].innerHTML;
let id = OWOP.player.id;
var hasClass = t.classList !== undefined ? Array.from(t.classList).indexOf('nK') > -1 : false;
if (!t.match(`(\\[${id}\\]: )|(${id}: )`) && t.match(`${id}`) && !hasClass) k[i].style = "background: #FF404059;";
}
}, 100);
OWOP.windowSys.class.window.prototype.move = (function (t, e) { document.getElementById('windows').appendChild(this.frame); document.getElementById('windows').appendChild(OWOP.windowSys.windows.Tools.frame); return this.opt.immobile || (this.frame.style.transform = "translate(" + t + "px," + e + "px)", this.x = t, this.y = e), this });
Object.keys(OWOP.windowSys.windows).forEach(e => OWOP.windowSys.windows[e].move = (function (t, e) { document.getElementById('windows').appendChild(this.frame); document.getElementById('windows').appendChild(OWOP.windowSys.windows.Tools.frame); return this.opt.immobile || (this.frame.style.transform = "translate(" + t + "px," + e + "px)", this.x = t, this.y = e), this }));
const toolSetURL = ``
var cursors = {
set: new Image(),
// start of default cursors
cursor: { imgpos: [0, 0], hotspot: [0, 0] },
move: { imgpos: [1, 0], hotspot: [18, 18] },
pipette: { imgpos: [0, 1], hotspot: [0, 28] },
erase: { imgpos: [0, 2], hotspot: [4, 26] },
zoom: { imgpos: [1, 2], hotspot: [19, 10] },
fill: { imgpos: [1, 1], hotspot: [3, 29] },
brush: { imgpos: [0, 3], hotspot: [0, 26] },
export: { imgpos: [2, 0], hotspot: [0, 0] }, // needs better hotspot
selectprotect: { imgpos: [4, 0], hotspot: [0, 0] },
copy: { imgpos: [3, 0], hotspot: [0, 0] }, // and this
paste: { imgpos: [3, 1], hotspot: [0, 0] }, // this too
cut: { imgpos: [3, 2], hotspot: [11, 5] },
line: { imgpos: [3, 3], hotspot: [0, 0] },
shield: { imgpos: [2, 3], hotspot: [18, 18] },
kick: { imgpos: [2, 1], hotspot: [3, 6] },
ban: { imgpos: [3, 0], hotspot: [10, 4] },
write: { imgpos: [1, 3], hotspot: [10, 4] }, // fix hotspot
// end of the default cursors
// start of neko cursors
"neko eraser": { imgpos: [0, 2], hotspot: [4, 26] },
"neko foreign pixel replacer": { imgpos: [4, 1], hotspot: [4, 26] },
"pixel perfect": { imgpos: [0, 3], hotspot: [0, 26] },
"palette color adder": { imgpos: [0, 5], hotspot: [0, 0] },
"rainbow cursor": { imgpos: [4, 4], hotspot: [0, 26] },
"rainbow line": { imgpos: [4, 3], hotspot: [0, 0] },
"rainbow fill": { imgpos: [4, 2], hotspot: [3, 29] },
"checkered fill": { imgpos: [0, 4], hotspot: [3, 29] },
"gradient wand": { imgpos: [4, 5], hotspot: [0, 0] },
"queue adder": { imgpos: [1, 5], hotspot: [0, 0] },
"queue filler": { imgpos: [2, 5], hotspot: [0, 0] },
"queue clearer": { imgpos: [3, 5], hotspot: [0, 0] },
// end of the neko cursors
};
function reduce(canvas) { /* Removes unused space from the image */
var nw = canvas.width;
var nh = canvas.height;
var ctx = canvas.getContext('2d');
var idat = ctx.getImageData(0, 0, canvas.width, canvas.height);
var u32dat = new Uint32Array(idat.data.buffer);
var xoff = 0;
var yoff = 0;
for (var y = 0, x, i = 0; y < idat.height; y++) {
for (x = idat.width; x--; i += u32dat[y * idat.width + x]);
if (i) { break; }
yoff++;
}
for (var x = 0, y, i = 0; x < idat.width; x++) {
for (y = nh; y--; i += u32dat[y * idat.width + x]);
if (i) { break; }
xoff++;
}
for (var y = idat.height, x, i = 0; y--;) {
for (x = idat.width; x--; i += u32dat[y * idat.width + x]);
if (i) { break; }
nh--;
}
for (var x = idat.width, y, i = 0; x--;) {
for (y = nh; y--; i += u32dat[y * idat.width + x]);
if (i) { break; }
nw--;
}
canvas.width = nw;
canvas.height = nh;
ctx.putImageData(idat, -xoff, -yoff);
}
function shadow(canvas, img) {
/* Make a bigger image so the shadow doesn't get cut */
canvas.width = 2 + img.width + 6;
canvas.height = 2 + img.height + 6;
var ctx = canvas.getContext('2d');
ctx.shadowColor = '#000000';
ctx.globalAlpha = 0.5; /* The shadow is too dark so we draw it transparent */
ctx.shadowBlur = 4;
ctx.shadowOffsetX = 2;
ctx.shadowOffsetY = 2;
ctx.drawImage(img, 2, 2);
ctx.globalAlpha = 1;
ctx.shadowColor = 'rgba(0, 0, 0, 0)'; /* disables the shadow */
ctx.drawImage(img, 2, 2);
}
/* makes a hole with the shape of the image */
function popOut(canvas, img) {
var shadowcolor = 0xFF3B314D;
var backgroundcolor = 0xFF5C637E;
canvas.width = img.width;
canvas.height = img.height;
var ctx = canvas.getContext('2d');
ctx.drawImage(img, 0, 0);
var idat = ctx.getImageData(0, 0, canvas.width, canvas.height);
var u32dat = new Uint32Array(idat.data.buffer);
var clr = (x, y) => {
return (x < 0 || y < 0 || x >= idat.width || y >= idat.height) ? 0
: u32dat[y * idat.width + x];
};
for (var i = u32dat.length; i--;) {
if (u32dat[i] !== 0) {
u32dat[i] = backgroundcolor;
}
}
for (var y = idat.height; y--;) {
for (var x = idat.width; x--;) {
if (clr(x, y) === backgroundcolor && (!clr(x, y - 1) || !clr(x - 1, y)) && !clr(x - 1, y - 1)) {
u32dat[y * idat.width + x] = shadowcolor;
}
}
}
for (var y = idat.height; y--;) {
for (var x = idat.width; x--;) {
if (clr(x, y - 1) === shadowcolor
&& clr(x - 1, y) === shadowcolor) {
u32dat[y * idat.width + x] = shadowcolor;
}
}
}
ctx.putImageData(idat, 0, 0);
}
var toolcss = (() => {
var style = document.createElement('style');
style.appendChild(document.createTextNode(''));
document.head.appendChild(style);
return style.sheet;
})();
function load() {
cursors.set.onload = () => {
var set = cursors.set;
var slotcanvas = document.createElement('canvas');
popOut(slotcanvas, set);
var j = Object.keys(cursors).length - 1 + 1; /* +1 slotset to blob url */
for (var tool in cursors) {
if (tool === 'set') continue;
cursors[tool].name = tool;
tool = cursors[tool];
var original = document.createElement('canvas');
var i = tool.img = {
shadowed: document.createElement('canvas'),
shadowblob: undefined
};
original.width = original.height = 36;
original.getContext('2d').drawImage(set,
tool.imgpos[0] * 36,
tool.imgpos[1] * 36,
36, 36, 0, 0, 36, 36
);
// reduce(original);
// shadow(i.shadowed, original);
// tool.hotspot[0] += 2;
// tool.hotspot[1] += 2; /* Check shadow() for explanation */
/* Blob-ify images */
original.toBlob(function (blob) {
this.img.shadowblob = URL.createObjectURL(blob);
toolcss.insertRule(`button[id='tool-${this.name}'] div {background-image: url(${this.img.shadowblob}) !important;background-position: 0 0 !important; background-repeat: no-repeat;}`);
}.bind(tool));
}
for (let tool in cursors) {
if (tool === "set") continue;
tool = cursors[tool];
var background = document.createElement('canvas');
background.width = background.height = 36;
background.getContext('2d').drawImage(slotcanvas,
tool.imgpos[0] * 36,
tool.imgpos[1] * 36,
36, 36, 0, 0, 36, 36
);
background.toBlob(blob => {
tool.imgbg = URL.createObjectURL(blob);
toolcss.insertRule(`button[id='tool-${tool.name}'].selected div {background-image: url(${tool.imgbg}) !important;background-position: 0 0 !important;}`);
});
}
}
cursors.set.src = toolSetURL;
}
load();
}
NS.hue = hue;
NS.rgb = rgb;
NS.Base64 = Base64;
NS.Base256 = Base256;
NS.clamp = clamp;
NS.degToRad = degToRad;
NS.radToDeg = radToDeg;
NS.RGBRotate = RGBRotate;
NS.chunkize = false;
if (window) window.NS = NS;
console.log("Neko's Scripts Loaded.");
console.timeEnd("Neko");
}
function init() {
let x = document.getElementById("load-scr");
if (x?.style?.transform) {
console.time("Neko");
console.log("Loading Neko's Scripts.");
NS.keysdown = [];
NS.OPM = !!OWOP.misc;
NS.extra = {};
NS.extra.log = false;
function keydown(event) {
var e = event.which || event.keyCode;
if ("TEXTAREA" !== document.activeElement.tagName && "INPUT" !== document.activeElement.tagName) {
NS.keysdown[e] = !0;
var n = OWOP.player.tool;
if (undefined !== OWOP?.world && n?.isEventDefined("keydown") && n?.call("keydown", [NS.keysdown, event])) return !1;
switch (e) {
case 80:
OWOP.player.tool = "pipette";
break;
case 77:
case 81:
OWOP.player.tool = "move";
break;
case 79:
OWOP.player.tool = "cursor";
break;
case 70:
break;
case 69:
//OWOP.player.tool = "neko eraser";
break;
case 66:
//OWOP.player.tool = "fill";
break;
case 72:
// make options window open/close
// options window will include options to switch the behavior of the tools, the game, and open/close all windows
break;
case 71:
if (NS.OPM) OWOP.renderer.showGrid(!OWOP.renderer.gridShown);
break;
case 90:
if (!event.ctrlKey) break;
NS.PM.undo(event.shiftKey);
event.preventDefault();
break;
case 89:
if (!event.ctrlKey) break;
NS.PM.redo(event.shiftKey);
event.preventDefault();
break;
case 112: // f1
event.preventDefault();
break;
case 107:
case 187:
++OWOP.camera.zoom;
break;
case 109:
case 189:
--OWOP.camera.zoom;
break;
case 76:
NS.extra.log = !NS.extra.log;
break;
}
(NS.extra.log && console.log(event));
}
}
function keyup(event) {
var e = event.which || event.keyCode;
if (delete NS.keysdown[e], "INPUT" !== document.activeElement.tagName) {
var n = OWOP.player.tool;
if (undefined !== OWOP?.world && n?.isEventDefined("keyup") && n?.call("keyup", [NS.keysdown, event])) return !1;
switch (event.key) {
case "Enter":
case "`":
document.getElementById("chat-input").focus();
break;
}
}
}
let t = EventTarget._eventlists;
let down = [/Custom color\\nType three values separated by a comma: r,g,b\\n\(\.\.\.or the hex string: #RRGGBB\)\\nYou can add multiple colors at a time separating them with a space\./];
let up = [/function\(.\)\{var .=.\.which\|\|.\.keyCode;if\(delete .\[.\],"INPUT"!==document\.activeElement\.tagName\){var .=.\.player\.tool;if\(null!==.&&null!==.\.world&&.\.isEventDefined\("keyup"\)&&.\.call\("keyup",\[.,.\]\)\)return!1;13==.\?.\.chatInput\.focus\(\):16==.&&\(.\.player\.tool="cursor"\)}}/];
NS.etdown = false;
NS.etup = false;
NS.et = false;
if (NS.OPM) {
up.push('function(t) {\n var e = t.which || t.keyCode;\n if (delete b[e], "INPUT" !== document.activeElement.tagName) {\n var n = f.player.tool;\n if (null !== n && null !== E.world && n.isEventDefined("keyup") && n.call("keyup", [b, t])) return !1;\n 13 == e ? k.chatInput.focus() : 16 == e && (f.player.tool = "cursor")\n }\n }');
up.push('function(t){var e=t.which||t.keyCode;if(delete b[e],"INPUT"!==document.activeElement.tagName){var n=f.player.tool;if(null!==n&&null!==E.world&&n.isEventDefined("keyup")&&n.call("keyup",[b,t]))return!1;13==e?k.chatInput.focus():16==e&&(f.player.tool="cursor")}}');
} else {
up.push('function(e){var t=e.which||e.keyCode;if(delete w[t],"INPUT"!==document.activeElement.tagName){var n=d.player.tool;if(null!==n&&null!==x.world&&n.isEventDefined("keyup")&&n.call("keyup",[w,e]))return!1;13==t?k.chatInput.focus():16==t&&(d.player.tool="cursor")}}');
}
for (let i = 0; i < t.length; i++) {
let temp = t[i];
if (NS.etdown !== true) for (let d in down) {
d = down[d];
if ((d instanceof RegExp && d.test(String(temp))) || String(temp) === d) {
NS.etdown = true;
NS.tempdown = temp;
console.log("found down", i);
}
if (NS.etdown === true) break;
}
if (NS.etup !== true) for (let u in up) {
u = up[u];
if (String(temp) === u) {
NS.etup = true
NS.tempup = temp;
console.log("found up", i);
}
if (NS.etup === true) break;
}
if (NS.etdown === true && NS.etup === true) break;
}
if (NS.etdown !== true) console.warn("down was not found");
if (NS.etup !== true) console.warn("up was not found");
if (NS.etdown === true && NS.etup === true) {
NS.et = true;
window.removeEventListener("keydown", NS.tempdown);
window.removeEventListener("keyup", NS.tempup);
window.addEventListener("keydown", keydown);
window.addEventListener("keyup", keyup);
}
delete EventTarget._eventlists;
install();
} else {
setTimeout(init, 100);
}
}
init();