From 63ae6b7b826cc9728e2204cdea0fea92e99a627d Mon Sep 17 00:00:00 2001
From: Evan Pratten <ewpratten@gmail.com>
Date: Fri, 17 Apr 2020 14:18:27 -0400
Subject: [PATCH] move all files to their own folder

---
 LICENSE => docs/LICENSE            |    0
 index.html => docs/index.html      |   48 +-
 pixelmix.ttf => docs/pixelmix.ttf  |  Bin
 {scripts => docs/scripts}/game.js  | 2124 ++++++++++++++--------------
 {scripts => docs/scripts}/index.js |   56 +-
 5 files changed, 1114 insertions(+), 1114 deletions(-)
 rename LICENSE => docs/LICENSE (100%)
 rename index.html => docs/index.html (96%)
 rename pixelmix.ttf => docs/pixelmix.ttf (100%)
 rename {scripts => docs/scripts}/game.js (97%)
 rename {scripts => docs/scripts}/index.js (86%)

diff --git a/LICENSE b/docs/LICENSE
similarity index 100%
rename from LICENSE
rename to docs/LICENSE
diff --git a/index.html b/docs/index.html
similarity index 96%
rename from index.html
rename to docs/index.html
index 09bb064..cd14837 100644
--- a/index.html
+++ b/docs/index.html
@@ -1,25 +1,25 @@
-<!DOCTYPE html>
-<html>
-    <head>
-        <!-- <link rel="icon" type="image/gif" href="icon.png"/> -->
-        <title>title</title>
-        <style>
-            html {
-                background-color: #242424;
-                color: white;
-            }
-            canvas {
-                position:absolute;
-                top:0;bottom:0;right:0;left:0;
-                margin:auto;
-                border: 5px solid rgb(63, 63, 63); 
-                border-radius: 10px;
-            }
-        </style>
-    </head>
-    <body>
-        <canvas width="1000" height="800" id="game"></canvas>
-        <script src="scripts/game.js"></script>
-        <script src="scripts/index.js"></script>
-    </body>
+<!DOCTYPE html>
+<html>
+    <head>
+        <!-- <link rel="icon" type="image/gif" href="icon.png"/> -->
+        <title>title</title>
+        <style>
+            html {
+                background-color: #242424;
+                color: white;
+            }
+            canvas {
+                position:absolute;
+                top:0;bottom:0;right:0;left:0;
+                margin:auto;
+                border: 5px solid rgb(63, 63, 63); 
+                border-radius: 10px;
+            }
+        </style>
+    </head>
+    <body>
+        <canvas width="1000" height="800" id="game"></canvas>
+        <script src="scripts/game.js"></script>
+        <script src="scripts/index.js"></script>
+    </body>
 </html>
\ No newline at end of file
diff --git a/pixelmix.ttf b/docs/pixelmix.ttf
similarity index 100%
rename from pixelmix.ttf
rename to docs/pixelmix.ttf
diff --git a/scripts/game.js b/docs/scripts/game.js
similarity index 97%
rename from scripts/game.js
rename to docs/scripts/game.js
index e9c1394..dcd3fee 100644
--- a/scripts/game.js
+++ b/docs/scripts/game.js
@@ -1,1063 +1,1063 @@
-function createCanvas(id) {
-    var tempCanvas = document.createElement("canvas");
-    tempCanvas.id = id;
-    tempCanvas.width = canvases.cvs.width;
-    tempCanvas.height = canvases.cvs.height;
-    tempCanvas.style = "image-rendering:pixelated;image-rendering: crisp-edges;display:none";//display:none;
-
-    document.body.appendChild(tempCanvas);
-
-    canvases[`${id}cvs`] = document.getElementById(id);
-    canvases[`${id}ctx`] = canvases[`${id}cvs`].getContext("2d");
-}
-
-function startLoops() {
-    try {draw} catch (err){console.warn(bug+" no draw function found");return null;}
-    try {update} catch (err){console.warn(bug+" no update function found");return null;}
-    try {input} catch (err){seperateInputLoop=false;}
-    onAssetsLoaded();
-
-    requestAnimationFrame(drawLoop);
-    setInterval(updateLoop,1000/updateFPS);
-
-    if(seperateInputLoop) {
-        setInterval(inputLoop,4);
-    }
-}
-
-function mousePosition() {
-    if(drawMode===0) {
-        return {x:(mousePos.x)-camera.x,y:(mousePos.y)-camera.y};
-    } else if(drawMode===1) {
-        var xoff = canvases.cvs.width/2;
-        var yoff = canvases.cvs.height/2;
-        return {x:((mousePos.x-xoff)/camera.zoom+xoff)-camera.x,y:((mousePos.y-yoff)/camera.zoom+yoff)-camera.y};
-    } else {
-        var xoff = canvases.cvs.width/2;
-        var yoff = canvases.cvs.height/2;
-        var tempPos = {x:((mousePos.x-xoff)/camera.zoom+xoff)-camera.x,y:((mousePos.y-yoff)/camera.zoom+yoff)-camera.y};
-
-        var center = {x:-camera.x + cw/2, y:-camera.y + ch/2};
-        var tempAngle = pointTo(center,tempPos) - camera.angle; 
-        var tempDist = dist(center,tempPos);
-
-        return {x:center.x + (Math.cos(tempAngle) * tempDist),y:center.y + (Math.sin(tempAngle) * tempDist)}
-    }
-}
-
-function addStyle() {
-    var tempStyle = document.createElement("style");
-    tempStyle.id="gamejsstyle";
-    document.head.appendChild(tempStyle);
-    var tempMeta = document.createElement("meta");
-    tempMeta.setAttribute("charset","utf-8");
-    document.head.appendChild(tempMeta);
-}
-
-function rand(min,max) {
-    return Math.floor(Math.random() * (max - min + 1)) + min;
-}
-
-function radToDeg(rad) {return rad / Math.PI * 180;}
-function degToRad(deg) {return deg * Math.PI / 180;}
-
-function velocity(angle) {
-    return {x:Math.sin(angle),y:Math.cos(angle)};
-}
-
-function pointTo(point,targetPoint) {
-    var adjacent = (targetPoint.x - point.x);
-    var opposite = (targetPoint.y - point.y);
-    var h = Math.atan2(opposite, adjacent);
-    return h;
-}
-
-function loadImagesAndSounds() {
-    var curpath="";
-    context = new AudioContext();
-    sfxVolumeNode = context.createGain();
-    sfxVolumeNode.connect(context.destination);
-    bmgVolumeNode = context.createGain();
-    bmgVolumeNode.connect(context.destination);
-    deeper(images,"image");
-    deeper(audio,"sound");
-    function deeper(curpos,type) {
-        let addedPath="";
-        for(let j=0;j<curpos.length;j++) {
-            if(typeof curpos[j]=="string") {
-                if(j==0) {
-                    curpath+=curpos[j];
-                    addedPath = curpos[j];
-                } else {
-                    if(type=="image") {
-                        let name = curpath + curpos[j];
-                        imagePaths.push(name);
-                        let temp = new Image();
-                        temp.src = name;
-                        temp.onerror = function () {
-                            console.warn(bug+" "+this.src + " was not found");
-                        };
-                        temp.onload = function() {spriteLoad(name,temp);}
-                        imgs.push(temp);
-                    } else if(type=="sound") {
-                        audioPaths.push(curpath + curpos[j]);
-                        newSound(curpath + curpos[j]);
-                    }
-                }
-            }
-            if(typeof curpos[j]=="object") {
-               deeper(curpos[j],type);
-            }
-        }
-        curpath = curpath.slice(0,curpath.length-addedPath.length);
-    }
-    
-    loadingCircle = new Image();
-    loadingCircle.src = "";
-    clickSound = new Audio("data:audio/x-wav;base64,UklGRowBAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YWgBAADa/6T/2/+S/x//pP769Xr4fPh5+H34evh7+Pv6gf18/QIAhQcIDxUZFR4VHhgeEx6TFgkPCgqAAnz49/X18HLu7ubo4eXc5dzj3Gjf5+Fr5G7pce759YECiwwRFBQZlxuXG5cbFRmTFo8RCAoAAPz69/X08G7udO5s7vz1dvj++nv9gP3+/wEAgwKBAogHkBEpLUNG1lzqcPV683r4eu51ZmnVV7w+qypy88fDoKAXlAqKBoUAgIeHlpYsrTu87Ot9/ZIRGxkoKDEtqionIxgZiwfj5lnaz9JN0E3QV9pd3+Tm9fUBAAQFBwWNDBEPFxSYFpsWEA9+/fP1dfNt83bz8fX9+gQFkAwPDxQPEQ+IBwIAdfjv8OLm2+HX3Nrh4+bm63n4BAUUDx0ZqyCrJSkelBGEAnb4a+5a5Njcztxc31jkZunx9QgFGhQfGa0gqCCuIKkgpRsaFIsHdfhj6dfh0NzM19Th6es=");
-    let pos = {x:cw/2-100,y:ch/2-100};
-    optionsButtons.screenSize = {x:pos.x+160,y:pos.y+12,w:50,h:20};
-    optionsButtons.sfx = {x:pos.x+125,y:pos.y+40,w:120,h:20};
-    optionsButtons.bmg = {x:pos.x+125,y:pos.y+70,w:120,h:20};
-    loadLoop();
-}
-
-function loadLoop() {
-    if(Object.keys(sprites).length == imagePaths.length && audioPaths.length == audioLoadedLength) {
-        startLoops();
-        imagePaths=[];
-        audioPaths=[];
-        imgs=[];
-    } else {
-        curCtx.fillStyle="#2d2d2d";
-        curCtx.fillRect(0,0,cw,ch);
-        text(`audio:   ${audioLoadedLength}/${audioPaths.length}`,10,30,"white",2);
-        text(`sprites: ${Object.keys(sprites).length}/${imagePaths.length}`,10,10,"white",2);
-        curCtx.setTransform(1, 0, 0, 1, Math.round(cw/2), Math.round(ch/2));
-        curCtx.rotate(loadAng);
-        loadAng+=0.1;
-        curCtx.drawImage(loadingCircle,Math.round(-8),Math.round(-8));
-        curCtx.setTransform(1, 0, 0, 1, 0, 0);
-        requestAnimationFrame(loadLoop);
-    }
-}
-
-function spriteLoad(path,image) {
-    let startpos;
-    let endpos = path.lastIndexOf(".");
-    for(let j=endpos-1;acceptableChars.includes(path[j]);j--) {startpos=j;}
-    let spriteName = path.slice(startpos,endpos)
-    let dsize=Math.max(image.width,image.height)/2;
-    sprites[spriteName] = {spr:image,drawLimitSize:dsize};
-    
-}
-
-function newSound(src) {
-    let startpos;
-    let endpos = src.lastIndexOf(".");
-    for(let j=endpos-1;acceptableChars.includes(src[j]);j--) {startpos=j;}
-    let soundName = src.slice(startpos,endpos); 
-    sounds[soundName] = {nodes:[],volNodes:[],src:src,type:"sfx",volume:1};
-    sounds[soundName].nodes = [1];
-
-    let loadingSound = new Audio();
-    loadingSound.onerror = function () {
-        console.warn(bug+" "+ src + " was not found");
-    };
-    loadingSound.src = src;
-    loadingSound.preload='auto';
-    loadingSound.addEventListener('canplaythrough', function() { 
-        audioLoadedLength++;
-     }, false);
-    sounds[soundName].nodes.push(loadingSound);
-
-    let soundNode = context.createMediaElementSource(loadingSound);
-    let gainNode = context.createGain();
-
-    soundNode.connect(gainNode);
-    gainNode.connect(sfxVolumeNode);
-
-    abuffer.push(soundNode);
-    volumeList.push(gainNode);
-    sounds[soundName].volNodes.push(volumeList.length-1);
-}
-
-function addSound(sound) {
-    let loadingSound = new Audio();
-    loadingSound.src = sound.src;
-    loadingSound.preload='auto';
-    sound.nodes.splice(sound.nodes[0],0,loadingSound);
-
-    let soundNode = context.createMediaElementSource(loadingSound);
-    let gainNode = context.createGain();
-    gainNode.gain.value=sound.volume;
-
-    soundNode.connect(gainNode);
-    gainNode.connect(sound.type=="sfx"?sfxVolumeNode:bmgVolumeNode);
-
-    abuffer.push(soundNode);
-    volumeList.push(gainNode);
-    sound.volNodes.push(volumeList.length-1);
-
-    volumeList[sound.volNodes[sound.volNodes.length-1]].gain.value = volumeList[sound.volNodes[0]].gain.value
-}
-
-
-
-function play(sound) {
-    s=sound.nodes;
-    if(s[s[0]].ended || !(s[s[0]].played.length)) {
-        s[s[0]].play();
-        s[0]++;
-        if(s[0]==s.length) {
-            s[0]=1;
-        }
-    } else {
-        addSound(sound);
-        s[s[0]].play();
-        s[0]++;
-        if(s[0]==s.length) {
-            s[0]=1;
-        }
-    }
-}
-
-function setVolume(sound,volume) {
-    for(let i=0,l=sound.volNodes.length;i<l;i++) {
-        volumeList[sound.volNodes[i]].gain.value = volume;
-    }
-}
-
-function setType(sound,newType) {
-    for(let i=0,l=sound.volNodes.length;i<l;i++) {
-        volumeList[sound.volNodes[i]].disconnect(sound.type=="sfx"?sfxVolumeNode:bmgVolumeNode);
-        volumeList[sound.volNodes[i]].connect(newType=="sfx"?sfxVolumeNode:bmgVolumeNode);
-    }
-    sound.type = newType;
-}
-
-function stop(sound) {
-    s=sound.nodes;
-    for(let i=1;i<s.length;i++) {
-        s[i].pause();
-        s[i].currentTime = 0;
-    }
-}
-
-function handleOptionsInput() {
-    let ImTierdMakemenuwork=true;
-    if(optionsMenu) {
-        if(mousePress[0]) {
-            if(rectpoint(optionsButtons.screenSize,mousePos)) {
-                if(screenSize=="1:1") {
-                    screenSize = "fit";
-                    canvasScale=0;
-                } else {
-                    screenSize = "1:1";
-                    canvasScale=1;
-                }
-            }
-            if(!rectpoint({x:cw/2,y:ch/2,w:200,h:200},mousePos)) {
-                optionsMenu=false;
-                ImTierdMakemenuwork=false;
-            }
-        }
-        if(mouseDown[0]) {
-            if(rectpoint(optionsButtons.sfx,mousePos)) {
-                volume.sfx = (mousePos.x-(optionsButtons.sfx.x-60))/120;
-            }
-            if(rectpoint(optionsButtons.bmg,mousePos)) {
-                volume.bgm = (mousePos.x-(optionsButtons.bmg.x-60))/120;
-            }
-        }
-    }
-    if(mousePos.x>cw-32&&mousePos.y<32) {
-        if(mousePress[0]&&ImTierdMakemenuwork) {
-            clickSound.play();
-            paused=true;
-            optionsMenu=!optionsMenu;
-        }
-        optionsHover = 25;
-    } else {
-        optionsHover = 0;
-    }
-    if(mousePos.x<cw-32&&mousePos.x>cw-64&&mousePos.y<32) {
-        pauseHover = 25;
-        if(mousePress[0]) {
-            clickSound.play();
-            paused=!paused;
-        }
-    } else {
-        pauseHover = 0;
-    }
-    
-}
-function addFont() {
-    var tempStyle = document.createElement("style");
-    tempStyle.innerHTML = `
-    @font-face {
-        font-family: 'PixelArial11';
-        src: url("./pixelmix.ttf") format('truetype');
-        font-weight: 900;
-        font-style: normal;
-    
-    }
-    html {font-family: 'PixelArial11' !important; font-size: 16px;}
-    `;
-    document.head.appendChild(tempStyle);
-    canvases.ctx.textBaseline = "hanging";
-    canvases.ctx.textAlign = "left";
-}
-var scaleDefault = 1;
-function img(img,x,y,angle=0,sx=scaleDefault,sy=scaleDefault) {
-    var half = img.drawLimitSize;
-    if((x+half>drawLimitLeft&&x-half<drawLimitRight&&y+half>drawLimitTop&&y-half<drawLimitBottom)||absDraw) {
-        let spr = img.spr;
-        if(angle===0&&sx===1&&sy===1) {
-            curCtx.drawImage(spr,Math.round(x+camera.x+difx-(spr.width/2)),Math.round(y+camera.y+dify-(spr.height/2)));
-        } else {
-            curCtx.setTransform(sx, 0, 0, sy, Math.round(x+camera.x+difx), Math.round(y+camera.y+dify));
-            curCtx.rotate(angle);
-            curCtx.drawImage(spr,Math.round(-spr.width/2),Math.round(-spr.height/2));
-            curCtx.setTransform(1, 0, 0, 1, 0, 0);
-        }
-    }
-}
-
-function imgIgnoreCutoff(img,x,y,angle=0,sx=1,sy=1) {
-    let spr = img.spr;
-    if(angle===0&&sx===1&&sy===1) {
-        curCtx.drawImage(spr,Math.round(x+camera.x+difx-(spr.width/2)),Math.round(y+camera.y+dify-(spr.height/2)));
-    } else {
-        curCtx.setTransform(sx, 0, 0, sy, Math.round(x+camera.x+difx), Math.round(y+camera.y+dify));
-        curCtx.rotate(angle);
-        curCtx.drawImage(spr,Math.round(-spr.width/2),Math.round(-spr.height/2));
-        curCtx.setTransform(1, 0, 0, 1, 0, 0);
-    }
-}
-
-function rect(x,y,w,h,color) {
-    curCtx.fillStyle = color;
-    curCtx.fillRect(x-(w/2)+camera.x+difx,y-(h/2)+camera.y+dify,w,h);
-}
-
-function circle(x,y,r,color) {
-    curCtx.beginPath();
-    curCtx.arc(x+camera.x+difx, y+camera.y+dify, r, 0, 2 * Math.PI, false);
-    curCtx.fillStyle = color;
-    curCtx.fill();
-}
-
-function shape(x,y,relitivePoints,color) {
-    x+=camera.x+difx;
-    y+=camera.y+dify;
-    curCtx.fillStyle = color;
-    curCtx.beginPath();
-    curCtx.moveTo(x+relitivePoints[0].x, y+relitivePoints[0].y);
-    for(let i=1,l=relitivePoints.length;i<l;i++) {
-        curCtx.lineTo(x+relitivePoints[i].x, y+relitivePoints[i].y);
-    }
-    curCtx.fill();
-}
-
-function text(txt,x,y,color="black",size=1,maxWidth=cw) {
-    txt = txt.toString();
-    curCtx.fillStyle = color;
-    curCtx.font = `${Math.round(size)*8}px PixelArial11`;
-                                                                                        //I hate text wrapping now 
-    var txtList = txt.split("\n");                                                      //split string on enters
-    for(let i=0;i<txtList.length;i++) {                                                 //go through array of strings
-        if(curCtx.measureText(txtList[i]).width>maxWidth) {                             //if the string is too big, divide up into smaller strings
-            var tempTxt = txtList[i].split(" ");                                        //split into individual words
-            var tempStr="";                                                             //string for measuring size
-            var addAmount=0;                                                            //track where in the txtList we are
-            txtList.splice(i,1);                                                        //remove the too long string
-            for(let j=0;j<tempTxt.length;j++) {                                         //go through the split up string
-                if(curCtx.measureText(tempStr + tempTxt[j] + " ").width<maxWidth) {     //if adding a word doesn't make tempStr too long, add it, other wise, add tempStr to txtList;
-                    tempStr += tempTxt[j] + " ";
-                } else {
-                    if(j==0) {tempStr+=tempTxt[j];}                                     //if we are here when j is 0, we have one word that is longer then the maxWidth, so we just draw it
-                    txtList.splice(i+addAmount,0,tempStr);                              //put tempStr in txtList
-                    addAmount++;                                                        //move the position we put the tempStr in
-                    tempStr="";                                                         //reset tempStr
-                    tempTxt.splice(0,(j==0?1:j));                                       //delete words that have been used
-                    j=-1;                                                               //make it so in the next loop, j starts at 0
-                }
-            }
-            if(tempStr.length!=0) {
-                txtList.splice(i+addAmount,0,tempStr);                                  //add any leftover text
-            }
-        }
-    }
-
-    for(let i=0;i<txtList.length;i++) {
-        curCtx.fillText(txtList[i],x+camera.x+difx,y+camera.y+dify+((i+(drawMode?1:0))*8*size+(size*i)));
-    }
-}
-
-function textWidth(txt,size=1) {
-    txt = txt.toString();
-    curCtx.font = `${Math.round(size)*8}px PixelArial11`;
-    return curCtx.measureText(txt).width;
-}
-
-function centerCameraOn(x,y) {
-    camera.x = -x+canvases.cvs.width/2;
-    camera.y = -y+canvases.cvs.height/2;
-}
-
-function moveCamera(x,y) {
-    camera.x -= y * Math.sin(camera.angle);
-    camera.y -= y * Math.cos(camera.angle);
-    camera.x -= x * Math.sin(camera.angle + 1.57079632);
-    camera.y -= x * Math.cos(camera.angle + 1.57079632);
-}
-
-function imgRotScale(x,y,angle,scale,pic,ctx) { //used for camera movement
-    ctx.setTransform(scale, 0, 0, scale, x, y);
-    ctx.rotate(angle);
-    ctx.drawImage(pic,-pic.width/2,-pic.height/2);
-    ctx.setTransform(1, 0, 0, 1, 0, 0);
-}
-
-function drawCursor() {
-    if(cursor.sprite&&mouseOnCanvas) {
-        if(cursor.alignment) {
-            canvases.ctx.drawImage(cursor.sprite.spr,mousePos.x-Math.round(cursor.sprite.spr.width/2),mousePos.y-Math.round(cursor.sprite.spr.height/2));
-        } else {
-            canvases.ctx.drawImage(cursor.sprite.spr,mousePos.x,mousePos.y);
-        }
-        cursor.show = false;
-    } else {
-        cursor.show = true;
-    }
-}
-
-function render() {
-    if(drawMode===1) {
-        imgRotScale(canvases.cvs.width/2,canvases.cvs.height/2,0,camera.zoom,canvases.buffer1cvs,canvases.ctx);
-    }
-    if(drawMode===2) {
-        imgRotScale(canvases.cvs.width/2,canvases.cvs.height/2,camera.angle,1,canvases.buffer2cvs,canvases.buffer1ctx);
-        imgRotScale(canvases.cvs.width/2,canvases.cvs.height/2,0,camera.zoom,canvases.buffer1cvs,canvases.ctx);
-    }
-}
-
-function clearCanvases() {
-    canvases.ctx.clearRect(0,0,canvases.cvs.width,canvases.cvs.height);
-    canvases.buffer1ctx.clearRect(0,0,canvases.buffer1cvs.width,canvases.buffer1cvs.height);
-    canvases.buffer2ctx.clearRect(0,0,canvases.buffer2cvs.width,canvases.buffer2cvs.height);
-}
-
-function switchDrawMode() {
-    if(camera.zoom<1) {camera.zoom=1;}
-    if(camera.angle!=0) {
-        drawMode=2;
-    } else if(camera.zoom!=1) {
-        drawMode=1;
-    } else {
-        drawMode=0;
-    }
-    switch (drawMode) {
-        case 0: curCtx = canvases.ctx; break;
-        case 1: curCtx = canvases.buffer1ctx; break;
-        case 2: curCtx = canvases.buffer2ctx; break;
-    }
-}
-
-function resizeBuffers() {
-    var tempSize = maxCvsSize/camera.zoom;
-    var tempSizeAndPadding = tempSize + (tempSize/2)
-
-    canvases.buffer2cvs.width = tempSizeAndPadding;
-    canvases.buffer2cvs.height = tempSizeAndPadding;
-    
-    if(drawMode===2) {
-        difx = (canvases.buffer2cvs.width - canvases.cvs.width)/2;
-        dify = (canvases.buffer2cvs.height - canvases.cvs.height)/2;
-    } else {
-        difx=0;
-        dify=0;
-    }
-    canvases.buffer2ctx.imageSmoothingEnabled = false;
-}
-
-function scaleCanvases() { //scales canvas by canvas scale, if scale is 0, canvas will try to fit screen
-    var style = document.getElementById("gamejsstyle");
-    if(canvasScale==0) {
-        var tempScale = Math.min(Math.floor(window.innerWidth/canvases.cvs.width),Math.floor(window.innerHeight/canvases.cvs.height));
-        tempScale=tempScale<1?1:tempScale;
-        autoScale=tempScale;
-        style.innerHTML = `#game {image-rendering:pixelated;image-rendering: crisp-edges;width:${tempScale*canvases.cvs.width}px;cursor: ${cursor.show?"crosshair":"none"};}`;
-    } else {
-        style.innerHTML = `#game {image-rendering:pixelated;image-rendering: crisp-edges;width:${Math.floor(canvasScale*canvases.cvs.width)}px;cursor: ${cursor.show?"crosshair":"none"};}`;
-    }
-}
-
-function drawButtons() {
-    let pos = {x:cw-16,y:16}; 
-    //options
-    rect(pos.x,pos.y,34,34,"#9c9c9c");
-    let c = optionsHover+45;
-    rect(pos.x,pos.y,32,32,`rgb(${c},${c},${c})`);
-    c = optionsHover + 69;
-    let cc = `rgb(${c},${c},${c})`;
-    rect(pos.x,pos.y-6,26,4,cc);
-        rect(pos.x-6,pos.y-6,4,8,cc);
-    rect(pos.x,pos.y+6,26,4,cc);
-        rect(pos.x+11,pos.y+6,4,8,cc);
-    //pause
-    pos.x-=33;
-    rect(pos.x,pos.y,34,34,"#9c9c9c");
-    c = pauseHover+45;
-    rect(pos.x,pos.y,32,32,`rgb(${c},${c},${c})`);
-    c = pauseHover + 69;
-    cc = `rgb(${c},${c},${c})`;
-    if(paused) {
-        shape(pos.x,pos.y,[{x:-7,y:-10},{x:-7,y:10},{x:10,y:0}],cc);
-    } else {
-        rect(pos.x+6,pos.y,6,20,cc);
-        rect(pos.x-6,pos.y,6,20,cc);
-    }
-}
-
-function drawOptionsMenu() {
-    if(optionsMenu) {
-        let pos = {x:cw/2-100,y:ch/2-100};
-        rect(cw/2,ch/2,200,200,"#242424");
-        text("Screen Size:",pos.x+2,pos.y+2,"white",2);
-            let b = optionsButtons.screenSize;
-            rect(b.x,b.y,b.w,b.h,"#444444");
-            text(screenSize,pos.x+145,pos.y+4,"white",2);
-        text("sfx",pos.x+2,pos.y+30,"white",2);
-            b = optionsButtons.sfx;
-            rect(b.x,b.y,b.w,b.h-10,"#444444");
-            rect((b.x-60)+(volume.sfx*120),b.y,8,20,"#444444");
-        text("bmg",pos.x+2,pos.y+60,"white",2);
-            b = optionsButtons.bmg;
-            rect(b.x,b.y,b.w,b.h-10,"#444444");
-            rect((b.x-60)+(volume.bgm*120),b.y,8,20,"#444444");
-    }
-}
-var k={a:65,b:66,c:67,d:68,e:69,f:70,g:71,h:72,i:73,j:74,k:75,l:76,m:77,n:78,o:79,p:80,q:81,r:82,s:83,t:84,u:85,v:86,w:87,x:88,y:89,z:90,0:48,1:49,2:50,3:51,4:52,5:53,6:54,7:55,8:56,9:57,BACKTICK:192,MINUS:189,EQUALS:187,OPENSQUARE:219,ENDSQUARE:221,SEMICOLON:186,SINGLEQUOTE:222,BACKSLASH:220,COMMA:188,PERIOD:190,SLASH:191,ENTER:13,BACKSPACE:8,TAB:9,CAPSLOCK:20,SHIFT:16,CONTROL:17,ALT:18,META:91,LEFTBACKSLASH:226,ESCAPE:27,HOME:36,END:35,PAGEUP:33,PAGEDOWN:34,DELETE:46,INSERT:45,PAUSE:19,UP:38,DOWN:40,LEFT:37,RIGHT:39,CONTEXT:93,SPACE:32,F1:112,F2:113,F3:114,F4:115,F5:116,F6:117,F7:118,F8:119,F9:120,F10:121,F11:122,F12:123};
-var keyPress = [];
-var keyDown = [];
-var mousePress = [];
-var mouseDown = [];
-var scroll = 0;
-var mousePos = {
-    x:0,
-    y:0
-}
-var preventedEvents = [false,true,true];
-
-function addListenersTo(elementToListenTo) {
-    window.addEventListener("keydown",kdown);
-    window.addEventListener("keyup",kup);
-    elementToListenTo.addEventListener("mousedown",mdown);
-    elementToListenTo.addEventListener("mouseup",mup);
-    elementToListenTo.addEventListener("mousemove",mmove);
-    elementToListenTo.addEventListener("contextmenu",cmenu);
-    elementToListenTo.addEventListener("wheel",scrl);
-}
-function removeListenersFrom(elementToListenTo) {
-    window.removeEventListener("keydown",kdown);
-    window.removeEventListener("keyup",kup);
-    elementToListenTo.removeEventListener("mousedown",mdown);
-    elementToListenTo.removeEventListener("mouseup",mup);
-    elementToListenTo.removeEventListener("mousemove",mmove);
-    elementToListenTo.removeEventListener("contextmenu",cmenu);
-    elementToListenTo.removeEventListener("wheel",scrl);
-}
-function resetInput() {
-    for(var i=0;i<keyPress.length;i++){if(keyPress[i]){keyPress[i]=0}}
-    for(var i=0;i<mousePress.length;i++){if(mousePress[i]){mousePress[i]=0}}
-    scroll=0;
-}
-function kdown(e) {
-    var h=e.keyCode;
-    keyPress[h]=keyPress[h]==[][[]]?1:0;
-    keyDown[h]=1;
-    if(preventedEvents[0]) {e.preventDefault()}
-}
-function kup(e) {
-    var h=e.keyCode;
-    delete keyPress[h];
-    delete keyDown[h];
-}
-function mdown(e) {
-    var h=e.button;
-    mousePress[h]=mousePress[h]==[][[]]?1:0;
-    mouseDown[h]=1;
-    if(preventedEvents[1]) {e.preventDefault()}
-}
-function mup(e) {
-    var h=e.button;
-    delete mousePress[h];
-    delete mouseDown[h];
-}
-function mmove(e) {
-    mousePos.x=e.offsetX/(!canvasScale?autoScale:canvasScale);
-    mousePos.y=e.offsetY/(!canvasScale?autoScale:canvasScale);    
-}
-function cmenu(e) {
-    if(preventedEvents[1]) {e.preventDefault()}
-}
-function scrl(e) {
-    scroll+=-1*(e.deltaY/100);
-    if(preventedEvents[2]) {e.preventDefault()}
-}
-function dist(point1,point2) {
-    let one = (point2.x - point1.x);
-    let two = (point2.y - point1.y);
-    return Math.sqrt((one*one)+(two*two));
-}
-
-function circlecircle(circle1,circle2) {
-    if( dist(circle1,circle2) < (circle1.r + circle2.r)) {
-		return true;
-	} else {
-        return false;
-    }
-}
-
-function circlepoint(circle,point) {
-    if( dist(circle,point) < circle.r) {
-		return true;
-	} else {
-        return false;
-    }
-}
-
-function rectrect(rect1,rect2) {
-    if(rect1.x + rect1.w/2 >= rect2.x - rect2.w/2 &&
-       rect1.x - rect1.w/2 <= rect2.x + rect2.w/2 &&
-       rect1.y + rect1.h/2 >= rect2.y - rect2.h/2 &&
-       rect1.y - rect1.h/2 <= rect2.y + rect2.h/2) {
-        return true;
-    } else {
-        return false;
-    }
-}
-
-function rectpoint(rect,point) {
-    if(rect.x + rect.w/2 >= point.x &&
-       rect.x - rect.w/2 <= point.x &&
-       rect.y + rect.h/2 >= point.y &&
-       rect.y - rect.h/2 <= point.y ) {
-        return true;
-    } else {
-        return false;
-    }
-}
-
-function circlerect(circle,rect) { //credit: https://yal.cc/rectangle-circle-intersection-test/
-    let rectHalfWidth  = rect.w/2;
-    let rectHalfHeight = rect.h/2;
-    let deltaX = circle.x - Math.max(rect.x - rectHalfWidth, Math.min(circle.x, rect.x + rectHalfWidth));
-    let deltaY = circle.y - Math.max(rect.y - rectHalfHeight, Math.min(circle.y, rect.y + rectHalfHeight));
-    return (deltaX * deltaX + deltaY * deltaY) < (circle.r * circle.r);
-}
-
-function circleOnSideRect(circle,rect) {
-    let rectHalfWidth  = rect.w/2;
-    let rectHalfHeight = rect.h/2;
-    let left   = rect.x - rectHalfWidth;
-    let right  = rect.x + rectHalfWidth;
-    let top    = rect.y - rectHalfHeight;
-    let bottom = rect.y + rectHalfHeight;
-    let cx = circle.x;
-    let cy = circle.y;
-    if(cy < top && cx > left && cx < right) { // top side
-        return 0;
-    } else if(cy > bottom && cx > left && cx < right) { // bottom side
-        return 2;
-    } else if (cx < left && cy > top && cy < bottom) { // left side
-        return 3;
-    } else if (cx > right && cy > top && cy < bottom) { // right side
-        return 1;
-    } else {
-        let returnValue=0; // 0 = top, 1 = right, 2 = bottom, 3 = left
-        let topleft = dist (circle,{x:left,y:top});
-        let topright = dist (circle,{x:right,y:top});
-        let bottomleft = dist (circle,{x:left,y:bottom});
-        let bottomright = dist (circle,{x:right,y:bottom});
-        switch(Math.min(topleft,topright,bottomleft,bottomright)) { // find what corner the cricle is closer to, then determine what side it is closer to
-            case topleft:
-                var m = slope(rect,{x:left,y:top});
-                var mperp = -(1/m);
-                var b = yIntercept(rect,m);
-                var bperp = yIntercept(circle,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 3;} else {returnValue = 0;}
-                break;
-            case topright:
-                var m = slope(rect,{x:right,y:top});
-                var mperp = -(1/m);
-                var b = yIntercept(rect,m);
-                var bperp = yIntercept(circle,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 0;} else {returnValue = 1;}
-                break;
-            case bottomleft:
-                var m = slope(rect,{x:left,y:bottom});
-                var mperp = -(1/m);
-                var b = yIntercept(rect,m);
-                var bperp = yIntercept(circle,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 3;} else {returnValue = 2;}
-                break;
-            case bottomright:
-                var m = slope(rect,{x:right,y:bottom});
-                var mperp = -(1/m);
-                var b = yIntercept(rect,m);
-                var bperp = yIntercept(circle,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 2;} else {returnValue = 1;}
-                break;
-        }
-        return returnValue;
-    }
-}
-
-function rectOnSideRect(rect1,rect2) {
-    let rectHalfWidth2  = rect2.w/2;
-    let rectHalfHeight2 = rect2.h/2;
-    let left2   = rect2.x - rectHalfWidth2;
-    let right2 = rect2.x + rectHalfWidth2;
-    let top2   = rect2.y - rectHalfHeight2;
-    let bottom2 = rect2.y + rectHalfHeight2;
-
-    let rectHalfWidth1  = rect1.w/2;
-    let rectHalfHeight1 = rect1.h/2;
-    let rx1 = rect1.x;
-    let ry1 = rect1.y;
-    let left1   = rx1 - rectHalfWidth1;
-    let right1 = rx1 + rectHalfWidth1;
-    let top1   = ry1 - rectHalfHeight1;
-    let bottom1 = ry1 + rectHalfHeight1;
-    // find what point is closer to the rectangle 
-    let topleft1 = dist (rect2,{x:left1,y:top1});
-    let topright1 = dist (rect2,{x:right1,y:top1});
-    let bottomleft1 = dist (rect2,{x:left1,y:bottom1});
-    let bottomright1 = dist (rect2,{x:right1,y:bottom1});
-    let topmiddle1 = dist (rect2,{x:rx1,y:top1});
-    let rightmiddle1 = dist (rect2,{x:right1,y:ry1});
-    let bottommiddle1 = dist (rect2,{x:rx1,y:bottom1});
-    let leftmiddle1 = dist (rect2,{x:left1,y:ry1});
-    let cx = rx1;
-    let cy = ry1;
-    switch(Math.min(topleft1,topright1,bottomleft1,bottomright1,topmiddle1,rightmiddle1,bottommiddle1,leftmiddle1)) {
-        //set the point we are testing to the closest point to the rectangle
-        case topleft1:
-            cx -= rect1.w/2;
-            cy -= rect1.h/2;
-            break;
-        case topright1:
-            cx += rect1.w/2;
-            cy -= rect1.h/2;
-            break;
-        case bottomleft1:
-            cx -= rect1.w/2;
-            cy += rect1.h/2;
-            break;
-        case bottomright1:
-            cx += rect1.w/2;
-            cy += rect1.h/2;
-            break;
-        case topmiddle1:
-            cy -= rect1.h/2;
-            break;
-        case rightmiddle1:
-            cx += rect1.w/2;
-            break;
-        case bottommiddle1:
-            cy += rect1.h/2;
-            break;
-        case leftmiddle1:
-            cx -= rect1.w/2;
-            break;
-    }
-    if(cy < top2 && cx > left2 && cx < right2) { // top side
-        return 0;
-    } else if(cy > bottom2 && cx > left2 && cx < right2) { // bottom side
-        return 2;
-    } else if (cx < left2 && cy > top2 && cy < bottom2) { // left side
-        return 3;
-    } else if (cx > right2 && cy > top2 && cy < bottom2) { // right side
-        return 1;
-    } else {
-        let returnValue=0; // 0 = top, 1 = right, 2 = bottom, 3 = left
-        let determiningPoint = {x:cx,y:cy};
-        let topleft = dist (determiningPoint,{x:left2,y:top2});
-        let topright = dist (determiningPoint,{x:right2,y:top2});
-        let bottomleft = dist (determiningPoint,{x:left2,y:bottom2});
-        let bottomright = dist (determiningPoint,{x:right2,y:bottom2});
-        switch(Math.min(topleft,topright,bottomleft,bottomright)) { // find what corner the point is closer to, then determine what side it is closer to
-            case topleft:
-                var m = slope(rect2,{x:left2,y:top2});
-                var mperp = -(1/m);
-                var b = yIntercept(rect2,m);
-                var bperp = yIntercept(determiningPoint,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 3;} else {returnValue = 0;}
-                break;
-            case topright:
-                var m = slope(rect2,{x:right2,y:top2});
-                var mperp = -(1/m);
-                var b = yIntercept(rect2,m);
-                var bperp = yIntercept(determiningPoint,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 0;} else {returnValue = 1;}
-                break;
-            case bottomleft:
-                var m = slope(rect2,{x:left2,y:bottom2});
-                var mperp = -(1/m);
-                var b = yIntercept(rect2,m);
-                var bperp = yIntercept(determiningPoint,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 3;} else {returnValue = 2;}
-                break;
-            case bottomright:
-                var m = slope(rect2,{x:right2,y:bottom2});
-                var mperp = -(1/m);
-                var b = yIntercept(rect2,m);
-                var bperp = yIntercept(determiningPoint,mperp);
-                var mid = POI(m,b,mperp,bperp);
-                if(cx<mid) {returnValue = 2;} else {returnValue = 1;}
-                break;
-        }
-        return returnValue;
-    }
-}
-
-function slope(point1,point2) {
-    return ((point2.y-point1.y)/(point2.x-point1.x));
-}
-
-function yIntercept(point,slope) {
-    return point.y - (slope * point.x);
-}
-
-function POI(m1,b1,m2,b2) {
-    x = (b2 - b1) / (m1 - m2);
-    return x;
-    //y = m1 * x + b1;
-}
-
-function ifRectOnEdgeBounce(rect) {
-    let rx = rect.x;
-    let ry = rect.y;
-    let rw = rect.w/2;
-    let rh = rect.h/2;
-    if(rx+rw>edge.right) {
-        rect.v.x *= -1;
-        rect.x = edge.right-rw;
-    }
-    if(rx-rw<edge.left) {
-        rect.v.x *= -1;
-        rect.x = edge.left+rw;
-    }
-    if(ry+rh>edge.bottom) {
-        rect.v.y *= -1;
-        rect.y = edge.bottom-rh;
-    }
-    if(ry-rh<edge.top) {
-        rect.v.y *= -1;
-        rect.y = edge.top+rh;
-    }
-}
-
-function ifCircleOnEdgeBounce(circle) {
-    let cx = circle.x;
-    let cy = circle.y;
-    let cr = circle.r;
-    if(cx+cr>edge.right) {
-        circle.v.x *= -1;
-        circle.x = edge.right-cr;
-    }
-    if(cx-cr<edge.left) {
-        circle.v.x *= -1;
-        circle.x = edge.left+cr;
-    }
-    if(cy+cr>edge.bottom) {
-        circle.v.y *= -1;
-        circle.y = edge.bottom-cr;
-    }
-    if(cy-cr<edge.top) {
-        circle.v.y *= -1;
-        circle.y = edge.top+cr;
-    }
-}
-// create globals
-var canvases={cvs:null,ctx:null,buffer1cvs:null,buffer1ctx:null,buffer2cvs:null,buffer2ctx:null}, // visable and hidden canvases
-cw, // canvas width
-ch, // canvas height
-camera={zoom:1,angle:0,x:0,y:0}, // affects how everything is drawn
-updateFPS=60,
-gameStarted=false,
-drawMode=0, // 0=normal, 1=zoomed, 2=zoomed/rotated, set automatically depending on camera
-absDraw=false,
-curCtx, // what canvas to draw to
-maxCvsSize, // used by second buffer
-canvasScale=1,
-difx=0, // offsets for drawing
-dify=0,
-seperateInputLoop=true,
-edge={top:null,bottom:null,left:null,right:null}, // used by if___OnEdgeBounce, set to canvas size at setup, can be changed whenever
-drawLimitLeft,
-drawLimitRight,
-drawLimitTop,
-drawLimitBottom,
-sizeDif,
-bug="\uD83D\uDC1B",
-loadingCircle,
-loadAng=0,
-optionsHover=0,
-pauseHover=0,
-optionsMenu=false,
-optionsButtons={},
-clickSound,
-paused=false,
-screenSize="1:1",
-autoScale=1,
-
-images=[], // put image paths here
-imagePaths=[],
-imgs=[],
-sprites={}, // loaded images
-
-audio=[], // put audio paths here
-audioPaths=[],
-sounds={}, // loaded sounds
-abuffer = [], // audio nodes shoved here
-volumeList = [], // gain nodes shoved here
-audioLoadedLength=0,
-volume={sfx:1,bgm:1};
-/* options
-    future:
-        modefileable checklist
-        key bindings
-*/
-
-cursor = {sprite:null,alignment:1,show:true}, // 0=topleft, 1=centered
-mouseOnCanvas=false;
-
-const acceptableChars="qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM1234567890_-. ";//for image names
-
-const AudioContext = window.AudioContext||window.webkitAudioContext;
-var context;
-var sfxVolumeNode;
-var bmgVolumeNode;
-
-document.getElementById("game").onmouseout = function()   {mouseOnCanvas = false;}
-document.getElementById("game").onmouseover = function()   {mouseOnCanvas = true;}
-
-//setup canvases and input
-function setup(physicsFPS) {
-    updateFPS = physicsFPS;
-    
-    canvases.cvs = document.getElementById("game");
-    canvases.ctx = canvases.cvs.getContext("2d", { alpha: false });
-
-    canvases.cvs.onmousedown = function () {if(!gameStarted){loadImagesAndSounds();gameStarted=true;}}
-
-    createCanvas("buffer1");
-    createCanvas("buffer2");
-
-    canvases.ctx.imageSmoothingEnabled = false;
-    canvases.buffer1ctx.imageSmoothingEnabled = false;
-    canvases.buffer2ctx.imageSmoothingEnabled = false;
-
-    maxCvsSize=Math.max(canvases.cvs.width,canvases.cvs.height);
-    sizeDif=maxCvsSize-Math.min(canvases.cvs.width,canvases.cvs.height);
-    cw=canvases.cvs.width;
-    ch=canvases.cvs.height;
-    
-    edge={top:0,bottom:ch,left:0,right:cw};
-
-    addFont();
-    addStyle();
-
-    addListenersTo(canvases.cvs);
-
-    curCtx = canvases.ctx;
-    requestAnimationFrame(startButton);
-    function startButton() {
-        curCtx.fillStyle="#2d2d2d";
-        curCtx.fillRect(0,0,cw,ch);//debugger;
-        circle(cw/2,ch/2,27,"#066312");
-        circle(cw/2,ch/2,23,"#149124");
-        shape(cw/2,ch/2,[{x:-7,y:-15},{x:-7,y:15},{x:15,y:0}],"#47f55d");
-        if(!gameStarted) {requestAnimationFrame(startButton);}
-    }
-}
-
-function drawLoop() {
-    cw=canvases.cvs.width;
-    ch=canvases.cvs.height;
-    scaleCanvases();
-
-    switchDrawMode();
-    
-    resizeBuffers();
-
-    clearCanvases();
-
-    var limitModifyer = 0;
-    if(drawMode==2) {limitModifyer=canvases.buffer2cvs.width-maxCvsSize;}
-    drawLimitLeft   = -camera.x - (drawMode==2?sizeDif:0) - limitModifyer;
-    drawLimitRight  = -camera.x + maxCvsSize + (drawMode==2?sizeDif:0) + limitModifyer;
-    drawLimitTop    = -camera.y -(drawMode==2?sizeDif:0) - limitModifyer;
-    drawLimitBottom = -camera.y + maxCvsSize + (drawMode==2?sizeDif:0) + limitModifyer;
-
-    draw();
-    
-    render();
-
-    curCtx=canvases.ctx;
-    difx=0;dify=0;
-    var camCache = {x:camera.x,y:camera.y};
-    var drawModeCache = drawMode;
-    camera.x=0;camera.y=0;
-    drawMode=0;
-    absDraw=true;
-    absoluteDraw();
-    absDraw=false;
-
-    drawButtons();
-    drawOptionsMenu();
-    drawCursor();
-
-    drawMode=drawModeCache;
-
-    camera.x = camCache.x;
-    camera.y = camCache.y;
-
-    requestAnimationFrame(drawLoop);
-}
-
-function updateLoop() {
-    if(seperateInputLoop==false) {
-        handleOptionsInput();
-    }
-    sfxVolumeNode.gain.value = volume.sfx;
-    bmgVolumeNode.gain.value = volume.bgm;
-    if(!paused) {
-        update();
-    }
-
-    if(seperateInputLoop==false) {
-        resetInput();
-    }
-}
-
-
-function inputLoop() {
-    handleOptionsInput();
-    if(!paused) {
-        input();
-    }
-
-    resetInput();
+function createCanvas(id) {
+    var tempCanvas = document.createElement("canvas");
+    tempCanvas.id = id;
+    tempCanvas.width = canvases.cvs.width;
+    tempCanvas.height = canvases.cvs.height;
+    tempCanvas.style = "image-rendering:pixelated;image-rendering: crisp-edges;display:none";//display:none;
+
+    document.body.appendChild(tempCanvas);
+
+    canvases[`${id}cvs`] = document.getElementById(id);
+    canvases[`${id}ctx`] = canvases[`${id}cvs`].getContext("2d");
+}
+
+function startLoops() {
+    try {draw} catch (err){console.warn(bug+" no draw function found");return null;}
+    try {update} catch (err){console.warn(bug+" no update function found");return null;}
+    try {input} catch (err){seperateInputLoop=false;}
+    onAssetsLoaded();
+
+    requestAnimationFrame(drawLoop);
+    setInterval(updateLoop,1000/updateFPS);
+
+    if(seperateInputLoop) {
+        setInterval(inputLoop,4);
+    }
+}
+
+function mousePosition() {
+    if(drawMode===0) {
+        return {x:(mousePos.x)-camera.x,y:(mousePos.y)-camera.y};
+    } else if(drawMode===1) {
+        var xoff = canvases.cvs.width/2;
+        var yoff = canvases.cvs.height/2;
+        return {x:((mousePos.x-xoff)/camera.zoom+xoff)-camera.x,y:((mousePos.y-yoff)/camera.zoom+yoff)-camera.y};
+    } else {
+        var xoff = canvases.cvs.width/2;
+        var yoff = canvases.cvs.height/2;
+        var tempPos = {x:((mousePos.x-xoff)/camera.zoom+xoff)-camera.x,y:((mousePos.y-yoff)/camera.zoom+yoff)-camera.y};
+
+        var center = {x:-camera.x + cw/2, y:-camera.y + ch/2};
+        var tempAngle = pointTo(center,tempPos) - camera.angle; 
+        var tempDist = dist(center,tempPos);
+
+        return {x:center.x + (Math.cos(tempAngle) * tempDist),y:center.y + (Math.sin(tempAngle) * tempDist)}
+    }
+}
+
+function addStyle() {
+    var tempStyle = document.createElement("style");
+    tempStyle.id="gamejsstyle";
+    document.head.appendChild(tempStyle);
+    var tempMeta = document.createElement("meta");
+    tempMeta.setAttribute("charset","utf-8");
+    document.head.appendChild(tempMeta);
+}
+
+function rand(min,max) {
+    return Math.floor(Math.random() * (max - min + 1)) + min;
+}
+
+function radToDeg(rad) {return rad / Math.PI * 180;}
+function degToRad(deg) {return deg * Math.PI / 180;}
+
+function velocity(angle) {
+    return {x:Math.sin(angle),y:Math.cos(angle)};
+}
+
+function pointTo(point,targetPoint) {
+    var adjacent = (targetPoint.x - point.x);
+    var opposite = (targetPoint.y - point.y);
+    var h = Math.atan2(opposite, adjacent);
+    return h;
+}
+
+function loadImagesAndSounds() {
+    var curpath="";
+    context = new AudioContext();
+    sfxVolumeNode = context.createGain();
+    sfxVolumeNode.connect(context.destination);
+    bmgVolumeNode = context.createGain();
+    bmgVolumeNode.connect(context.destination);
+    deeper(images,"image");
+    deeper(audio,"sound");
+    function deeper(curpos,type) {
+        let addedPath="";
+        for(let j=0;j<curpos.length;j++) {
+            if(typeof curpos[j]=="string") {
+                if(j==0) {
+                    curpath+=curpos[j];
+                    addedPath = curpos[j];
+                } else {
+                    if(type=="image") {
+                        let name = curpath + curpos[j];
+                        imagePaths.push(name);
+                        let temp = new Image();
+                        temp.src = name;
+                        temp.onerror = function () {
+                            console.warn(bug+" "+this.src + " was not found");
+                        };
+                        temp.onload = function() {spriteLoad(name,temp);}
+                        imgs.push(temp);
+                    } else if(type=="sound") {
+                        audioPaths.push(curpath + curpos[j]);
+                        newSound(curpath + curpos[j]);
+                    }
+                }
+            }
+            if(typeof curpos[j]=="object") {
+               deeper(curpos[j],type);
+            }
+        }
+        curpath = curpath.slice(0,curpath.length-addedPath.length);
+    }
+    
+    loadingCircle = new Image();
+    loadingCircle.src = "";
+    clickSound = new Audio("data:audio/x-wav;base64,UklGRowBAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YWgBAADa/6T/2/+S/x//pP769Xr4fPh5+H34evh7+Pv6gf18/QIAhQcIDxUZFR4VHhgeEx6TFgkPCgqAAnz49/X18HLu7ubo4eXc5dzj3Gjf5+Fr5G7pce759YECiwwRFBQZlxuXG5cbFRmTFo8RCAoAAPz69/X08G7udO5s7vz1dvj++nv9gP3+/wEAgwKBAogHkBEpLUNG1lzqcPV683r4eu51ZmnVV7w+qypy88fDoKAXlAqKBoUAgIeHlpYsrTu87Ot9/ZIRGxkoKDEtqionIxgZiwfj5lnaz9JN0E3QV9pd3+Tm9fUBAAQFBwWNDBEPFxSYFpsWEA9+/fP1dfNt83bz8fX9+gQFkAwPDxQPEQ+IBwIAdfjv8OLm2+HX3Nrh4+bm63n4BAUUDx0ZqyCrJSkelBGEAnb4a+5a5Njcztxc31jkZunx9QgFGhQfGa0gqCCuIKkgpRsaFIsHdfhj6dfh0NzM19Th6es=");
+    let pos = {x:cw/2-100,y:ch/2-100};
+    optionsButtons.screenSize = {x:pos.x+160,y:pos.y+12,w:50,h:20};
+    optionsButtons.sfx = {x:pos.x+125,y:pos.y+40,w:120,h:20};
+    optionsButtons.bmg = {x:pos.x+125,y:pos.y+70,w:120,h:20};
+    loadLoop();
+}
+
+function loadLoop() {
+    if(Object.keys(sprites).length == imagePaths.length && audioPaths.length == audioLoadedLength) {
+        startLoops();
+        imagePaths=[];
+        audioPaths=[];
+        imgs=[];
+    } else {
+        curCtx.fillStyle="#2d2d2d";
+        curCtx.fillRect(0,0,cw,ch);
+        text(`audio:   ${audioLoadedLength}/${audioPaths.length}`,10,30,"white",2);
+        text(`sprites: ${Object.keys(sprites).length}/${imagePaths.length}`,10,10,"white",2);
+        curCtx.setTransform(1, 0, 0, 1, Math.round(cw/2), Math.round(ch/2));
+        curCtx.rotate(loadAng);
+        loadAng+=0.1;
+        curCtx.drawImage(loadingCircle,Math.round(-8),Math.round(-8));
+        curCtx.setTransform(1, 0, 0, 1, 0, 0);
+        requestAnimationFrame(loadLoop);
+    }
+}
+
+function spriteLoad(path,image) {
+    let startpos;
+    let endpos = path.lastIndexOf(".");
+    for(let j=endpos-1;acceptableChars.includes(path[j]);j--) {startpos=j;}
+    let spriteName = path.slice(startpos,endpos)
+    let dsize=Math.max(image.width,image.height)/2;
+    sprites[spriteName] = {spr:image,drawLimitSize:dsize};
+    
+}
+
+function newSound(src) {
+    let startpos;
+    let endpos = src.lastIndexOf(".");
+    for(let j=endpos-1;acceptableChars.includes(src[j]);j--) {startpos=j;}
+    let soundName = src.slice(startpos,endpos); 
+    sounds[soundName] = {nodes:[],volNodes:[],src:src,type:"sfx",volume:1};
+    sounds[soundName].nodes = [1];
+
+    let loadingSound = new Audio();
+    loadingSound.onerror = function () {
+        console.warn(bug+" "+ src + " was not found");
+    };
+    loadingSound.src = src;
+    loadingSound.preload='auto';
+    loadingSound.addEventListener('canplaythrough', function() { 
+        audioLoadedLength++;
+     }, false);
+    sounds[soundName].nodes.push(loadingSound);
+
+    let soundNode = context.createMediaElementSource(loadingSound);
+    let gainNode = context.createGain();
+
+    soundNode.connect(gainNode);
+    gainNode.connect(sfxVolumeNode);
+
+    abuffer.push(soundNode);
+    volumeList.push(gainNode);
+    sounds[soundName].volNodes.push(volumeList.length-1);
+}
+
+function addSound(sound) {
+    let loadingSound = new Audio();
+    loadingSound.src = sound.src;
+    loadingSound.preload='auto';
+    sound.nodes.splice(sound.nodes[0],0,loadingSound);
+
+    let soundNode = context.createMediaElementSource(loadingSound);
+    let gainNode = context.createGain();
+    gainNode.gain.value=sound.volume;
+
+    soundNode.connect(gainNode);
+    gainNode.connect(sound.type=="sfx"?sfxVolumeNode:bmgVolumeNode);
+
+    abuffer.push(soundNode);
+    volumeList.push(gainNode);
+    sound.volNodes.push(volumeList.length-1);
+
+    volumeList[sound.volNodes[sound.volNodes.length-1]].gain.value = volumeList[sound.volNodes[0]].gain.value
+}
+
+
+
+function play(sound) {
+    s=sound.nodes;
+    if(s[s[0]].ended || !(s[s[0]].played.length)) {
+        s[s[0]].play();
+        s[0]++;
+        if(s[0]==s.length) {
+            s[0]=1;
+        }
+    } else {
+        addSound(sound);
+        s[s[0]].play();
+        s[0]++;
+        if(s[0]==s.length) {
+            s[0]=1;
+        }
+    }
+}
+
+function setVolume(sound,volume) {
+    for(let i=0,l=sound.volNodes.length;i<l;i++) {
+        volumeList[sound.volNodes[i]].gain.value = volume;
+    }
+}
+
+function setType(sound,newType) {
+    for(let i=0,l=sound.volNodes.length;i<l;i++) {
+        volumeList[sound.volNodes[i]].disconnect(sound.type=="sfx"?sfxVolumeNode:bmgVolumeNode);
+        volumeList[sound.volNodes[i]].connect(newType=="sfx"?sfxVolumeNode:bmgVolumeNode);
+    }
+    sound.type = newType;
+}
+
+function stop(sound) {
+    s=sound.nodes;
+    for(let i=1;i<s.length;i++) {
+        s[i].pause();
+        s[i].currentTime = 0;
+    }
+}
+
+function handleOptionsInput() {
+    let ImTierdMakemenuwork=true;
+    if(optionsMenu) {
+        if(mousePress[0]) {
+            if(rectpoint(optionsButtons.screenSize,mousePos)) {
+                if(screenSize=="1:1") {
+                    screenSize = "fit";
+                    canvasScale=0;
+                } else {
+                    screenSize = "1:1";
+                    canvasScale=1;
+                }
+            }
+            if(!rectpoint({x:cw/2,y:ch/2,w:200,h:200},mousePos)) {
+                optionsMenu=false;
+                ImTierdMakemenuwork=false;
+            }
+        }
+        if(mouseDown[0]) {
+            if(rectpoint(optionsButtons.sfx,mousePos)) {
+                volume.sfx = (mousePos.x-(optionsButtons.sfx.x-60))/120;
+            }
+            if(rectpoint(optionsButtons.bmg,mousePos)) {
+                volume.bgm = (mousePos.x-(optionsButtons.bmg.x-60))/120;
+            }
+        }
+    }
+    if(mousePos.x>cw-32&&mousePos.y<32) {
+        if(mousePress[0]&&ImTierdMakemenuwork) {
+            clickSound.play();
+            paused=true;
+            optionsMenu=!optionsMenu;
+        }
+        optionsHover = 25;
+    } else {
+        optionsHover = 0;
+    }
+    if(mousePos.x<cw-32&&mousePos.x>cw-64&&mousePos.y<32) {
+        pauseHover = 25;
+        if(mousePress[0]) {
+            clickSound.play();
+            paused=!paused;
+        }
+    } else {
+        pauseHover = 0;
+    }
+    
+}
+function addFont() {
+    var tempStyle = document.createElement("style");
+    tempStyle.innerHTML = `
+    @font-face {
+        font-family: 'PixelArial11';
+        src: url("./pixelmix.ttf") format('truetype');
+        font-weight: 900;
+        font-style: normal;
+    
+    }
+    html {font-family: 'PixelArial11' !important; font-size: 16px;}
+    `;
+    document.head.appendChild(tempStyle);
+    canvases.ctx.textBaseline = "hanging";
+    canvases.ctx.textAlign = "left";
+}
+var scaleDefault = 1;
+function img(img,x,y,angle=0,sx=scaleDefault,sy=scaleDefault) {
+    var half = img.drawLimitSize;
+    if((x+half>drawLimitLeft&&x-half<drawLimitRight&&y+half>drawLimitTop&&y-half<drawLimitBottom)||absDraw) {
+        let spr = img.spr;
+        if(angle===0&&sx===1&&sy===1) {
+            curCtx.drawImage(spr,Math.round(x+camera.x+difx-(spr.width/2)),Math.round(y+camera.y+dify-(spr.height/2)));
+        } else {
+            curCtx.setTransform(sx, 0, 0, sy, Math.round(x+camera.x+difx), Math.round(y+camera.y+dify));
+            curCtx.rotate(angle);
+            curCtx.drawImage(spr,Math.round(-spr.width/2),Math.round(-spr.height/2));
+            curCtx.setTransform(1, 0, 0, 1, 0, 0);
+        }
+    }
+}
+
+function imgIgnoreCutoff(img,x,y,angle=0,sx=1,sy=1) {
+    let spr = img.spr;
+    if(angle===0&&sx===1&&sy===1) {
+        curCtx.drawImage(spr,Math.round(x+camera.x+difx-(spr.width/2)),Math.round(y+camera.y+dify-(spr.height/2)));
+    } else {
+        curCtx.setTransform(sx, 0, 0, sy, Math.round(x+camera.x+difx), Math.round(y+camera.y+dify));
+        curCtx.rotate(angle);
+        curCtx.drawImage(spr,Math.round(-spr.width/2),Math.round(-spr.height/2));
+        curCtx.setTransform(1, 0, 0, 1, 0, 0);
+    }
+}
+
+function rect(x,y,w,h,color) {
+    curCtx.fillStyle = color;
+    curCtx.fillRect(x-(w/2)+camera.x+difx,y-(h/2)+camera.y+dify,w,h);
+}
+
+function circle(x,y,r,color) {
+    curCtx.beginPath();
+    curCtx.arc(x+camera.x+difx, y+camera.y+dify, r, 0, 2 * Math.PI, false);
+    curCtx.fillStyle = color;
+    curCtx.fill();
+}
+
+function shape(x,y,relitivePoints,color) {
+    x+=camera.x+difx;
+    y+=camera.y+dify;
+    curCtx.fillStyle = color;
+    curCtx.beginPath();
+    curCtx.moveTo(x+relitivePoints[0].x, y+relitivePoints[0].y);
+    for(let i=1,l=relitivePoints.length;i<l;i++) {
+        curCtx.lineTo(x+relitivePoints[i].x, y+relitivePoints[i].y);
+    }
+    curCtx.fill();
+}
+
+function text(txt,x,y,color="black",size=1,maxWidth=cw) {
+    txt = txt.toString();
+    curCtx.fillStyle = color;
+    curCtx.font = `${Math.round(size)*8}px PixelArial11`;
+                                                                                        //I hate text wrapping now 
+    var txtList = txt.split("\n");                                                      //split string on enters
+    for(let i=0;i<txtList.length;i++) {                                                 //go through array of strings
+        if(curCtx.measureText(txtList[i]).width>maxWidth) {                             //if the string is too big, divide up into smaller strings
+            var tempTxt = txtList[i].split(" ");                                        //split into individual words
+            var tempStr="";                                                             //string for measuring size
+            var addAmount=0;                                                            //track where in the txtList we are
+            txtList.splice(i,1);                                                        //remove the too long string
+            for(let j=0;j<tempTxt.length;j++) {                                         //go through the split up string
+                if(curCtx.measureText(tempStr + tempTxt[j] + " ").width<maxWidth) {     //if adding a word doesn't make tempStr too long, add it, other wise, add tempStr to txtList;
+                    tempStr += tempTxt[j] + " ";
+                } else {
+                    if(j==0) {tempStr+=tempTxt[j];}                                     //if we are here when j is 0, we have one word that is longer then the maxWidth, so we just draw it
+                    txtList.splice(i+addAmount,0,tempStr);                              //put tempStr in txtList
+                    addAmount++;                                                        //move the position we put the tempStr in
+                    tempStr="";                                                         //reset tempStr
+                    tempTxt.splice(0,(j==0?1:j));                                       //delete words that have been used
+                    j=-1;                                                               //make it so in the next loop, j starts at 0
+                }
+            }
+            if(tempStr.length!=0) {
+                txtList.splice(i+addAmount,0,tempStr);                                  //add any leftover text
+            }
+        }
+    }
+
+    for(let i=0;i<txtList.length;i++) {
+        curCtx.fillText(txtList[i],x+camera.x+difx,y+camera.y+dify+((i+(drawMode?1:0))*8*size+(size*i)));
+    }
+}
+
+function textWidth(txt,size=1) {
+    txt = txt.toString();
+    curCtx.font = `${Math.round(size)*8}px PixelArial11`;
+    return curCtx.measureText(txt).width;
+}
+
+function centerCameraOn(x,y) {
+    camera.x = -x+canvases.cvs.width/2;
+    camera.y = -y+canvases.cvs.height/2;
+}
+
+function moveCamera(x,y) {
+    camera.x -= y * Math.sin(camera.angle);
+    camera.y -= y * Math.cos(camera.angle);
+    camera.x -= x * Math.sin(camera.angle + 1.57079632);
+    camera.y -= x * Math.cos(camera.angle + 1.57079632);
+}
+
+function imgRotScale(x,y,angle,scale,pic,ctx) { //used for camera movement
+    ctx.setTransform(scale, 0, 0, scale, x, y);
+    ctx.rotate(angle);
+    ctx.drawImage(pic,-pic.width/2,-pic.height/2);
+    ctx.setTransform(1, 0, 0, 1, 0, 0);
+}
+
+function drawCursor() {
+    if(cursor.sprite&&mouseOnCanvas) {
+        if(cursor.alignment) {
+            canvases.ctx.drawImage(cursor.sprite.spr,mousePos.x-Math.round(cursor.sprite.spr.width/2),mousePos.y-Math.round(cursor.sprite.spr.height/2));
+        } else {
+            canvases.ctx.drawImage(cursor.sprite.spr,mousePos.x,mousePos.y);
+        }
+        cursor.show = false;
+    } else {
+        cursor.show = true;
+    }
+}
+
+function render() {
+    if(drawMode===1) {
+        imgRotScale(canvases.cvs.width/2,canvases.cvs.height/2,0,camera.zoom,canvases.buffer1cvs,canvases.ctx);
+    }
+    if(drawMode===2) {
+        imgRotScale(canvases.cvs.width/2,canvases.cvs.height/2,camera.angle,1,canvases.buffer2cvs,canvases.buffer1ctx);
+        imgRotScale(canvases.cvs.width/2,canvases.cvs.height/2,0,camera.zoom,canvases.buffer1cvs,canvases.ctx);
+    }
+}
+
+function clearCanvases() {
+    canvases.ctx.clearRect(0,0,canvases.cvs.width,canvases.cvs.height);
+    canvases.buffer1ctx.clearRect(0,0,canvases.buffer1cvs.width,canvases.buffer1cvs.height);
+    canvases.buffer2ctx.clearRect(0,0,canvases.buffer2cvs.width,canvases.buffer2cvs.height);
+}
+
+function switchDrawMode() {
+    if(camera.zoom<1) {camera.zoom=1;}
+    if(camera.angle!=0) {
+        drawMode=2;
+    } else if(camera.zoom!=1) {
+        drawMode=1;
+    } else {
+        drawMode=0;
+    }
+    switch (drawMode) {
+        case 0: curCtx = canvases.ctx; break;
+        case 1: curCtx = canvases.buffer1ctx; break;
+        case 2: curCtx = canvases.buffer2ctx; break;
+    }
+}
+
+function resizeBuffers() {
+    var tempSize = maxCvsSize/camera.zoom;
+    var tempSizeAndPadding = tempSize + (tempSize/2)
+
+    canvases.buffer2cvs.width = tempSizeAndPadding;
+    canvases.buffer2cvs.height = tempSizeAndPadding;
+    
+    if(drawMode===2) {
+        difx = (canvases.buffer2cvs.width - canvases.cvs.width)/2;
+        dify = (canvases.buffer2cvs.height - canvases.cvs.height)/2;
+    } else {
+        difx=0;
+        dify=0;
+    }
+    canvases.buffer2ctx.imageSmoothingEnabled = false;
+}
+
+function scaleCanvases() { //scales canvas by canvas scale, if scale is 0, canvas will try to fit screen
+    var style = document.getElementById("gamejsstyle");
+    if(canvasScale==0) {
+        var tempScale = Math.min(Math.floor(window.innerWidth/canvases.cvs.width),Math.floor(window.innerHeight/canvases.cvs.height));
+        tempScale=tempScale<1?1:tempScale;
+        autoScale=tempScale;
+        style.innerHTML = `#game {image-rendering:pixelated;image-rendering: crisp-edges;width:${tempScale*canvases.cvs.width}px;cursor: ${cursor.show?"crosshair":"none"};}`;
+    } else {
+        style.innerHTML = `#game {image-rendering:pixelated;image-rendering: crisp-edges;width:${Math.floor(canvasScale*canvases.cvs.width)}px;cursor: ${cursor.show?"crosshair":"none"};}`;
+    }
+}
+
+function drawButtons() {
+    let pos = {x:cw-16,y:16}; 
+    //options
+    rect(pos.x,pos.y,34,34,"#9c9c9c");
+    let c = optionsHover+45;
+    rect(pos.x,pos.y,32,32,`rgb(${c},${c},${c})`);
+    c = optionsHover + 69;
+    let cc = `rgb(${c},${c},${c})`;
+    rect(pos.x,pos.y-6,26,4,cc);
+        rect(pos.x-6,pos.y-6,4,8,cc);
+    rect(pos.x,pos.y+6,26,4,cc);
+        rect(pos.x+11,pos.y+6,4,8,cc);
+    //pause
+    pos.x-=33;
+    rect(pos.x,pos.y,34,34,"#9c9c9c");
+    c = pauseHover+45;
+    rect(pos.x,pos.y,32,32,`rgb(${c},${c},${c})`);
+    c = pauseHover + 69;
+    cc = `rgb(${c},${c},${c})`;
+    if(paused) {
+        shape(pos.x,pos.y,[{x:-7,y:-10},{x:-7,y:10},{x:10,y:0}],cc);
+    } else {
+        rect(pos.x+6,pos.y,6,20,cc);
+        rect(pos.x-6,pos.y,6,20,cc);
+    }
+}
+
+function drawOptionsMenu() {
+    if(optionsMenu) {
+        let pos = {x:cw/2-100,y:ch/2-100};
+        rect(cw/2,ch/2,200,200,"#242424");
+        text("Screen Size:",pos.x+2,pos.y+2,"white",2);
+            let b = optionsButtons.screenSize;
+            rect(b.x,b.y,b.w,b.h,"#444444");
+            text(screenSize,pos.x+145,pos.y+4,"white",2);
+        text("sfx",pos.x+2,pos.y+30,"white",2);
+            b = optionsButtons.sfx;
+            rect(b.x,b.y,b.w,b.h-10,"#444444");
+            rect((b.x-60)+(volume.sfx*120),b.y,8,20,"#444444");
+        text("bmg",pos.x+2,pos.y+60,"white",2);
+            b = optionsButtons.bmg;
+            rect(b.x,b.y,b.w,b.h-10,"#444444");
+            rect((b.x-60)+(volume.bgm*120),b.y,8,20,"#444444");
+    }
+}
+var k={a:65,b:66,c:67,d:68,e:69,f:70,g:71,h:72,i:73,j:74,k:75,l:76,m:77,n:78,o:79,p:80,q:81,r:82,s:83,t:84,u:85,v:86,w:87,x:88,y:89,z:90,0:48,1:49,2:50,3:51,4:52,5:53,6:54,7:55,8:56,9:57,BACKTICK:192,MINUS:189,EQUALS:187,OPENSQUARE:219,ENDSQUARE:221,SEMICOLON:186,SINGLEQUOTE:222,BACKSLASH:220,COMMA:188,PERIOD:190,SLASH:191,ENTER:13,BACKSPACE:8,TAB:9,CAPSLOCK:20,SHIFT:16,CONTROL:17,ALT:18,META:91,LEFTBACKSLASH:226,ESCAPE:27,HOME:36,END:35,PAGEUP:33,PAGEDOWN:34,DELETE:46,INSERT:45,PAUSE:19,UP:38,DOWN:40,LEFT:37,RIGHT:39,CONTEXT:93,SPACE:32,F1:112,F2:113,F3:114,F4:115,F5:116,F6:117,F7:118,F8:119,F9:120,F10:121,F11:122,F12:123};
+var keyPress = [];
+var keyDown = [];
+var mousePress = [];
+var mouseDown = [];
+var scroll = 0;
+var mousePos = {
+    x:0,
+    y:0
+}
+var preventedEvents = [false,true,true];
+
+function addListenersTo(elementToListenTo) {
+    window.addEventListener("keydown",kdown);
+    window.addEventListener("keyup",kup);
+    elementToListenTo.addEventListener("mousedown",mdown);
+    elementToListenTo.addEventListener("mouseup",mup);
+    elementToListenTo.addEventListener("mousemove",mmove);
+    elementToListenTo.addEventListener("contextmenu",cmenu);
+    elementToListenTo.addEventListener("wheel",scrl);
+}
+function removeListenersFrom(elementToListenTo) {
+    window.removeEventListener("keydown",kdown);
+    window.removeEventListener("keyup",kup);
+    elementToListenTo.removeEventListener("mousedown",mdown);
+    elementToListenTo.removeEventListener("mouseup",mup);
+    elementToListenTo.removeEventListener("mousemove",mmove);
+    elementToListenTo.removeEventListener("contextmenu",cmenu);
+    elementToListenTo.removeEventListener("wheel",scrl);
+}
+function resetInput() {
+    for(var i=0;i<keyPress.length;i++){if(keyPress[i]){keyPress[i]=0}}
+    for(var i=0;i<mousePress.length;i++){if(mousePress[i]){mousePress[i]=0}}
+    scroll=0;
+}
+function kdown(e) {
+    var h=e.keyCode;
+    keyPress[h]=keyPress[h]==[][[]]?1:0;
+    keyDown[h]=1;
+    if(preventedEvents[0]) {e.preventDefault()}
+}
+function kup(e) {
+    var h=e.keyCode;
+    delete keyPress[h];
+    delete keyDown[h];
+}
+function mdown(e) {
+    var h=e.button;
+    mousePress[h]=mousePress[h]==[][[]]?1:0;
+    mouseDown[h]=1;
+    if(preventedEvents[1]) {e.preventDefault()}
+}
+function mup(e) {
+    var h=e.button;
+    delete mousePress[h];
+    delete mouseDown[h];
+}
+function mmove(e) {
+    mousePos.x=e.offsetX/(!canvasScale?autoScale:canvasScale);
+    mousePos.y=e.offsetY/(!canvasScale?autoScale:canvasScale);    
+}
+function cmenu(e) {
+    if(preventedEvents[1]) {e.preventDefault()}
+}
+function scrl(e) {
+    scroll+=-1*(e.deltaY/100);
+    if(preventedEvents[2]) {e.preventDefault()}
+}
+function dist(point1,point2) {
+    let one = (point2.x - point1.x);
+    let two = (point2.y - point1.y);
+    return Math.sqrt((one*one)+(two*two));
+}
+
+function circlecircle(circle1,circle2) {
+    if( dist(circle1,circle2) < (circle1.r + circle2.r)) {
+		return true;
+	} else {
+        return false;
+    }
+}
+
+function circlepoint(circle,point) {
+    if( dist(circle,point) < circle.r) {
+		return true;
+	} else {
+        return false;
+    }
+}
+
+function rectrect(rect1,rect2) {
+    if(rect1.x + rect1.w/2 >= rect2.x - rect2.w/2 &&
+       rect1.x - rect1.w/2 <= rect2.x + rect2.w/2 &&
+       rect1.y + rect1.h/2 >= rect2.y - rect2.h/2 &&
+       rect1.y - rect1.h/2 <= rect2.y + rect2.h/2) {
+        return true;
+    } else {
+        return false;
+    }
+}
+
+function rectpoint(rect,point) {
+    if(rect.x + rect.w/2 >= point.x &&
+       rect.x - rect.w/2 <= point.x &&
+       rect.y + rect.h/2 >= point.y &&
+       rect.y - rect.h/2 <= point.y ) {
+        return true;
+    } else {
+        return false;
+    }
+}
+
+function circlerect(circle,rect) { //credit: https://yal.cc/rectangle-circle-intersection-test/
+    let rectHalfWidth  = rect.w/2;
+    let rectHalfHeight = rect.h/2;
+    let deltaX = circle.x - Math.max(rect.x - rectHalfWidth, Math.min(circle.x, rect.x + rectHalfWidth));
+    let deltaY = circle.y - Math.max(rect.y - rectHalfHeight, Math.min(circle.y, rect.y + rectHalfHeight));
+    return (deltaX * deltaX + deltaY * deltaY) < (circle.r * circle.r);
+}
+
+function circleOnSideRect(circle,rect) {
+    let rectHalfWidth  = rect.w/2;
+    let rectHalfHeight = rect.h/2;
+    let left   = rect.x - rectHalfWidth;
+    let right  = rect.x + rectHalfWidth;
+    let top    = rect.y - rectHalfHeight;
+    let bottom = rect.y + rectHalfHeight;
+    let cx = circle.x;
+    let cy = circle.y;
+    if(cy < top && cx > left && cx < right) { // top side
+        return 0;
+    } else if(cy > bottom && cx > left && cx < right) { // bottom side
+        return 2;
+    } else if (cx < left && cy > top && cy < bottom) { // left side
+        return 3;
+    } else if (cx > right && cy > top && cy < bottom) { // right side
+        return 1;
+    } else {
+        let returnValue=0; // 0 = top, 1 = right, 2 = bottom, 3 = left
+        let topleft = dist (circle,{x:left,y:top});
+        let topright = dist (circle,{x:right,y:top});
+        let bottomleft = dist (circle,{x:left,y:bottom});
+        let bottomright = dist (circle,{x:right,y:bottom});
+        switch(Math.min(topleft,topright,bottomleft,bottomright)) { // find what corner the cricle is closer to, then determine what side it is closer to
+            case topleft:
+                var m = slope(rect,{x:left,y:top});
+                var mperp = -(1/m);
+                var b = yIntercept(rect,m);
+                var bperp = yIntercept(circle,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 3;} else {returnValue = 0;}
+                break;
+            case topright:
+                var m = slope(rect,{x:right,y:top});
+                var mperp = -(1/m);
+                var b = yIntercept(rect,m);
+                var bperp = yIntercept(circle,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 0;} else {returnValue = 1;}
+                break;
+            case bottomleft:
+                var m = slope(rect,{x:left,y:bottom});
+                var mperp = -(1/m);
+                var b = yIntercept(rect,m);
+                var bperp = yIntercept(circle,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 3;} else {returnValue = 2;}
+                break;
+            case bottomright:
+                var m = slope(rect,{x:right,y:bottom});
+                var mperp = -(1/m);
+                var b = yIntercept(rect,m);
+                var bperp = yIntercept(circle,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 2;} else {returnValue = 1;}
+                break;
+        }
+        return returnValue;
+    }
+}
+
+function rectOnSideRect(rect1,rect2) {
+    let rectHalfWidth2  = rect2.w/2;
+    let rectHalfHeight2 = rect2.h/2;
+    let left2   = rect2.x - rectHalfWidth2;
+    let right2 = rect2.x + rectHalfWidth2;
+    let top2   = rect2.y - rectHalfHeight2;
+    let bottom2 = rect2.y + rectHalfHeight2;
+
+    let rectHalfWidth1  = rect1.w/2;
+    let rectHalfHeight1 = rect1.h/2;
+    let rx1 = rect1.x;
+    let ry1 = rect1.y;
+    let left1   = rx1 - rectHalfWidth1;
+    let right1 = rx1 + rectHalfWidth1;
+    let top1   = ry1 - rectHalfHeight1;
+    let bottom1 = ry1 + rectHalfHeight1;
+    // find what point is closer to the rectangle 
+    let topleft1 = dist (rect2,{x:left1,y:top1});
+    let topright1 = dist (rect2,{x:right1,y:top1});
+    let bottomleft1 = dist (rect2,{x:left1,y:bottom1});
+    let bottomright1 = dist (rect2,{x:right1,y:bottom1});
+    let topmiddle1 = dist (rect2,{x:rx1,y:top1});
+    let rightmiddle1 = dist (rect2,{x:right1,y:ry1});
+    let bottommiddle1 = dist (rect2,{x:rx1,y:bottom1});
+    let leftmiddle1 = dist (rect2,{x:left1,y:ry1});
+    let cx = rx1;
+    let cy = ry1;
+    switch(Math.min(topleft1,topright1,bottomleft1,bottomright1,topmiddle1,rightmiddle1,bottommiddle1,leftmiddle1)) {
+        //set the point we are testing to the closest point to the rectangle
+        case topleft1:
+            cx -= rect1.w/2;
+            cy -= rect1.h/2;
+            break;
+        case topright1:
+            cx += rect1.w/2;
+            cy -= rect1.h/2;
+            break;
+        case bottomleft1:
+            cx -= rect1.w/2;
+            cy += rect1.h/2;
+            break;
+        case bottomright1:
+            cx += rect1.w/2;
+            cy += rect1.h/2;
+            break;
+        case topmiddle1:
+            cy -= rect1.h/2;
+            break;
+        case rightmiddle1:
+            cx += rect1.w/2;
+            break;
+        case bottommiddle1:
+            cy += rect1.h/2;
+            break;
+        case leftmiddle1:
+            cx -= rect1.w/2;
+            break;
+    }
+    if(cy < top2 && cx > left2 && cx < right2) { // top side
+        return 0;
+    } else if(cy > bottom2 && cx > left2 && cx < right2) { // bottom side
+        return 2;
+    } else if (cx < left2 && cy > top2 && cy < bottom2) { // left side
+        return 3;
+    } else if (cx > right2 && cy > top2 && cy < bottom2) { // right side
+        return 1;
+    } else {
+        let returnValue=0; // 0 = top, 1 = right, 2 = bottom, 3 = left
+        let determiningPoint = {x:cx,y:cy};
+        let topleft = dist (determiningPoint,{x:left2,y:top2});
+        let topright = dist (determiningPoint,{x:right2,y:top2});
+        let bottomleft = dist (determiningPoint,{x:left2,y:bottom2});
+        let bottomright = dist (determiningPoint,{x:right2,y:bottom2});
+        switch(Math.min(topleft,topright,bottomleft,bottomright)) { // find what corner the point is closer to, then determine what side it is closer to
+            case topleft:
+                var m = slope(rect2,{x:left2,y:top2});
+                var mperp = -(1/m);
+                var b = yIntercept(rect2,m);
+                var bperp = yIntercept(determiningPoint,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 3;} else {returnValue = 0;}
+                break;
+            case topright:
+                var m = slope(rect2,{x:right2,y:top2});
+                var mperp = -(1/m);
+                var b = yIntercept(rect2,m);
+                var bperp = yIntercept(determiningPoint,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 0;} else {returnValue = 1;}
+                break;
+            case bottomleft:
+                var m = slope(rect2,{x:left2,y:bottom2});
+                var mperp = -(1/m);
+                var b = yIntercept(rect2,m);
+                var bperp = yIntercept(determiningPoint,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 3;} else {returnValue = 2;}
+                break;
+            case bottomright:
+                var m = slope(rect2,{x:right2,y:bottom2});
+                var mperp = -(1/m);
+                var b = yIntercept(rect2,m);
+                var bperp = yIntercept(determiningPoint,mperp);
+                var mid = POI(m,b,mperp,bperp);
+                if(cx<mid) {returnValue = 2;} else {returnValue = 1;}
+                break;
+        }
+        return returnValue;
+    }
+}
+
+function slope(point1,point2) {
+    return ((point2.y-point1.y)/(point2.x-point1.x));
+}
+
+function yIntercept(point,slope) {
+    return point.y - (slope * point.x);
+}
+
+function POI(m1,b1,m2,b2) {
+    x = (b2 - b1) / (m1 - m2);
+    return x;
+    //y = m1 * x + b1;
+}
+
+function ifRectOnEdgeBounce(rect) {
+    let rx = rect.x;
+    let ry = rect.y;
+    let rw = rect.w/2;
+    let rh = rect.h/2;
+    if(rx+rw>edge.right) {
+        rect.v.x *= -1;
+        rect.x = edge.right-rw;
+    }
+    if(rx-rw<edge.left) {
+        rect.v.x *= -1;
+        rect.x = edge.left+rw;
+    }
+    if(ry+rh>edge.bottom) {
+        rect.v.y *= -1;
+        rect.y = edge.bottom-rh;
+    }
+    if(ry-rh<edge.top) {
+        rect.v.y *= -1;
+        rect.y = edge.top+rh;
+    }
+}
+
+function ifCircleOnEdgeBounce(circle) {
+    let cx = circle.x;
+    let cy = circle.y;
+    let cr = circle.r;
+    if(cx+cr>edge.right) {
+        circle.v.x *= -1;
+        circle.x = edge.right-cr;
+    }
+    if(cx-cr<edge.left) {
+        circle.v.x *= -1;
+        circle.x = edge.left+cr;
+    }
+    if(cy+cr>edge.bottom) {
+        circle.v.y *= -1;
+        circle.y = edge.bottom-cr;
+    }
+    if(cy-cr<edge.top) {
+        circle.v.y *= -1;
+        circle.y = edge.top+cr;
+    }
+}
+// create globals
+var canvases={cvs:null,ctx:null,buffer1cvs:null,buffer1ctx:null,buffer2cvs:null,buffer2ctx:null}, // visable and hidden canvases
+cw, // canvas width
+ch, // canvas height
+camera={zoom:1,angle:0,x:0,y:0}, // affects how everything is drawn
+updateFPS=60,
+gameStarted=false,
+drawMode=0, // 0=normal, 1=zoomed, 2=zoomed/rotated, set automatically depending on camera
+absDraw=false,
+curCtx, // what canvas to draw to
+maxCvsSize, // used by second buffer
+canvasScale=1,
+difx=0, // offsets for drawing
+dify=0,
+seperateInputLoop=true,
+edge={top:null,bottom:null,left:null,right:null}, // used by if___OnEdgeBounce, set to canvas size at setup, can be changed whenever
+drawLimitLeft,
+drawLimitRight,
+drawLimitTop,
+drawLimitBottom,
+sizeDif,
+bug="\uD83D\uDC1B",
+loadingCircle,
+loadAng=0,
+optionsHover=0,
+pauseHover=0,
+optionsMenu=false,
+optionsButtons={},
+clickSound,
+paused=false,
+screenSize="1:1",
+autoScale=1,
+
+images=[], // put image paths here
+imagePaths=[],
+imgs=[],
+sprites={}, // loaded images
+
+audio=[], // put audio paths here
+audioPaths=[],
+sounds={}, // loaded sounds
+abuffer = [], // audio nodes shoved here
+volumeList = [], // gain nodes shoved here
+audioLoadedLength=0,
+volume={sfx:1,bgm:1};
+/* options
+    future:
+        modefileable checklist
+        key bindings
+*/
+
+cursor = {sprite:null,alignment:1,show:true}, // 0=topleft, 1=centered
+mouseOnCanvas=false;
+
+const acceptableChars="qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM1234567890_-. ";//for image names
+
+const AudioContext = window.AudioContext||window.webkitAudioContext;
+var context;
+var sfxVolumeNode;
+var bmgVolumeNode;
+
+document.getElementById("game").onmouseout = function()   {mouseOnCanvas = false;}
+document.getElementById("game").onmouseover = function()   {mouseOnCanvas = true;}
+
+//setup canvases and input
+function setup(physicsFPS) {
+    updateFPS = physicsFPS;
+    
+    canvases.cvs = document.getElementById("game");
+    canvases.ctx = canvases.cvs.getContext("2d", { alpha: false });
+
+    canvases.cvs.onmousedown = function () {if(!gameStarted){loadImagesAndSounds();gameStarted=true;}}
+
+    createCanvas("buffer1");
+    createCanvas("buffer2");
+
+    canvases.ctx.imageSmoothingEnabled = false;
+    canvases.buffer1ctx.imageSmoothingEnabled = false;
+    canvases.buffer2ctx.imageSmoothingEnabled = false;
+
+    maxCvsSize=Math.max(canvases.cvs.width,canvases.cvs.height);
+    sizeDif=maxCvsSize-Math.min(canvases.cvs.width,canvases.cvs.height);
+    cw=canvases.cvs.width;
+    ch=canvases.cvs.height;
+    
+    edge={top:0,bottom:ch,left:0,right:cw};
+
+    addFont();
+    addStyle();
+
+    addListenersTo(canvases.cvs);
+
+    curCtx = canvases.ctx;
+    requestAnimationFrame(startButton);
+    function startButton() {
+        curCtx.fillStyle="#2d2d2d";
+        curCtx.fillRect(0,0,cw,ch);//debugger;
+        circle(cw/2,ch/2,27,"#066312");
+        circle(cw/2,ch/2,23,"#149124");
+        shape(cw/2,ch/2,[{x:-7,y:-15},{x:-7,y:15},{x:15,y:0}],"#47f55d");
+        if(!gameStarted) {requestAnimationFrame(startButton);}
+    }
+}
+
+function drawLoop() {
+    cw=canvases.cvs.width;
+    ch=canvases.cvs.height;
+    scaleCanvases();
+
+    switchDrawMode();
+    
+    resizeBuffers();
+
+    clearCanvases();
+
+    var limitModifyer = 0;
+    if(drawMode==2) {limitModifyer=canvases.buffer2cvs.width-maxCvsSize;}
+    drawLimitLeft   = -camera.x - (drawMode==2?sizeDif:0) - limitModifyer;
+    drawLimitRight  = -camera.x + maxCvsSize + (drawMode==2?sizeDif:0) + limitModifyer;
+    drawLimitTop    = -camera.y -(drawMode==2?sizeDif:0) - limitModifyer;
+    drawLimitBottom = -camera.y + maxCvsSize + (drawMode==2?sizeDif:0) + limitModifyer;
+
+    draw();
+    
+    render();
+
+    curCtx=canvases.ctx;
+    difx=0;dify=0;
+    var camCache = {x:camera.x,y:camera.y};
+    var drawModeCache = drawMode;
+    camera.x=0;camera.y=0;
+    drawMode=0;
+    absDraw=true;
+    absoluteDraw();
+    absDraw=false;
+
+    drawButtons();
+    drawOptionsMenu();
+    drawCursor();
+
+    drawMode=drawModeCache;
+
+    camera.x = camCache.x;
+    camera.y = camCache.y;
+
+    requestAnimationFrame(drawLoop);
+}
+
+function updateLoop() {
+    if(seperateInputLoop==false) {
+        handleOptionsInput();
+    }
+    sfxVolumeNode.gain.value = volume.sfx;
+    bmgVolumeNode.gain.value = volume.bgm;
+    if(!paused) {
+        update();
+    }
+
+    if(seperateInputLoop==false) {
+        resetInput();
+    }
+}
+
+
+function inputLoop() {
+    handleOptionsInput();
+    if(!paused) {
+        input();
+    }
+
+    resetInput();
 }
\ No newline at end of file
diff --git a/scripts/index.js b/docs/scripts/index.js
similarity index 86%
rename from scripts/index.js
rename to docs/scripts/index.js
index a9e60f6..6c8fd79 100644
--- a/scripts/index.js
+++ b/docs/scripts/index.js
@@ -1,29 +1,29 @@
-images = [
-    ""
-];
-
-audio = [
-    ""
-];
-
-function update() {
-
-}
-
-function input() {
-
-}
-
-function draw() {
-
-}
-
-function absoluteDraw() {
-
-}
-
-function onAssetsLoaded() {
-
-}
-
+images = [
+    ""
+];
+
+audio = [
+    ""
+];
+
+function update() {
+
+}
+
+function input() {
+
+}
+
+function draw() {
+
+}
+
+function absoluteDraw() {
+
+}
+
+function onAssetsLoaded() {
+
+}
+
 setup(60);
\ No newline at end of file