Neko's Scripts

Script for OWOP

Versión del día 02/06/2022. Echa un vistazo a la versión más reciente.

// ==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 = "&nbsp;&nbsp;&nbsp;&nbsp;";
          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();