Rocketer Utilities

Adds a lot new settings to the game https://rocketer.glitch.me/

// ==UserScript==
// @name         Rocketer Utilities
// @namespace    http://tampermonkey.net/
// @version      1.4
// @description  Adds a lot new settings to the game https://rocketer.glitch.me/
// @author       DB423 (Impsaccrain)
// @match        http*://rocketer.glitch.me/*
// @icon         
// @grant        none
// @license      DISTRIBUTION
// ==/UserScript==

(function() {
    'use strict';

    localStorage.RocketerUtilities = localStorage.RocketerUtilities || JSON.stringify({ Options: { AuraState: 1, AutoRespawn: 0, Theme: 0 }, version: 1.4 });

    if (JSON.parse(localStorage.RocketerUtilities).version != 1.4) {
        localStorage.RocketerUtilities = JSON.stringify({ Options: { AuraState: 1, AutoRespawn: 0, Theme: 0 }, version: 1.4 });
    };

    const Utils = JSON.parse(localStorage.RocketerUtilities);
    const Options = Utils.Options;

    var settings = document.getElementById('settingsPopup');
    var closesettingspopup = document.getElementById('closeSettingsPopup');
    var themes = document.getElementById('theme');
    changetheme = function (selectObject) {
        colortheme = selectObject.value;
        let newutils = JSON.parse(localStorage.RocketerUtilities);
        newutils.Options.Theme = document.getElementById('theme').selectedIndex;
        newutils = JSON.stringify(newutils);
        localStorage.RocketerUtilities = newutils;
    }
    var playershape = 'circle';

    var whitetheme = document.createElement('option');
    whitetheme.value = 'whitetheme';
    whitetheme.textContent = 'White';
    themes.appendChild(whitetheme);

    var simplistic = document.createElement('option');
    simplistic.value = 'simplistic';
    simplistic.textContent = 'Simplistic';
    themes.appendChild(simplistic);

    for (let i = 3; i < 15; i++) {
        shapecolors[i].whitetheme = {
            color: "#FFFFFF",
            outline: "#FFFFFF",
            hitcolor: "#FFFFFF",
            hitoutline: "#FFFFFF",
        };
        shapecolors[i].simplistic = {
            color: shapecolors[i].default.color,
            outline: shapecolors[i].default.color,
            hitcolor: shapecolors[i].default.hitcolor,
            hitoutline: shapecolors[i].default.hitcolor,
        };
    };

    var chatstate = true;
    function toggleChat() {
        let chatinput = document.getElementById('chat');
        chatstate = !chatstate;
        if (!chatstate) {
            chatinput.style.display = 'none';
        } else {
            chatinput.style.display = 'block';
        };
    };

    var aurastate = Options.AuraState == 1 ? true : false;
    function toggleAura() {
        aurastate = !aurastate
        Options.AuraState = aurastate ? 1 : 0;
        localStorage.RocketerUtilities = JSON.stringify(Utils);
    };
    var autorespawn = Options.AutoRespawn == 1 ? true : false;
    function toggleAutoRespawn() {
        autorespawn = !autorespawn
        Options.AutoRespawn = autorespawn ? 1 : 0;
        localStorage.RocketerUtilities = JSON.stringify(Utils);
    };

    var chattoggle = document.createElement('label');
    chattoggle.className = 'switch';
    var cti = document.createElement('input');
    cti.type = 'checkbox';
    cti.checked = true;
    cti.onclick = toggleChat;
    var cts = document.createElement('span');
    cts.className = 'slider round';
    chattoggle.appendChild(cti);
    chattoggle.appendChild(cts);

    var auratoggle = document.createElement('label');
    auratoggle.className = 'switch';
    var ati = document.createElement('input');
    ati.type = 'checkbox';
    ati.checked = Options.AuraState == 1 ? false : true;
    ati.onclick = toggleAura;
    var ats = document.createElement('span');
    ats.className = 'slider round';
    auratoggle.appendChild(ati);
    auratoggle.appendChild(ats);

    var respawntoggle = document.createElement('label');
    respawntoggle.className = 'switch';
    var rti = document.createElement('input');
    rti.type = 'checkbox';
    rti.checked = Options.AutoRespawn == 1 ? true : false;
    rti.onclick = toggleAutoRespawn;
    var rts = document.createElement('span');
    rts.className = 'slider round';
    respawntoggle.appendChild(rti);
    respawntoggle.appendChild(rts);

    var utilities = document.createElement('span');
    utilities.style = 'font-weight: 700; font-size: 25px;';
    utilities.textContent = 'Rocketer Utilities';

    settings.appendChild(document.createElement('br'));
    settings.appendChild(document.createElement('br'));
    settings.appendChild(utilities);
    settings.appendChild(document.createElement('hr'));
    settings.appendChild(document.createTextNode("Chat "));
    settings.appendChild(chattoggle);
    settings.appendChild(document.createElement('br'));
    settings.appendChild(document.createElement('br'));
    settings.appendChild(document.createTextNode("Invisible auras "));
    settings.appendChild(auratoggle);
    settings.appendChild(document.createElement('br'));
    settings.appendChild(document.createElement('br'));
    settings.appendChild(document.createTextNode("Auto-respawn "));
    settings.appendChild(respawntoggle);

    var playershapediv = document.createElement('div');
    playershapediv.appendChild(document.createElement('br'));
    playershapediv.appendChild(document.createTextNode('Player shape: '));
    var playershapeselect = document.createElement('select');
    playershapeselect.id = 'player-shape-select'
    playershapeselect.style.backgroundColor = 'rgba(0, 0, 0, 0.5)';
    playershapeselect.style.color = 'white';
    playershapeselect.style.borderRadius = '7px';
    playershapeselect.style.width = '100px';
    playershapeselect.style.height = '30px';

    var playershapecircle = document.createElement('option');
    playershapecircle.value = 'circle';
    playershapecircle.textContent = 'Circle';
    playershapeselect.appendChild(playershapecircle);

    var playershapesquare = document.createElement('option');
    playershapesquare.value = 'square';
    playershapesquare.textContent = 'Square';
    playershapeselect.appendChild(playershapesquare);

    var playershapetriangle = document.createElement('option');
    playershapetriangle.value = 'triangle';
    playershapetriangle.textContent = 'Triangle';
    playershapeselect.appendChild(playershapetriangle);

    playershapediv.appendChild(playershapeselect);

    setTimeout(function() {

    let changelogDisplayElement = document.getElementById('changelogDisplay');
    changelogDisplayElement.appendChild(document.createElement('br'));
    changelogDisplayElement.appendChild(document.createElement('br'));
    let rucspan = document.createElement('span');
    rucspan.id = 'rocketer-utils-changelog';
    let ruc = document.createTextNode('ROCKETER UTILITIES CHANGELOG - 1.4 - 6 December 2023');
    rucspan.style.color = 'orange';
    rucspan.appendChild(ruc);
    let rucp = document.createElement('p');
    function cct(text, br) {
        rucp.appendChild(document.createTextNode(text));
        if (br) {
            rucp.appendChild(document.createElement('br'));
        };
    };
    cct('- BUGFIX: Fixed the chat doubling itself', true);
    cct('- FEATURE: Added the smooth chat transitions (they didn\'t appear)', false)
    rucspan.appendChild(rucp);
    changelogDisplayElement.appendChild(rucspan);
    themes.selectedIndex = Options.Theme;
    changetheme(themes);

    }, 500);

    function newdrawplayer(canvas, object, fov, spawnProtect, playercolor, playeroutline, eternal, objectangle){//only barrels and body (no heath bars, names, and chats)
    const CRTP = document.getElementById('theme').value;
    //objectangle refers to angle rotated before triggering this function
    //fov is clientFovMultiplier for ctx, hctx is 1
      canvas.lineJoin = "round"; //make nice round corners
      //draw assets below body, e.g. rammer body base
      for (let assetID in object.assets){
        var asset = object.assets[assetID];
        if (asset.type == "under") {
          if (('angle' in asset) && asset.angle != 0) {
            canvas.rotate(asset.angle * Math.PI / 180);
          }
          canvas.translate(
            (object.width / fov) * asset.x,
            (object.width / fov) * asset.y
          );
          canvas.fillStyle = asset.color;
          canvas.strokeStyle = asset.outline;
          canvas.lineWidth = asset.outlineThickness / fov;
          if (asset.sides == 0) {
            canvas.beginPath();
            canvas.arc(0, 0, (object.width / fov) * asset.size, 0, 2 * Math.PI);
            canvas.fill();
            if (CRTP != 'simplistic') {
            canvas.stroke();
            };
          } else {
            canvas.beginPath();
            let baseSides = asset.sides;
            canvas.moveTo((object.width / fov) * asset.size, 0);
            for (let i = 1; i <= baseSides; i++) {
              canvas.lineTo((object.width / fov) * asset.size * Math.cos((i * 2 * Math.PI) / baseSides), (object.width / fov) * asset.size * Math.sin((i * 2 * Math.PI) / baseSides));
            }
            canvas.fill();
            if (CRTP != 'simplistic') {
            canvas.stroke();
            };
          }
          canvas.translate(
            (-object.width / fov) * asset.x,
            (-object.width / fov) * asset.y
          );
          if (('angle' in asset) && asset.angle != 0) {
            canvas.rotate(-asset.angle * Math.PI / 180);
          }
        }
      }

      //draw barrel
      canvas.lineWidth = 4 / fov;
      //weapon barrels
      for (let barrel in object.barrels){
        let thisBarrel = object.barrels[barrel];
        canvas.rotate((thisBarrel.additionalAngle * Math.PI) / 180); //rotate to barrel angle
        canvas.fillStyle = bodyColors.barrel.col;
        canvas.strokeStyle = bodyColors.barrel.outline;
        if (spawnProtect == "yes") {
          //if have spawn protection
          canvas.fillStyle = bodyColors.barrel.hitCol;
          canvas.strokeStyle = bodyColors.barrel.hitOutline;
        }
        //bullet barrel
        //note: barrelHeightChange refers to reduction in barrel height for barrel animation when shooting
        if (thisBarrel.barrelType == "bullet") {
          drawBulletBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //drone barrel
        else if (thisBarrel.barrelType == "drone") {
          drawDroneBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //trap barrel
        else if (thisBarrel.barrelType == "trap") {
          drawTrapBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //mine barrel
        else if (thisBarrel.barrelType == "mine") {
          drawMineBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //minion barrel
        else if (thisBarrel.barrelType == "minion") {
          drawMinionBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        canvas.rotate((-thisBarrel.additionalAngle * Math.PI) / 180); //rotate back
      }

      //draw player body
      canvas.fillStyle = playercolor;
      canvas.strokeStyle = playeroutline;
      if (eternal == "no") {
        //not a tier 6 tank
        canvas.beginPath();
        canvas.arc(0, 0, object.width / fov, 0, 2 * Math.PI);
        canvas.fill();
        if (CRTP != 'simplistic') {
            canvas.stroke();
        };
      } else {
        //if a tier 6 tank
        canvas.beginPath();
        let baseSides = 6;
        canvas.moveTo((object.width / fov), 0);
        for (var i = 1; i <= baseSides; i++) {
          canvas.lineTo((object.width / fov) * Math.cos((i * 2 * Math.PI) / baseSides), (object.width / fov) * Math.sin((i * 2 * Math.PI) / baseSides));
        }
        canvas.fill();
        if (CRTP != 'simplistic') {
            canvas.stroke();
        };
      }

      //barrels in body upgrade
      for (let barrel in object.bodybarrels){
        let thisBarrel = object.bodybarrels[barrel];
        canvas.rotate(thisBarrel.additionalAngle - objectangle); //rotate to barrel angle
        canvas.fillStyle = bodyColors.barrel.col;
        canvas.strokeStyle = bodyColors.barrel.outline;
        if (spawnProtect == "yes") {
          //if have spawn protection
          canvas.fillStyle = bodyColors.barrel.hitCol;
          canvas.strokeStyle = bodyColors.barrel.hitOutline;
        }
        //bullet barrel
        if (thisBarrel.barrelType == "bullet") {
          drawBulletBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //drone barrel
        else if (thisBarrel.barrelType == "drone") {
          drawDroneBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //trap barrel (doesnt exist atm)
        else if (thisBarrel.barrelType == "trap") {
          drawTrapBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //mine barrel (doesnt exist atm)
        else if (thisBarrel.barrelType == "mine") {
          drawMineBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        //minion barrel (doesnt exist atm)
        else if (thisBarrel.barrelType == "minion") {
          drawMinionBarrel(canvas,thisBarrel.x,thisBarrel.barrelWidth,thisBarrel.barrelHeight,thisBarrel.barrelHeightChange,fov)
        }
        canvas.rotate(-thisBarrel.additionalAngle + objectangle); //rotate back
      }
      //draw turret base
      if ('turretBaseSize' in object){
        canvas.fillStyle = bodyColors.barrel.col;
        canvas.strokeStyle = bodyColors.barrel.outline;
        canvas.beginPath();
        canvas.arc(0, 0, (object.width / clientFovMultiplier) * object.turretBaseSize, 0, 2 * Math.PI);
        canvas.fill();
        if (CRTP != 'simplistic') {
            canvas.stroke();
        };
      }

      //draw assets above body, e.g. aura assets
      for (let assetID in object.assets){
        var asset = object.assets[assetID];
        if (asset.type == "above") {
          if (('angle' in asset) && asset.angle != 0) {
            canvas.rotate(asset.angle * Math.PI / 180);
          }
          canvas.translate(
            (object.width / fov) * asset.x,
            (object.width / fov) * asset.y
          );
          canvas.fillStyle = asset.color;
          canvas.strokeStyle = asset.outline;
          canvas.lineWidth = asset.outlineThickness / fov;
          if (asset.sides == 0) {
            canvas.beginPath();
            canvas.arc(0, 0, (object.width / fov) * asset.size, 0, 2 * Math.PI);
            canvas.fill();
            if (CRTP != 'simplistic') {
                canvas.stroke();
            };
          } else {
            canvas.beginPath();
            let baseSides = asset.sides;
            canvas.moveTo((object.width / fov) * asset.size, 0);
            for (var i = 1; i <= baseSides; i++) {
              canvas.lineTo((object.width / fov) * asset.size * Math.cos((i * 2 * Math.PI) / baseSides), (object.width / fov) * asset.size * Math.sin((i * 2 * Math.PI) / baseSides));
            }
            canvas.fill();
            if (CRTP != 'simplistic') {
                canvas.stroke();
            };
          }
          canvas.translate(
            (-object.width / fov) * asset.x,
            (-object.width / fov) * asset.y
          );
          if (('angle' in asset) && asset.angle != 0) {
            canvas.rotate(-asset.angle * Math.PI / 180);
          }
        }
      }

      canvas.lineJoin = "miter"; //change back
  };
    function newdraw(object, id, playerstring, auraWidth) {
    const CRTP = document.getElementById('theme').value;
    //function for drawing objects on the canvas. need to provide aura width because this fuction cannot access variables outside
    var drawingX =
      (object.x - px) / clientFovMultiplier + canvas.width / 2; //calculate the location on canvas to draw object
    var drawingY =
      (object.y - py) / clientFovMultiplier + canvas.height / 2;

    if (object.type == "bullet") {
      //draw bullet
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = object.deadOpacity;
      }
      var chooseflash = 3;
      if (object.hit > 0 && object.bulletType != "aura") {
        //if shape is hit AND bullet is not aura, choose whether it's color is white or original color to create flashing effect
        chooseflash = Math.floor(Math.random() * 3); //random number 0, 1 or 2
      }
      if (chooseflash == 0) {
        ctx.fillStyle = "white";
      } else if (chooseflash == 1) {
        ctx.fillStyle = "pink";
      } else {
        if (object.ownsIt == "yes" || object.bulletType == "aura") {
          //if it's an aura or client's tank owns the bullet
          ctx.fillStyle = object.color;
        } else {
          ctx.fillStyle = "#f04f54"; //bullet color is red
        }
      }
      if (object.bulletType == "aura") {
        var choosing;
        if (aurastate && CRTP != 'simplistic') {
        choosing = Math.floor(Math.random() * 2); //choose if particle spawn
        } else {
        choosing = 0;
        };
        if (choosing == 1) {
          //spawn a particle
          var angleDegrees = Math.floor(Math.random() * 360); //choose angle in degrees
          var angleRadians = (angleDegrees * Math.PI) / 180; //convert to radians
          var randomDistFromCenter =
            Math.floor(Math.random() * object.width * 2) - object.width;
          radparticles[particleID] = {
            angle: angleRadians,
            x: object.x + randomDistFromCenter * Math.cos(angleRadians),
            y: object.y + randomDistFromCenter * Math.sin(angleRadians),
            width: 5,
            height: 5,
            speed: 1,
            timer: 30,
            maxtimer: 50,
            color: object.color,
            outline: object.outline,
            type: "particle",
          };
          particleID++;
        }
      }

      if (object.ownsIt == "yes" || object.bulletType == "aura") {
        //if it's an aura or client's tank owns the bullet
        ctx.strokeStyle = object.outline;
      } else {
        ctx.strokeStyle = "#b33b3f"; //bullet is red
      }


      //bullet is purple even if bullet belongs to enemy
      if (object.color == "#934c93") {
        ctx.fillStyle = object.color;
      }
      if (object.outline == "#660066") {
        ctx.strokeStyle = object.outline;
      }

      //team colors
      if (object.team == "blue" || object.team == "green" || object.team == "red" || object.team == "purple" || object.team == "eternal" || object.team == "magenta" || object.team == "fallen" || object.team == "celestial") {
        ctx.fillStyle = bodyColors[object.team].col;
        ctx.strokeStyle = bodyColors[object.team].outline;
      }

      if (object.bulletType == "aura"){
        //color is aura color, regardless of team
        ctx.fillStyle = object.color;
        ctx.strokeStyle = object.outline;
      }

      if (object.passive == "yes") {
        if (object.bulletType == "aura") {
          ctx.strokeStyle = "rgba(128,128,128,.2)";
          ctx.fillStyle = "rgba(128,128,128,.2)";
        } else {
          ctx.strokeStyle = "dimgrey";
          ctx.fillStyle = "grey";
        }
      }

      if (object.team=="mob"){
        //dune mob's bullets is the colo of mob
        ctx.fillStyle = botcolors[object.ownerName].color;
        ctx.strokeStyle = botcolors[object.ownerName].outline;
      }

      ctx.lineWidth = 4 / clientFovMultiplier;
      if (object.bulletType == "bullet" || object.bulletType == "aura") {
        if (!object.color.includes('rgba(56,183,100')){//not a heal aura
          ctx.beginPath();
          ctx.arc(
            drawingX,
            drawingY,
            object.width / clientFovMultiplier,
            0,
            2 * Math.PI
          );
          if (aurastate || object.bulletType != "aura") {
          ctx.fill();
          };
          if (object.bulletType == 'aura') {
          ctx.stroke();
          } else if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        }
        else{//8 sides for healing aura
          ctx.beginPath();
          ctx.moveTo((object.width / clientFovMultiplier) + drawingX, drawingY);
          for (var i = 1; i <= 8 + 1; i += 1) {
            ctx.lineTo(
              (object.width / clientFovMultiplier) *
                  Math.cos((i * 2 * Math.PI) / 8) + drawingX,
              (object.width / clientFovMultiplier) *
                  Math.sin((i * 2 * Math.PI) / 8) + drawingY
            );
          }
           if (aurastate || object.bulletType != "aura") {
          ctx.fill();
           };
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        }
      } else if (object.bulletType == "trap") {
        //width is the radius, so need to times two to get total width
        //note: x and y of object are the center of object, but when drawing rectangles, the x and y coordinates given need to be the top left corner of the rectangle, so need to minus the width and height
        ctx.fillRect(
          drawingX - object.width / clientFovMultiplier,
          drawingY - object.width / clientFovMultiplier,
          (object.width * 2) / clientFovMultiplier,
          (object.width * 2) / clientFovMultiplier
        );
          if (CRTP != 'simplistic') {
        ctx.strokeRect(
          drawingX - object.width / clientFovMultiplier,
          drawingY - object.width / clientFovMultiplier,
          (object.width * 2) / clientFovMultiplier,
          (object.width * 2) / clientFovMultiplier
        );
          };
      } else if (object.bulletType == "drone") {
        ctx.save();
        ctx.translate(drawingX, drawingY);
        ctx.rotate(object.moveAngle);
        //ctx.rotate((object.moveAngle*180/Math.PI - 90) *Math.PI/180);//cannot straightaway use the angle, must add 90 degrees to it, because 0 degrees is pointing right, but we are drawing the triangle upwards
        ctx.beginPath();
        ctx.moveTo(
          (object.width / clientFovMultiplier) * Math.cos(0),
          (object.width / clientFovMultiplier) * Math.sin(0)
        );
        for (var i = 1; i <= 3; i += 1) {
          ctx.lineTo(
            (object.width / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / 3),
            (object.width / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / 3)
          );
        }
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.restore();
      } else if (object.bulletType == "mine" || object.bulletType == "minion") {
        //console.log(object.moveAngle/Math.PI*180)
        //mine is trap with barrel, minion is bullet with barrel
        ctx.save();
        ctx.translate(drawingX, drawingY);
        ctx.rotate(object.moveAngle);
        //ctx.rotate((object.moveAngle*180/Math.PI - 90) *Math.PI/180);//cannot straightaway use the angle, must add 90 degrees to it, because 0 degrees is pointing right, but we are drawing the triangle upwards

        if (object.bulletType == "minion"){
          //draw barrels underneath
          var prevfill = ctx.fillStyle;
          var prevstroke = ctx.strokeStyle;//store previous bullet color so can change back later
          ctx.fillStyle = bodyColors.barrel.col;
          ctx.strokeStyle = bodyColors.barrel.outline;
          Object.keys(object.barrels).forEach((barrel) => {
          let thisBarrel = object.barrels[barrel];
          ctx.rotate(thisBarrel.additionalAngle); //rotate to barrel angle
          if (thisBarrel.barrelType == "bullet") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            };
          }
          //drone barrel
          else if (thisBarrel.barrelType == "drone") {
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              0
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                (thisBarrel.x * 2) / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                (thisBarrel.x * 2) / clientFovMultiplier,
              0
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          //trap barrel
          else if (thisBarrel.barrelType == "trap") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            };
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          //mine barrel
          else if (thisBarrel.barrelType == "mine") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            };
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
        //minion barrel
        else if (thisBarrel.barrelType == "minion") {
          ctx.fillRect(
            -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier,
            thisBarrel.barrelWidth / clientFovMultiplier,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier,
            thisBarrel.barrelWidth / clientFovMultiplier,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier
          );
          };
          ctx.fillRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier
          );
           if (CRTP != 'simplistic') {
          ctx.strokeRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier
          );
           };
          ctx.fillRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange)  / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /5/ clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange)  / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /5 /clientFovMultiplier
          );
          };
        }
          })
          ctx.fillStyle = prevfill;
          ctx.strokeStyle = prevstroke;
        }
        ctx.beginPath();
        if (object.bulletType == "mine"){//mine
          ctx.moveTo(
            (object.width / clientFovMultiplier) * Math.cos(0),
            (object.width / clientFovMultiplier) * Math.sin(0)
          );
          for (var i = 1; i <= 3; i += 1) {
            ctx.lineTo(
              (object.width / clientFovMultiplier) *
                Math.cos((i * 2 * Math.PI) / 3),
              (object.width / clientFovMultiplier) *
                Math.sin((i * 2 * Math.PI) / 3)
            );
          }
        }
        else{//minion
          ctx.arc(0, 0, object.width / clientFovMultiplier, 0, 2 * Math.PI);
        }
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.rotate(-object.moveAngle); //rotate back
        //BARREL FOR THE MINE TRAP
        if (object.bulletType == "mine"){
        Object.keys(object.barrels).forEach((barrel) => {
          let thisBarrel = object.barrels[barrel];
          ctx.rotate(thisBarrel.additionalAngle); //rotate to barrel angle
          ctx.fillStyle = "grey";
          ctx.strokeStyle = "#5e5e5e";
          if (thisBarrel.barrelType == "bullet") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            };
          }
          //drone barrel
          else if (thisBarrel.barrelType == "drone") {
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              0
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                (thisBarrel.x * 2) / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                (thisBarrel.x * 2) / clientFovMultiplier,
              0
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          //trap barrel
          else if (thisBarrel.barrelType == "trap") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            };
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          //mine barrel
          else if (thisBarrel.barrelType == "mine") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight -
                thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            };
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(
                thisBarrel.barrelHeight - thisBarrel.barrelHeightChange
              ) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
        //minion barrel
        else if (thisBarrel.barrelType == "minion") {
          ctx.fillRect(
            -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier,
            thisBarrel.barrelWidth / clientFovMultiplier,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier,
            thisBarrel.barrelWidth / clientFovMultiplier,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier
          );
          };
          ctx.fillRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier
          );
          };
          ctx.fillRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange)  / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /5/ clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange)  / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /5 /clientFovMultiplier
          );
          };
        }
          ctx.beginPath();
          ctx.arc(
            0,
            0,
            thisBarrel.barrelWidth / clientFovMultiplier,
            0,
            2 * Math.PI
          );
          ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          ctx.rotate(-thisBarrel.additionalAngle); //rotate back
        });
      }

        ctx.restore();
      }
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = 1.0; //reset opacity
      }
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "bot") {
      //draw bot
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = object.deadOpacity;
      }
      ctx.lineWidth = 4 / clientFovMultiplier;
      ctx.lineJoin = "round"; //prevent spikes above the capital letter "M"
      ctx.save();
      ctx.translate(drawingX, drawingY);
      ctx.rotate(object.angle);
      //draw barrels
      if (object.name!="Pillbox"){//pillbox's barrel is visually a turret
        Object.keys(object.barrels).forEach((barrel) => {
          let thisBarrel = object.barrels[barrel];
          ctx.rotate(((thisBarrel.additionalAngle + 90) * Math.PI) / 180); //rotate to barrel angle
          ctx.fillStyle = bodyColors.barrel.col;
          ctx.strokeStyle = bodyColors.barrel.outline;
          if (thisBarrel.barrelType == "bullet") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            };
          }
          //drone barrel
          else if (thisBarrel.barrelType == "drone") {
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              0
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                (thisBarrel.x * 2) / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                (thisBarrel.x * 2) / clientFovMultiplier,
              0
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          //trap barrel
          else if (thisBarrel.barrelType == "trap") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            };
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          //mine barrel
          else if (thisBarrel.barrelType == "mine") {
            ctx.fillRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            if (CRTP != 'simplistic') {
            ctx.strokeRect(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier,
              thisBarrel.barrelWidth / clientFovMultiplier,
              (((thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            };
            ctx.beginPath();
            ctx.moveTo(
              -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.lineTo(
              -thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                clientFovMultiplier
            );
            ctx.lineTo(
              thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                thisBarrel.x / clientFovMultiplier,
              ((-(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                3) *
                2) /
                clientFovMultiplier
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
        //minion barrel
        else if (thisBarrel.barrelType == "minion") {
          ctx.fillRect(
            -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier,
            thisBarrel.barrelWidth / clientFovMultiplier,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier,
            thisBarrel.barrelWidth / clientFovMultiplier,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
              clientFovMultiplier
          );
          };
          ctx.fillRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) / 1.5 / clientFovMultiplier
          );
          };
          ctx.fillRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange)  / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /5/ clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            (-thisBarrel.barrelWidth * 1.5) / 2 / clientFovMultiplier +
              thisBarrel.x / clientFovMultiplier,
            -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange)  / clientFovMultiplier,
            (thisBarrel.barrelWidth / clientFovMultiplier) * 1.5,
            (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /5 /clientFovMultiplier
          );
          };
        }
          ctx.rotate((-(thisBarrel.additionalAngle + 90) * Math.PI) / 180); //rotate back
        });
      }
      if (object.name=="Cluster"){
        //draw the spawning barrels
        let barrelwidth = object.width*0.7;
        let barrelheight = object.width*1.2;
        ctx.fillStyle = bodyColors.barrel.col;
        ctx.strokeStyle = bodyColors.barrel.outline;
        ctx.save();
        ctx.rotate(90 * Math.PI / 180);
        for (let i = 0; i < 5; i++){
          if (i!=0){
            ctx.rotate(72 * Math.PI / 180); //rotate 72 for each barrel
          }
          ctx.beginPath();
          ctx.moveTo(
            -barrelwidth / 5 / clientFovMultiplier,
            0
          );
          ctx.lineTo(
            -barrelwidth / clientFovMultiplier,
            -barrelheight / clientFovMultiplier
          );
          ctx.lineTo(
            barrelwidth / clientFovMultiplier,
            -barrelheight / clientFovMultiplier
          );
          ctx.lineTo(
            barrelwidth / 5 / clientFovMultiplier,
            0
          );
          ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        }
        ctx.restore();
      }
      else if (object.name=="Infestor"){
        //draw the spawning barrels
        let barrelwidth = object.width*0.7;
        let barrelheight = object.width*1.2;
        ctx.fillStyle = bodyColors.barrel.col;
        ctx.strokeStyle = bodyColors.barrel.outline;
        ctx.save();
        for (let i = 0; i < 4; i++){//normal barrels
          if (i!=0){
            ctx.rotate(90 * Math.PI / 180);
          }
          ctx.fillRect(
            -barrelwidth / 2 / clientFovMultiplier,
            -barrelheight / clientFovMultiplier,
            barrelwidth / clientFovMultiplier,
            barrelheight / clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            -barrelwidth / 2 / clientFovMultiplier,
            -barrelheight / clientFovMultiplier,
            barrelwidth / clientFovMultiplier,
            barrelheight / clientFovMultiplier
          );
          };
        }
        ctx.restore();
        ctx.save();
        ctx.rotate(45 * Math.PI / 180);
        barrelwidth = object.width*0.6;
        barrelheight = object.width*2;
        for (let i = 0; i < 4; i++){//traplike barrels
          if (i!=0){
            ctx.rotate(90 * Math.PI / 180);
          }
          ctx.fillRect(
            -barrelwidth / 2 / clientFovMultiplier,
            -barrelheight * 0.55 / clientFovMultiplier,
            barrelwidth / clientFovMultiplier,
            barrelheight * 0.5 / clientFovMultiplier
          );
          if (CRTP != 'simplistic') {
          ctx.strokeRect(
            -barrelwidth / 2 / clientFovMultiplier,
            -barrelheight * 0.55 / clientFovMultiplier,
            barrelwidth / clientFovMultiplier,
            barrelheight * 0.5 / clientFovMultiplier
          );
          };
          ctx.beginPath();
          ctx.moveTo(
            -barrelwidth / 2 / clientFovMultiplier,
            -barrelheight * 0.55 / clientFovMultiplier
          );
          ctx.lineTo(
            -barrelwidth/1.7 / clientFovMultiplier,
            -barrelheight * 0.65 / clientFovMultiplier
          );
          ctx.lineTo(
            barrelwidth/1.7 / clientFovMultiplier,
            -barrelheight * 0.65 / clientFovMultiplier
          );
          ctx.lineTo(
            barrelwidth / 2 / clientFovMultiplier,
            -barrelheight * 0.55 / clientFovMultiplier
          );
          ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        }
        ctx.restore();
      }
      else if (object.name=="Champion"){
        //draw spikes
        var numberOfSpikes = 5;
        var outerRadius = object.width / clientFovMultiplier * 1.3;
        var innerRadius = object.width / clientFovMultiplier /1.3;
        var rot = (Math.PI / 2) * 3;//dont change this, or else will have strange extra lines
        var x = 0;
        var y = 0;
        ctx.fillStyle = bodyColors.barrel.col;
        ctx.strokeStyle = bodyColors.barrel.outline;
        ctx.save();
        ctx.rotate(90 * Math.PI / 180);
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.restore();
      }
      var chooseflash = 3;
      if (object.hit > 0) {
        //if shape is hit, choose whether it's color is white or original color to create flashing effect
        chooseflash = Math.floor(Math.random() * 3); //random number 0, 1 or 2
      }
      if (chooseflash == 0) {
        ctx.fillStyle = "white";
      } else if (chooseflash == 1) {
        ctx.fillStyle = "pink";
      } else {
        ctx.fillStyle = botcolors[object.name].color;
      }
      ctx.strokeStyle = botcolors[object.name].outline;
      //draw body
      if (object.side==0) {
        //draw circle
        ctx.beginPath();
        ctx.arc(0, 0, object.width / clientFovMultiplier, 0, 2 * Math.PI);
        ctx.fill();
        ctx.stroke();
      } else if (object.side>=0) {
        if (object.hasOwnProperty('randomPointsArrayX')){
          //draw for rock and boulder
          //POLYGON WITH IRREGULAR SIDES
          ctx.rotate(-object.angle); //rotate back so that rock wont rotate to face you
          var rockSides = object.side;
          ctx.beginPath();
          ctx.moveTo(
            0 + (object.width / clientFovMultiplier) * Math.cos(0),
            0 + (object.width / clientFovMultiplier) * Math.sin(0)
          );
          for (var i = 1; i <= rockSides; i += 1) {
            var XRandom = object.randomPointsArrayX[i - 1];
            var YRandom = object.randomPointsArrayY[i - 1];
            ctx.lineTo(XRandom + (object.width / clientFovMultiplier) * Math.cos((i * 2 * Math.PI) / rockSides),
              YRandom + (object.width / clientFovMultiplier) * Math.sin((i * 2 * Math.PI) / rockSides)
            );
          }
          ctx.fill();
          ctx.stroke();
        }
        else{//normal spawner
          if (object.name=="Cluster"||object.name=="Pursuer"||object.name=="Champion"||object.name=="Infestor"){
            //need to rotate 72/2 degrees so that pentagon not facing vertex towards player
            ctx.rotate(Math.PI/object.side);//2 PI / sides / 2
          }
          ctx.beginPath();
          ctx.moveTo((object.width / clientFovMultiplier), 0);
          for (var i = 1; i <= object.side + 1; i += 1) {
            ctx.lineTo(
              (object.width / clientFovMultiplier) *
                  Math.cos((i * 2 * Math.PI) / object.side),
              (object.width / clientFovMultiplier) *
                  Math.sin((i * 2 * Math.PI) / object.side)
            );
          }
          ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          if (object.name=="Cluster"||object.name=="Pursuer"){
            ctx.rotate(-Math.PI/object.side);//rotate back
            //draw circle on top
            ctx.fillStyle = bodyColors.barrel.col;//light grey
            ctx.strokeStyle = bodyColors.barrel.outline;
            ctx.beginPath();
            ctx.arc(0, 0, object.width/2 / clientFovMultiplier, 0, 2 * Math.PI);
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          else if (object.name=="Champion"){
            ctx.rotate(-Math.PI/object.side);//rotate back
            //draw circle on top
            ctx.fillStyle = "grey";//darker grey
            ctx.strokeStyle = "#5e5e5e";
            ctx.beginPath();
            ctx.arc(0, 0, object.width/2.5 / clientFovMultiplier, 0, 2 * Math.PI);
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          else if (object.name=="Infestor"){
            ctx.rotate(-Math.PI/object.side);//rotate back
            //draw circle on top
            ctx.fillStyle = bodyColors.barrel.col;//light grey
            ctx.strokeStyle = bodyColors.barrel.outline;
            ctx.beginPath();
            ctx.arc(0, 0, object.width/5 / clientFovMultiplier, 0, 2 * Math.PI);
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          else if (object.name=="Leech"){
            //draw circle on top
            ctx.fillStyle = bodyColors.barrel.col;//light grey
            ctx.strokeStyle = bodyColors.barrel.outline;
            ctx.beginPath();
            ctx.arc(0, 0, object.width/2 / clientFovMultiplier, 0, 2 * Math.PI);
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
          }
          else if (object.name=="Pillbox"){//pillbox's barrel is visually a turret
            ctx.lineJoin = "round"; //make nice round corners
            ctx.rotate(90 * Math.PI / 180);
            Object.keys(object.barrels).forEach((barrel) => {
              //note that you must use [barrel] instead of .barrel, if not there will be an error
              let thisBarrel = object.barrels[barrel];
              ctx.fillStyle = bodyColors.barrel.col;
              ctx.strokeStyle = bodyColors.barrel.outline;
              ctx.fillRect(
                -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                  thisBarrel.x,
                -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                  clientFovMultiplier,
                thisBarrel.barrelWidth / clientFovMultiplier,
                (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                  clientFovMultiplier
              );
              if (CRTP != 'simplistic') {
              ctx.strokeRect(
                -thisBarrel.barrelWidth / 2 / clientFovMultiplier +
                  thisBarrel.x,
                -(thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                  clientFovMultiplier,
                thisBarrel.barrelWidth / clientFovMultiplier,
                (thisBarrel.barrelHeight - thisBarrel.barrelHeightChange) /
                  clientFovMultiplier
              );
              };
            });
            ctx.rotate(-90 * Math.PI / 180);
            //draw turret base
            ctx.beginPath();
            ctx.arc(
              0,
              0,
              (object.width / clientFovMultiplier) * 0.6,
              0,
              2 * Math.PI
            );
            ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
            ctx.lineJoin = "miter"; //change back
          }
        }
      } else{//negative sides, draw a star! (cactus)
        var numberOfSpikes = -object.side;
        var outerRadius = object.width / clientFovMultiplier * 1.5;
        var innerRadius = object.width / clientFovMultiplier;

        var rot = (Math.PI / 2) * 3;//dont change this, or else will have strange extra lines
        var x = 0;
        var y = 0;
        ctx.rotate(-object.angle); //rotate back so that rock wont rotate to face you
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      }
      ctx.restore();
      if (object.health < object.maxhealth) {
        //draw health bar background
        var w = (object.width * 2) / clientFovMultiplier;
        var h = 7 / clientFovMultiplier;
        var r = h / 2;
        var x = drawingX - object.width / clientFovMultiplier;
        var y = drawingY + object.width / clientFovMultiplier + 10;
        ctx.fillStyle = "black";
        ctx.strokeStyle = "black";
        ctx.lineWidth = 2.5 / clientFovMultiplier;
        ctx.beginPath();
        ctx.moveTo(x + r, y);
        ctx.arcTo(x + w, y, x + w, y + h, r);
        ctx.arcTo(x + w, y + h, x, y + h, r);
        ctx.arcTo(x, y + h, x, y, r);
        ctx.arcTo(x, y, x + w, y, r);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        //draw health bar
        if (object.health > 0) {
          w = (w / object.maxhealth) * object.health;
          if (r * 2 > w) {
            //prevent weird shape when radius more than width
            r = w / 2;
            y += (h - w) / 2; //move health bar so that it is centered vertically in black bar
            h = w;
          }
          ctx.fillStyle = botcolors[object.name].color;
          ctx.beginPath();
          ctx.moveTo(x + r, y);
          ctx.arcTo(x + w, y, x + w, y + h, r);
          ctx.arcTo(x + w, y + h, x, y + h, r);
          ctx.arcTo(x, y + h, x, y, r);
          ctx.arcTo(x, y, x + w, y, r);
          ctx.closePath();
          ctx.fill();
          ctx.stroke();
        }
      }
      ctx.fillStyle = "white";
      ctx.strokeStyle = "black";
      ctx.lineWidth = 5 / clientFovMultiplier;
      ctx.font = "700 " + 20 / clientFovMultiplier + "px Roboto";
      ctx.textAlign = "center";
      ctx.lineJoin = "round"; //prevent spikes above the capital letter "M"
      //note: if you stroke then fill, the words will be thicker and nicer. If you fill then stroke, the words are thinner.
      if ((showStaticMobName == "yes"||botcolors[object.name].static=="no") && (showMinionMobName == "yes"||botcolors[object.name].minion=="no")){//settings for showing static and minion names
        if (botcolors[object.name].specialty != "") {
          var specialtyText = " (" + botcolors[object.name].specialty + ")";
        } else {
          var specialtyText = "";
        }
        ctx.strokeText(
          object.name + specialtyText,
          drawingX,
          drawingY - object.width / clientFovMultiplier - 10
        );
        ctx.fillText(
          object.name + specialtyText,
          drawingX,
          drawingY - object.width / clientFovMultiplier - 10
        );
      }
      ctx.lineJoin = "miter"; //prevent spikes above the capital letter "M"
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = 1.0; //reset opacity
      }
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "shape") {
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = object.deadOpacity;
      }
      var radiantAuraSize =
        document.getElementById("sizevalue").innerHTML * auraWidth; //aura size determined by settings, but default is 5
      //draw shape
      ctx.save();
      ctx.translate(drawingX, drawingY);
      ctx.rotate((object.angle * Math.PI) / 180);
      if (object.hasOwnProperty("radtier")) {
        //radiant shape
        if (!radiantShapes.hasOwnProperty(id)) {
          var randomstate = Math.floor(Math.random() * 3); //randomly choose a color state for the radiant shape to start (if not when you spawn in cavern, all shapes same color)
          var randomtype = Math.floor(Math.random() * 2) + 1; //choose animation color type (1 or 2)
          if (randomtype == 1) {
            if (randomstate == 0) {
              radiantShapes[id] = {
                red: 255,
                blue: 0,
                green: 0,
                rgbstate: 1,
                radtype: randomtype,
              }; //keep track of radiant shape colors (done in client code)
            } else if (randomstate == 1) {
              radiantShapes[id] = {
                red: 199,
                blue: 0,
                green: 150,
                rgbstate: 2,
                radtype: randomtype,
              };
            } else if (randomstate == 2) {
              radiantShapes[id] = {
                red: -1,
                blue: 200,
                green: 0,
                rgbstate: 3,
                radtype: randomtype,
              };
            }
          } else {
            if (randomstate == 0) {
              radiantShapes[id] = {
                red: 118,
                blue: 168,
                green: 151,
                rgbstate: 1,
                radtype: randomtype,
              };
            } else if (randomstate == 1) {
              radiantShapes[id] = {
                red: 209,
                blue: 230,
                green: 222,
                rgbstate: 2,
                radtype: randomtype,
              };
            } else if (randomstate == 2) {
              radiantShapes[id] = {
                red: 234,
                blue: 240,
                green: 180,
                rgbstate: 3,
                radtype: randomtype,
              };
            }
          }
        }
        object.red = radiantShapes[id].red;
        object.blue = radiantShapes[id].blue;
        object.green = radiantShapes[id].green;
      }
      if (object.hasOwnProperty("red")) {
        //calculate color of spikes, which would be 20 higher than actual rgb value
        if (object.red + 150 <= 255) {
          var spikeRed = object.red + 150;
        } else {
          var spikeRed = 255;
        }
        if (object.blue + 150 <= 255) {
          var spikeBlue = object.blue + 150;
        } else {
          var spikeBlue = 255;
        }
        if (object.green + 150 <= 255) {
          var spikeGreen = object.green + 150;
        } else {
          var spikeGreen = 255;
        }
        if (object.radtier == 3) {
          //for high rarity radiant shapes, draw spikes
          ctx.rotate((extraSpikeRotate * Math.PI) / 180);
          ctx.fillStyle =
            "rgba(" +
            spikeRed +
            ", " +
            spikeGreen +
            ", " +
            spikeBlue +
            ", 0.7)";
          ctx.strokeStyle =
            "rgba(" +
            spikeRed +
            ", " +
            spikeGreen +
            ", " +
            spikeBlue +
            ", 0.3)";
          var numberOfSpikes = 6;
          var outerRadius =
            ((object.width * radiantAuraSize * 3) / clientFovMultiplier) *
            0.75;
          var innerRadius = (object.width / clientFovMultiplier) * 0.75;

          var rot = (Math.PI / 2) * 3;
          var x = 0;
          var y = 0;

          ctx.beginPath();
          ctx.moveTo(0, 0 - outerRadius);
          for (i = 0; i < numberOfSpikes; i++) {
            x = 0 + Math.cos(rot) * outerRadius;
            y = 0 + Math.sin(rot) * outerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
            x = 0 + Math.cos(rot) * innerRadius;
            y = 0 + Math.sin(rot) * innerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
          }
          ctx.lineTo(0, 0 - outerRadius);
          ctx.closePath();
          ctx.lineWidth = 3 / clientFovMultiplier;
          ctx.fill();
          ctx.stroke();
          ctx.rotate((-extraSpikeRotate * Math.PI) / 180);
        } else if (object.radtier == 4) {
          //for high rarity radiant shapes, draw spikes
          ctx.rotate((extraSpikeRotate1 * Math.PI) / 180);
          ctx.fillStyle =
            "rgba(" +
            spikeRed +
            ", " +
            spikeGreen +
            ", " +
            spikeBlue +
            ", 0.7)";
          ctx.strokeStyle =
            "rgba(" +
            spikeRed +
            ", " +
            spikeGreen +
            ", " +
            spikeBlue +
            ", 0.3)";
          var numberOfSpikes = 3;
          var outerRadius =
            (object.width * radiantAuraSize * 3) / clientFovMultiplier;
          var innerRadius = (object.width / clientFovMultiplier) * 0.5;
          var rot = (Math.PI / 2) * 3;
          var x = 0;
          var y = 0;
          ctx.beginPath();
          ctx.moveTo(0, 0 - outerRadius);
          for (i = 0; i < numberOfSpikes; i++) {
            x = 0 + Math.cos(rot) * outerRadius;
            y = 0 + Math.sin(rot) * outerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
            x = 0 + Math.cos(rot) * innerRadius;
            y = 0 + Math.sin(rot) * innerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
          }
          ctx.lineTo(0, 0 - outerRadius);
          ctx.closePath();
          ctx.lineWidth = 3 / clientFovMultiplier;
          ctx.fill();
          ctx.stroke();
          ctx.rotate((-extraSpikeRotate1 * Math.PI) / 180);
          ctx.rotate((extraSpikeRotate2 * Math.PI) / 180);
          var numberOfSpikes = 6;
          var outerRadius =
            ((object.width * radiantAuraSize * 3) / clientFovMultiplier) *
            0.5;
          var innerRadius = (object.width / clientFovMultiplier) * 0.5;
          var rot = (Math.PI / 2) * 3;
          var x = 0;
          var y = 0;
          ctx.beginPath();
          ctx.moveTo(0, 0 - outerRadius);
          for (i = 0; i < numberOfSpikes; i++) {
            x = 0 + Math.cos(rot) * outerRadius;
            y = 0 + Math.sin(rot) * outerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
            x = 0 + Math.cos(rot) * innerRadius;
            y = 0 + Math.sin(rot) * innerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
          }
          ctx.lineTo(0, 0 - outerRadius);
          ctx.closePath();
          ctx.lineWidth = 3 / clientFovMultiplier;
          ctx.fill();
          ctx.stroke();
          ctx.rotate((-extraSpikeRotate2 * Math.PI) / 180);
        } else if (object.radtier == 5) {
          //for high rarity radiant shapes, draw spikes
          ctx.rotate((extraSpikeRotate1 * Math.PI) / 180);
          ctx.fillStyle =
            "rgba(" +
            spikeRed +
            ", " +
            spikeGreen +
            ", " +
            spikeBlue +
            ", 0.7)";
          ctx.strokeStyle =
            "rgba(" +
            spikeRed +
            ", " +
            spikeGreen +
            ", " +
            spikeBlue +
            ", 0.3)";
          var numberOfSpikes = 3;
          var outerRadius =
            ((object.width * radiantAuraSize * 3) / clientFovMultiplier) *
            1.5;
          var innerRadius = (object.width / clientFovMultiplier) * 0.5;
          var rot = (Math.PI / 2) * 3;
          var x = 0;
          var y = 0;
          ctx.beginPath();
          ctx.moveTo(0, 0 - outerRadius);
          for (i = 0; i < numberOfSpikes; i++) {
            x = 0 + Math.cos(rot) * outerRadius;
            y = 0 + Math.sin(rot) * outerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
            x = 0 + Math.cos(rot) * innerRadius;
            y = 0 + Math.sin(rot) * innerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
          }
          ctx.lineTo(0, 0 - outerRadius);
          ctx.closePath();
          ctx.lineWidth = 3 / clientFovMultiplier;
          ctx.fill();
          ctx.stroke();
          ctx.rotate((-extraSpikeRotate1 * Math.PI) / 180);
          ctx.rotate((extraSpikeRotate2 * Math.PI) / 180);
          var numberOfSpikes = 3;
          var outerRadius =
            ((object.width * radiantAuraSize * 3) / clientFovMultiplier) *
            0.5;
          var innerRadius = (object.width / clientFovMultiplier) * 0.5;
          var rot = (Math.PI / 2) * 3;
          var x = 0;
          var y = 0;
          ctx.beginPath();
          ctx.moveTo(0, 0 - outerRadius);
          for (i = 0; i < numberOfSpikes; i++) {
            x = 0 + Math.cos(rot) * outerRadius;
            y = 0 + Math.sin(rot) * outerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
            x = 0 + Math.cos(rot) * innerRadius;
            y = 0 + Math.sin(rot) * innerRadius;
            ctx.lineTo(x, y);
            rot += Math.PI / numberOfSpikes;
          }
          ctx.lineTo(0, 0 - outerRadius);
          ctx.closePath();
          ctx.lineWidth = 3 / clientFovMultiplier;
          ctx.fill();
          ctx.stroke();
          ctx.rotate((-extraSpikeRotate2 * Math.PI) / 180);
        }
        //if shape is radiant
        //draw aura

        //old code where aura was a gradient
        /*
            const gradient = ctx.createRadialGradient(0, 0, object.width/clientFovMultiplier, 0, 0, object.width/clientFovMultiplier*radiantAuraSize);
            gradient.addColorStop(0, 'rgba(' + object.red + ', ' + object.green + ', ' + object.blue + ', 0.3)');
            gradient.addColorStop(0.5, 'rgba(' + object.red + ', ' + object.green + ', ' + object.blue + ', 0.1)');
            gradient.addColorStop(1, 'rgba(' + object.red + ', ' + object.green + ', ' + object.blue + ', 0.0)');
            ctx.fillStyle = gradient;
            ctx.beginPath();
            */

        //old code where aura have shape
        ctx.fillStyle =
          "rgba(" +
          object.red +
          ", " +
          object.green +
          ", " +
          object.blue +
          ", 0.3)";
        ctx.strokeStyle =
          "rgba(" +
          object.red +
          ", " +
          object.green +
          ", " +
          object.blue +
          ", 0.3)";
        ctx.lineWidth = 3 / clientFovMultiplier;
        ctx.beginPath();

        var shapeaurasize = object.radtier;
        if (shapeaurasize > 3) {
          shapeaurasize = 3; //prevent huge auras
        }
        ctx.moveTo(
          0 +
            ((object.width * radiantAuraSize * shapeaurasize) /
              clientFovMultiplier) *
              Math.cos(0),
          0 +
            ((object.width * radiantAuraSize * shapeaurasize) /
              clientFovMultiplier) *
              Math.sin(0)
        );
        for (var i = 1; i <= object.sides + 1; i += 1) {
          ctx.lineTo(
            0 +
              ((object.width * radiantAuraSize * shapeaurasize) /
                clientFovMultiplier) *
                Math.cos((i * 2 * Math.PI) / object.sides),
            0 +
              ((object.width * radiantAuraSize * shapeaurasize) /
                clientFovMultiplier) *
                Math.sin((i * 2 * Math.PI) / object.sides)
          );
        }

        //ctx.arc(0, 0, object.width/clientFovMultiplier*radiantAuraSize, 0, 2 * Math.PI);
        ctx.fill();
        ctx.stroke();
        var shadeFactor = 3 / 4; //smaller the value, darker the shade
        ctx.strokeStyle =
          "rgb(" +
          object.red * shadeFactor +
          ", " +
          object.green * shadeFactor +
          ", " +
          object.blue * shadeFactor +
          ")";
        ctx.fillStyle =
          "rgb(" +
          object.red +
          ", " +
          object.green +
          ", " +
          object.blue +
          ")";
        if (object.hit > 0) {
          //if shape is hit
          ctx.strokeStyle =
            "rgb(" +
            (object.red * shadeFactor + 20) +
            ", " +
            (object.green * shadeFactor + 20) +
            ", " +
            (object.blue * shadeFactor + 20) +
            ")";
          ctx.fillStyle =
            "rgb(" +
            (object.red + 20) +
            ", " +
            (object.green + 20) +
            ", " +
            (object.blue + 20) +
            ")";
        }

        //choose whether a particle would spawn
        //particle spawn chance based on number of sides the shape has, so square has less particles
        if (spawnradparticle == "yes"){
          var chooseValue = 20 - object.sides * 2; //lower the number means more particles spawned
          if (chooseValue < 5) {
            //5 refers to mimimum particle spawn chance
            chooseValue = 5;
          }
          var choosing = Math.floor(Math.random() * chooseValue); //choose if particle spawn
          if (choosing == 1) {
            //spawn a particle
            var angleDegrees = Math.floor(Math.random() * 360); //choose angle in degrees
            var angleRadians = (angleDegrees * Math.PI) / 180; //convert to radians
            var randomDistFromCenter =
              Math.floor(Math.random() * object.width * 2) - object.width;
            radparticles[particleID] = {
              angle: angleRadians,
              x: object.x + randomDistFromCenter * Math.cos(angleRadians),
              y: object.y + randomDistFromCenter * Math.sin(angleRadians),
              width: 5,
              height: 5,
              speed: 1,
              timer: 50,
              maxtimer: 50,
              color:
                "rgba(" +
                object.red +
                "," +
                object.green +
                "," +
                object.blue +
                ",.5)",
              outline:
                "rgba(" +
                (object.red* shadeFactor + 20) +
                "," +
                (object.green* shadeFactor + 20) +
                "," +
                (object.blue* shadeFactor + 20) +
                ",.5)",
              type: "particle",
            };
            particleID++;
          }
        }
      } else {
        //if not radiant
        //get shape colors in client code based on theme
        ctx.fillStyle = shapecolors[object.sides][colortheme].color;
        ctx.strokeStyle = shapecolors[object.sides][colortheme].outline;
        if (object.hit > 0) {
          //if shape is hit
          ctx.fillStyle = shapecolors[object.sides][colortheme].hitcolor;
          ctx.strokeStyle =
            shapecolors[object.sides][colortheme].hitoutline;
        }
      }
      ctx.lineJoin = "round"; //make corners of shape round
      if (object.sides == "star") {
        //draw a star

        var numberOfSpikes = 5;
        var outerRadius = object.width / clientFovMultiplier;
        var innerRadius = (object.width / clientFovMultiplier / 3) * 2;

        var rot = (Math.PI / 2) * 3;
        var x = 0;
        var y = 0;

        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.lineWidth = 4 / clientFovMultiplier;
        ctx.fill();
        ctx.stroke();
      } else {
        ctx.lineWidth = 4 / clientFovMultiplier;
        ctx.beginPath();
        ctx.moveTo(
          0 + (object.width / clientFovMultiplier) * Math.cos(0),
          0 + (object.width / clientFovMultiplier) * Math.sin(0)
        );
        for (var i = 1; i <= object.sides + 1; i += 1) {
          ctx.lineTo(
            0 +
              (object.width / clientFovMultiplier) *
                Math.cos((i * 2 * Math.PI) / object.sides),
            0 +
              (object.width / clientFovMultiplier) *
                Math.sin((i * 2 * Math.PI) / object.sides)
          );
        }
        ctx.fill();
        ctx.stroke();
      }
      ctx.lineJoin = "miter"; //change back to default
      ctx.restore(); //must restore to reset angle rotation so health bar wont be rotated sideways
      //draw shape's health bar
      if (object.health < object.maxhealth) {
        //draw health bar background
        var w = (object.width / clientFovMultiplier) * 2;
        var h = 7 / clientFovMultiplier;
        var r = h / 2;
        var x = drawingX - object.width / clientFovMultiplier;
        var y = drawingY + object.width / clientFovMultiplier + 10;
        ctx.fillStyle = "black";
        ctx.strokeStyle = "black";
        ctx.lineWidth = 2.5 / clientFovMultiplier;//determines with of black area
        ctx.beginPath();
        ctx.moveTo(x + r, y);
        ctx.arcTo(x + w, y, x + w, y + h, r);
        ctx.arcTo(x + w, y + h, x, y + h, r);
        ctx.arcTo(x, y + h, x, y, r);
        ctx.arcTo(x, y, x + w, y, r);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        //draw health bar
        if (object.health > 0) {
          //dont draw health bar if negative health
          w = (w / object.maxhealth) * object.health;
          if (r * 2 > w) {
            //prevent weird shape when radius more than width
            r = w / 2;
            y += (h - w) / 2; //move health bar so that it is centered vertically in black bar
            h = w;
          }
          if (object.hasOwnProperty("red")) {
            //if shape is radiant
            ctx.fillStyle =
              "rgb(" +
              object.red +
              ", " +
              object.green +
              ", " +
              object.blue +
              ")";
          } else {
            ctx.fillStyle = shapecolors[object.sides][colortheme].color;
            if (object.sides==10||object.sides==11||object.sides==14){//these shapes are very dark, cannot see health bar
              ctx.fillStyle = shapecolors[12][colortheme].color;//use ddecagon's grey color for health bar
            }
          }
          ctx.beginPath();
          ctx.moveTo(x + r, y);
          ctx.arcTo(x + w, y, x + w, y + h, r);
          ctx.arcTo(x + w, y + h, x, y + h, r);
          ctx.arcTo(x, y + h, x, y, r);
          ctx.arcTo(x, y, x + w, y, r);
          ctx.closePath();
          ctx.fill();
          ctx.stroke();
        }
      }
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = 1.0; //reset opacity
      }
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "spawner") {
      //spawner in sanctuary
      ctx.save();
      ctx.translate(drawingX, drawingY);
      ctx.rotate(object.angle);
      ctx.lineJoin = "round"; //make corners of shape round

      //actual body
      ctx.fillStyle = object.baseColor;
      ctx.strokeStyle = object.baseOutline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.basewidth6 / clientFovMultiplier) * Math.cos(0),
        0 + (object.basewidth6 / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.basewidth6 / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.basewidth6 / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.width / clientFovMultiplier) * Math.cos(0),
        0 + (object.width / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.width / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.width / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.baseColor;
      ctx.strokeStyle = object.baseOutline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.basewidth4 / clientFovMultiplier) * Math.cos(0),
        0 + (object.basewidth4 / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.basewidth4 / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.basewidth4 / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.lineWidth = 4 / clientFovMultiplier;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.basewidth5 / clientFovMultiplier) * Math.cos(0),
        0 + (object.basewidth5 / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.basewidth5 / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.basewidth5 / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.baseColor;
      ctx.strokeStyle = object.baseOutline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.basewidth1 / clientFovMultiplier) * Math.cos(0),
        0 + (object.basewidth1 / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.basewidth1 / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.basewidth1 / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.barrelColor;
      ctx.strokeStyle = object.barrelOutline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.basewidth2 / clientFovMultiplier) * Math.cos(0),
        0 + (object.basewidth2 / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.basewidth2 / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.basewidth2 / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.lineWidth = 4 / clientFovMultiplier;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.basewidth3 / clientFovMultiplier) * Math.cos(0),
        0 + (object.basewidth3 / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.basewidth3 / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.basewidth3 / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      //draw barrels
      ctx.fillStyle = object.barrelColor;
      ctx.strokeStyle = object.barrelOutline;
      //trapezoid at the tip
      var barrelwidth = 140;
      var barrelheight = 28;
      //rectangle
      var barrelwidth2 = 180;
      var barrelheight2 = 28;
      //base trapezoid
      var barrelwidth3 = 140;
      var barrelheight3 = 80;
      //note that trapezoids and rectangles are drawn differently

      var barrelDistanceFromCenter = (object.width * (Math.cos(Math.PI/object.sides)));//width of middle of polygon (less than width of circle)

      function drawSancBarrel(barNum){
        var barAngle = 360/object.sides*(barNum+0.5);//half of a side, cuz barrel is in between sides
        var barrelX = Math.cos((barAngle * Math.PI) / 180) * (barrelDistanceFromCenter+ barrelheight+ barrelheight2+ barrelheight3);//object.width * 0.9
        var barrelY = Math.sin((barAngle * Math.PI) / 180) * (barrelDistanceFromCenter+ barrelheight+ barrelheight2+ barrelheight3);
        var barrelX2 = Math.cos((barAngle * Math.PI) / 180) * (barrelDistanceFromCenter + barrelheight2 + barrelheight3); //move rectangle barrel downwards
        var barrelY2 = Math.sin((barAngle * Math.PI) / 180) * (barrelDistanceFromCenter + barrelheight2 + barrelheight3);
        var barrelX3 = Math.cos((barAngle * Math.PI) / 180) * (barrelDistanceFromCenter + barrelheight3); //move base trapezoid barrel downwards
        var barrelY3 = Math.sin((barAngle * Math.PI) / 180) * (barrelDistanceFromCenter + barrelheight3);
        //base trapezoid
        ctx.save();
        ctx.translate(
          barrelX3 / clientFovMultiplier,
          barrelY3 / clientFovMultiplier
        );
        ctx.rotate(((barAngle - 90) * Math.PI) / 180);
        ctx.beginPath();
        ctx.moveTo(
          ((-barrelwidth3 / 3) * 2) / clientFovMultiplier,
          -barrelheight3 / clientFovMultiplier
        );
        ctx.lineTo(-barrelwidth3 / clientFovMultiplier, 0);
        ctx.lineTo(barrelwidth3 / clientFovMultiplier, 0);
        ctx.lineTo(
          ((barrelwidth3 / 3) * 2) / clientFovMultiplier,
          -barrelheight3 / clientFovMultiplier
        );
        ctx.lineTo(
          ((-barrelwidth3 / 3) * 2) / clientFovMultiplier,
          -barrelheight3 / clientFovMultiplier
        );
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.restore();
        //rectangle
        ctx.save();
        ctx.translate(
          barrelX2 / clientFovMultiplier,
          barrelY2 / clientFovMultiplier
        );
        ctx.rotate(((barAngle - 90) * Math.PI) / 180);
        ctx.fillRect(
          -barrelwidth2 / 2 / clientFovMultiplier,
          -barrelheight2 / clientFovMultiplier,
          barrelwidth2 / clientFovMultiplier,
          barrelheight2 / clientFovMultiplier
        );
        if (CRTP != 'simplistic') {
        ctx.strokeRect(
          -barrelwidth2 / 2 / clientFovMultiplier,
          -barrelheight2 / clientFovMultiplier,
          barrelwidth2 / clientFovMultiplier,
          barrelheight2 / clientFovMultiplier
        );
        };
        ctx.restore();
        //trapezium at the tip
        ctx.save();
        ctx.translate(
          barrelX / clientFovMultiplier,
          barrelY / clientFovMultiplier
        );
        ctx.rotate(((barAngle - 90) * Math.PI) / 180);
        ctx.beginPath();
        ctx.moveTo(-barrelwidth / 2 / clientFovMultiplier, 0);
        ctx.lineTo(
          -barrelwidth / clientFovMultiplier,
          -barrelheight / clientFovMultiplier
        );
        ctx.lineTo(
          barrelwidth / clientFovMultiplier,
          -barrelheight / clientFovMultiplier
        );
        ctx.lineTo(barrelwidth / 2 / clientFovMultiplier, 0);
        ctx.lineTo(-barrelwidth / 2 / clientFovMultiplier, 0);
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.restore();
      }

      for (let i = 0; i < object.sides; i++) {
        drawSancBarrel(i);
      }
      //draw aura
      ctx.fillStyle = object.auraColor;
      ctx.lineWidth = 4 / clientFovMultiplier;
      ctx.beginPath();
      ctx.moveTo(
        0 + (object.auraWidth / clientFovMultiplier) * Math.cos(0),
        0 + (object.auraWidth / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= object.sides + 1; i += 1) {
        ctx.lineTo(
          0 +
            (object.auraWidth / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / object.sides),
          0 +
            (object.auraWidth / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / object.sides)
        );
      }
      ctx.fill();
      ctx.lineJoin = "miter"; //change back
      ctx.restore();
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "player") {
      var spawnProtectionFlashDuration = 3; //higher number indicates longer duration between flashes.
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = object.deadOpacity;
      }
      //draw players
      ctx.save(); //save so later can restore
      //translate canvas to location of player so that the player is at 0,0 coordinates, allowing rotation around the center of player's body
      ctx.translate(drawingX, drawingY);

      let objectangle = object.angle;
      if (
        id == playerstring &&
        object.autorotate != "yes" &&
        object.fastautorotate != "yes"
      ) {
        //if this player is the tank that the client is controlling
        objectangle = clientAngle;
        ctx.rotate(clientAngle); //instead of using client's actual tank angle, use the angle to the mouse. this reduces lag effect
      } else {
        ctx.rotate(object.angle);
      }

      let spawnProtect = "no";
      if (object.spawnProtection < object.spawnProtectionDuration && object.spawnProtection % spawnProtectionFlashDuration == 0) {
        spawnProtect = "yes";
      }

      let playercolor = "undefined";
      let playeroutline = "undefined";
      let eternal = "no";
      if (object.team == "none") {
        if (id == playerstring) {
          playercolor = bodyColors.blue.col;
          playeroutline = bodyColors.blue.outline;
          if (object.hit > 0 || spawnProtect == "yes") {
            playercolor = bodyColors.blue.hitCol
            playeroutline = bodyColors.blue.hitOutline
          }
        }
        else{
          playercolor = bodyColors.red.col;
          playeroutline = bodyColors.red.outline;
          if (object.hit > 0 || spawnProtect == "yes") {
            playercolor = bodyColors.red.hitCol
            playeroutline = bodyColors.red.hitOutline
          }
        }
      } else if (object.team == "blue" || object.team == "green" || object.team == "red" || object.team == "purple" || object.team == "eternal" || object.team == "magenta" || object.team == "fallen" || object.team == "celestial") {
          playercolor = bodyColors[object.team].col;
          playeroutline = bodyColors[object.team].outline;
          if (object.hit > 0 || spawnProtect == "yes") {
            playercolor = bodyColors[object.team].hitCol;
            playeroutline = bodyColors[object.team].hitOutline;
          }
          if (object.team == "eternal"){
            eternal = "yes";
          }
      }
      if (object.developer == "yes") {
        //if a developer
        playercolor = object.color;
        playeroutline = object.outline;
      }

      //store player color for upgrade buttons
      if (id == playerstring){
        playerBodyCol = playercolor;
        playerBodyOutline = playeroutline;
      }

      drawPlayer(ctx, object, clientFovMultiplier, spawnProtect, playercolor, playeroutline, eternal, objectangle)//draw barrel and body
      ctx.restore(); //restore coordinates to saved

      //write player name if not the client's tank
      if (id != playerstring) {
        ctx.fillStyle = "white";
        ctx.strokeStyle = "black";
        ctx.lineWidth = 8 / clientFovMultiplier;
        ctx.font = "700 " + 35 / clientFovMultiplier + "px Roboto";
        ctx.textAlign = "center";
        ctx.miterLimit = 2;//prevent text spikes, alternative to linejoin round
        //ctx.lineJoin = "round"; //prevent spikes above the capital letter "M"
        //note: if you stroke then fill, the words will be thicker and nicer. If you fill then stroke, the words are thinner.
        if (object.name == "unnamed"){
          //this guy is unnamed, add a 3 digit identifier
          let thisID = id.substr(id.length - 3);//last 3 digits of ID
          object.name += (" #" + thisID);
        }
        ctx.strokeText(
          object.name,
          drawingX,
          drawingY - (object.width + 40) / clientFovMultiplier
        );
        ctx.fillText(
          object.name,
          drawingX,
          drawingY - (object.width + 40) / clientFovMultiplier
        );
        //write player level
        ctx.font = "700 " + 18 / clientFovMultiplier + "px Roboto";
        ctx.strokeText(
          "Lvl " +
            object.level +
            " " +
            object.tankType +
            "-" +
            object.bodyType,
          drawingX,
          drawingY - (object.width + 10) / clientFovMultiplier
        );
        ctx.fillText(
          "Lvl " +
            object.level +
            " " +
            object.tankType +
            "-" +
            object.bodyType,
          drawingX,
          drawingY - (object.width + 10) / clientFovMultiplier
        );
        ctx.lineJoin = "miter"; //change it back
      }
      //draw player health
      if (object.health < object.maxhealth) {
        //draw health bar background
        var w = (object.width / clientFovMultiplier) * 2;
        var h = 7 / clientFovMultiplier;
        var r = h / 2;
        var x = drawingX - object.width / clientFovMultiplier;
        var y = drawingY + object.width / clientFovMultiplier + 10;
        ctx.fillStyle = "black";
        ctx.strokeStyle = "black";
        ctx.lineWidth = 2.5 / clientFovMultiplier;
        ctx.beginPath();
        ctx.moveTo(x + r, y);
        ctx.arcTo(x + w, y, x + w, y + h, r);
        ctx.arcTo(x + w, y + h, x, y + h, r);
        ctx.arcTo(x, y + h, x, y, r);
        ctx.arcTo(x, y, x + w, y, r);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        //draw health bar
        if (object.health > 0) {
          w = (w / object.maxhealth) * object.health;
          //if (id == playerstring) {
            //if this player is the tank that the client is controlling
            if (object.team == "none") {
              if (id == playerstring) {
                ctx.fillStyle = bodyColors.blue.col;
              }
              else{
                ctx.fillStyle = bodyColors.red.col;
              }
            } else if (object.team == "blue" || object.team == "red" || object.team == "purple" || object.team == "green" || object.team == "eternal" || object.team == "magenta" || object.team == "fallen" || object.team == "celestial") {
              ctx.fillStyle = bodyColors[object.team].col;
            }
          if (r * 2 > w) {
            //prevent weird shape when radius more than width
            r = w / 2;
            y += (h - w) / 2; //move health bar so that it is centered vertically in black bar
            h = w;
          }
          ctx.beginPath();
          ctx.moveTo(x + r, y);
          ctx.arcTo(x + w, y, x + w, y + h, r);
          ctx.arcTo(x + w, y + h, x, y + h, r);
          ctx.arcTo(x, y + h, x, y, r);
          ctx.arcTo(x, y, x + w, y, r);
          ctx.closePath();
          ctx.fill();
          ctx.stroke();
        }
      }

      //write chats
      if (chatstate) {
//write chats
      if (id != playerstring) {
        var firstChatY = object.width / clientFovMultiplier /5*4 + 55 / clientFovMultiplier;
      }
      else{
        var firstChatY = object.width / clientFovMultiplier /5*4;//chat nearer to player body if no need to display name
      }
      ctx.font = "700 25px Roboto";
      ctx.textAlign = "center";
      ctx.lineJoin = "round"; //prevent spikes above the capital letter "M"
      var xpadding = 15;
      var ypadding = 10;
      var lineheight = 30;

      var timeWhenChatRemove = 200;//when change on server code, remember to change here too

      if (!(chatlist[id])){//used for animating chat positions
        chatlist[id] = JSON.parse(JSON.stringify(object.chats));
      }
      else{
        let tempArray = [];
        let messages = {};//prevent bug when multiple chats have same message
        object.chats.forEach(function (item, index) {
          let occurence = 0;//prevent bug when multiple chats have same message
          let foundit = 0;
          for (var i = 0; i < chatlist[id].length; i++) {//check if oldchats hae this message, to preserve the position for animation
            if (chatlist[id][i].chat == item.chat){
              if (messages[item.chat]){//saw a chat with the exact same message before!
                if (messages[item.chat] <= occurence){//this is a different chat
                  let k = JSON.parse(JSON.stringify(chatlist[id][i]));
                  k.time = item.time;
                  tempArray.push(k);
                  messages[chatlist[id][i].chat]++;
                  foundit = 1;
                  break
                }
                else{//this is the same chat that you saw before, continue hunting for the chat
                  occurence++;
                }
              }
              else{
                let k = JSON.parse(JSON.stringify(chatlist[id][i]));
                k.time = item.time;
                tempArray.push(k);
                messages[chatlist[id][i].chat] = 1;
                foundit = 1;
                break
              }
            }
          }
          if (foundit == 0){//new chat message
            let k = JSON.parse(JSON.stringify(item));
            k.opacity = 0;
            tempArray.push(k);
          }
        });
        chatlist[id] = tempArray;
      }

      object.chats.slice().reverse().forEach((chatObj, index) => {//slice and reverse to loop though array backwards (so older messages are above)
        ctx.fillStyle = "rgba(69,69,69,.7)";

        var longestLine = 0;

        //multiline chat
        const wrapText = function(ctx, text, x, y, maxWidth, lineHeight) {
          // First, start by splitting all of our text into words, but splitting it into an array split by spaces
          let words = text.split(' ');
          let line = ''; // This will store the text of the current line
          let testLine = ''; // This will store the text when we add a word, to test if it's too long
          let lineArray = []; // This is an array of lines, which the function will return

          // Lets iterate over each word
          for(var n = 0; n < words.length; n++) {
              // Create a test line, and measure it..
              testLine += `${words[n]} `;
              let metrics = ctx.measureText(testLine);
              let testWidth = metrics.width;
              // If the width of this test line is more than the max width
              if (testWidth > maxWidth && n > 0) {
                  // Then the line is finished, push the current line into "lineArray"
                  line = line.slice(0, -1);//remove space at the end of the line
                  lineArray.push([line, x, y]);
                  let thislinewidth = ctx.measureText(line).width;
                  if (thislinewidth > longestLine){
                    longestLine = thislinewidth;
                  }
                  // Increase the line height, so a new line is started
                  y += lineHeight;
                  // Update line and test line to use this word as the first word on the next line
                  line = `${words[n]} `;
                  testLine = `${words[n]} `;
              }
              else {
                  // If the test line is still less than the max width, then add the word to the current line
                  line += `${words[n]} `;
              }
              // If we never reach the full max width, then there is only one line.. so push it into the lineArray so we return something
              if(n === words.length - 1) {
                  line = line.slice(0, -1);//remove space at the end of the line
                  lineArray.push([line, x, y]);
                  let thislinewidth = ctx.measureText(line).width;
                  if (thislinewidth > longestLine){
                    longestLine = thislinewidth;
                  }
              }
          }
          // Return the line array
          return lineArray;
        }

        let wrappedText = wrapText(ctx, chatObj.chat, drawingX, drawingY - firstChatY, 900, lineheight);//split message into multiline text
        //draw rect
        var w = longestLine + xpadding * 2;
        var h = lineheight * wrappedText.length + ypadding * 2;
        if (wrappedText.length == 1){//remove spacing between text for single-line text
          h = 25 + ypadding * 2;
        }
        var r = 15;
        var x = drawingX - longestLine / 2 - xpadding;
        var y = drawingY - firstChatY - ypadding - h - 20;//the actual y location of this chat message
        //aniamte towards this y position
        //remember that the loop is reversed, so indexes are reversed here too
        let thischat = chatlist[id][chatlist[id].length - 1 - index];
        let diffpos = 0;
        if (!thischat.y){
          thischat.y = y;
        }
        else{
          if (y > thischat.y){
            thischat.y+=(y - thischat.y)/2*deltaTime;
            if (y < thischat.y){
              thischat.y = y;
            }
          }
          else if (y < thischat.y){
            thischat.y-=(thischat.y - y)/2*deltaTime;
            if (y > thischat.y){
              thischat.y = y;
            }
          }
          if (Math.abs(y - thischat.y)<0.1){//small difference between current position and actual position
            thischat.y = y;
          }
          diffpos = y - thischat.y;
          y = thischat.y;
        }
        if (thischat.opacity < 1){
          thischat.opacity+=0.1;
        }
        ctx.globalAlpha = thischat.opacity;
        ctx.beginPath();
        ctx.moveTo(x + r, y);
        ctx.arcTo(x + w, y, x + w, y + h, r);
        ctx.arcTo(x + w, y + h, x, y + h, r);
        ctx.arcTo(x, y + h, x, y, r);
        ctx.arcTo(x, y, x + w, y, r);
        ctx.closePath();
        ctx.fill();
        if (index == 0){
          //if this is first chat message, draw triangle
          let trianglewidth = 20;
          let triangleheight = 10;
          ctx.beginPath();
          ctx.moveTo(x + w/2 - trianglewidth/2, y + h);
          ctx.lineTo(x + w/2 + trianglewidth/2, y + h);
          ctx.lineTo(x + w/2, y + h + triangleheight);
          ctx.fill();
        }
        //write words
        ctx.fillStyle = "white";
        wrappedText.forEach(function(item) {
            ctx.fillText(item[0], item[1], item[2]-h-diffpos);//write text
        })
        ctx.globalAlpha = 1.0;
        firstChatY += (h + 10); //height of chat plus space between chats
      });
      ctx.lineJoin = "miter"; //change it back
      }
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = 1.0; //reset opacity
      }
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "portal") {
      //draw the aura below the portal
      var auraSpeed = 75; //higher number means slower speed
      var auraWidth = 4; //reative to portal size
      var portalAuraSize = object.timer % auraSpeed;
      var portalwidth = portalwidths[id]; //use this for portal width. it keeps track size changes when players touch portal
      var portalsizeincrease = portalwidths[id] / object.width; //increase in width when someone touch it (needed for the spikes)
      //first aura
      var opacityCalculation =
        1 - ((auraWidth / auraSpeed) * portalAuraSize) / auraWidth; //goes from 0 to 0.3
      if (opacityCalculation > 0.3) {
        //max opacity for portal aura
        opacityCalculation = 0.3;
      }
      if (object.hasOwnProperty("red")) {
        //if portal is radiant
        ctx.fillStyle =
          "rgba(" +
          object.red +
          ", " +
          object.green +
          ", " +
          object.blue +
          "," +
          opacityCalculation +
          ")";
      } else {
        ctx.fillStyle =
          "rgba(" + object.color + "," + opacityCalculation + ")";
      }
      ctx.beginPath();
      ctx.arc(
        drawingX,
        drawingY,
        (portalwidth * ((auraWidth / auraSpeed) * portalAuraSize)) /
          clientFovMultiplier,
        0,
        2 * Math.PI
      );
      ctx.fill();
      //second smaller aura
      portalAuraSize = (object.timer - auraSpeed / 2) % auraSpeed;
      if (portalAuraSize > 0) {
        var opacityCalculation =
          1 - ((auraWidth / auraSpeed) * portalAuraSize) / auraWidth;
        if (opacityCalculation > 0.3) {
          //max opacity for portal aura
          opacityCalculation = 0.3;
        }
        if (object.hasOwnProperty("red")) {
          //if portal is radiant
          ctx.fillStyle =
            "rgba(" +
            object.red +
            ", " +
            object.green +
            ", " +
            object.blue +
            "," +
            opacityCalculation +
            ")";
        } else {
          ctx.fillStyle =
            "rgba(" + object.color + "," + opacityCalculation + ")";
        }
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          (portalwidth * ((auraWidth / auraSpeed) * portalAuraSize)) /
            clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.fill();
      }

      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = object.deadOpacity;
      }
      //drawing portals
      //create gradient
      //const gradient = ctx.createRadialGradient(drawingX, drawingY, object.width/3/clientFovMultiplier, drawingX, drawingY, object.width/clientFovMultiplier);

      // Add two color stops
      //caluclate color of outline of portal based on time until it die
      var portalColorCalc = object.timer / object.maxtimer;
      var portalColor = 255 - portalColorCalc * 255;
      var portalRGB =
        "rgb(" +
        portalColor +
        "," +
        portalColor +
        "," +
        portalColor +
        ")";
      var portalRGBoutline =
        "rgb(" +
        (portalColor - 20) +
        "," +
        (portalColor - 20) +
        "," +
        (portalColor - 20) +
        ")";
      if (object.ruptured == 1) {
        //portal is ruptured!
        //draw the stars
        ctx.save(); //save so later can restore
        ctx.translate(drawingX, drawingY);
        ctx.fillStyle = "white";
        ctx.strokeStyle = "lightgrey";
        ctx.lineWidth = 3 / clientFovMultiplier;
        ctx.lineJoin = "round";
        //first star: 3 spikes
        ctx.rotate((extraSpikeRotate * Math.PI) / 180);
        var numberOfSpikes = 3;
        var outerRadius =
          ((object.width * 3) / clientFovMultiplier) * portalsizeincrease;
        var innerRadius =
          (object.width / 3 / clientFovMultiplier) * portalsizeincrease;
        var rot = (Math.PI / 2) * 3;
        var x = 0;
        var y = 0;
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        ctx.rotate((-extraSpikeRotate * Math.PI) / 180);
        //second star: 6 spikes in opposite direction
        ctx.rotate(((360 - extraSpikeRotate) * 2 * Math.PI) / 180);
        var numberOfSpikes = 6;
        var outerRadius =
          ((object.width * 1.5) / clientFovMultiplier) *
          portalsizeincrease;
        var innerRadius =
          (object.width / 1.2 / clientFovMultiplier) * portalsizeincrease;
        var rot = (Math.PI / 2) * 3;
        var x = 0;
        var y = 0;
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        ctx.rotate((-(360 - extraSpikeRotate) * 2 * Math.PI) / 180);
        //third star: 6 spikes
        ctx.rotate((extraSpikeRotate * 2 * Math.PI) / 180);
        var numberOfSpikes = 6;
        var outerRadius =
          ((object.width * 1.5) / clientFovMultiplier) *
          portalsizeincrease;
        var innerRadius =
          (object.width / 1.2 / clientFovMultiplier) * portalsizeincrease;
        var rot = (Math.PI / 2) * 3;
        var x = 0;
        var y = 0;
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        ctx.rotate((-extraSpikeRotate * 2 * Math.PI) / 180);
        //fourth star: 6 dark spikes in opposite direction
        ctx.fillStyle = portalRGB;
        ctx.strokeStyle = portalRGBoutline;
        ctx.rotate(((360 - extraSpikeRotate) * 3 * Math.PI) / 180); //times 2 to make it faster
        var numberOfSpikes = 6;
        var outerRadius =
          ((object.width * 1.5) / clientFovMultiplier) *
          portalsizeincrease;
        var innerRadius =
          (object.width / 2 / clientFovMultiplier) * portalsizeincrease;
        var rot = (Math.PI / 2) * 3;
        var x = 0;
        var y = 0;
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        ctx.rotate((-(360 - extraSpikeRotate) * 3 * Math.PI) / 180);
        //fifth star: tiny black spikes
        ctx.rotate((extraSpikeRotate * 3 * Math.PI) / 180); //times 2 to make it faster
        var numberOfSpikes = 6;
        var outerRadius =
          ((object.width * 1.25) / clientFovMultiplier) *
          portalsizeincrease;
        var innerRadius =
          (object.width / 4 / clientFovMultiplier) * portalsizeincrease;
        var rot = (Math.PI / 2) * 3;
        var x = 0;
        var y = 0;
        ctx.beginPath();
        ctx.moveTo(0, 0 - outerRadius);
        for (i = 0; i < numberOfSpikes; i++) {
          x = 0 + Math.cos(rot) * outerRadius;
          y = 0 + Math.sin(rot) * outerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
          x = 0 + Math.cos(rot) * innerRadius;
          y = 0 + Math.sin(rot) * innerRadius;
          ctx.lineTo(x, y);
          rot += Math.PI / numberOfSpikes;
        }
        ctx.lineTo(0, 0 - outerRadius);
        ctx.closePath();
        ctx.fill();
        ctx.stroke();
        ctx.rotate((-extraSpikeRotate * 3 * Math.PI) / 180);
        ctx.restore();
        ctx.lineJoin = "miter";
      }
      ctx.fillStyle = portalRGB;
      ctx.strokeStyle = portalRGBoutline;
      ctx.lineWidth = 3 / clientFovMultiplier;
      ctx.beginPath();
      ctx.arc(
        drawingX,
        drawingY,
        portalwidth / clientFovMultiplier,
        0,
        2 * Math.PI
      );
      ctx.fill();
      ctx.stroke();
      if (object.hasOwnProperty("deadOpacity")) {
        //if this is an animation of a dead object
        ctx.globalAlpha = 1.0; //reset opacity
      }

      //spawn particles
      var choosing = Math.floor(Math.random() * 3); //choose if particle spawn. Lower number means more particles
      if (choosing == 1) {
        var angleDegrees = Math.floor(Math.random() * 360); //choose angle in degrees
        var angleRadians = (angleDegrees * Math.PI) / 180; //convert to radians
        portalparticles[particleID] = {
          angle: angleRadians,
          x: object.x,
          y: object.y,
          width: 50,
          height: 50,
          speed: 10,
          timer: 30,
          maxtimer: 15, //difference between timer and maxtimer is the opacity change of the particle. Larger difference means more or less transparent
          color: "white",
          outline: "lightgrey",
          type: "particle",
        };
        particleID++;
      }

      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "Fixedportal") {
      //drawing rectangular fixed portals, e.g. the portal at top left corner of dune
      ctx.save(); //save so later can restore
      ctx.translate(drawingX, drawingY); //translate so white portal is at 0,0 coordinates so can rotate around center of portal
      ctx.rotate((object.angleDegrees * Math.PI) / 180); //rotate portal
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.fillRect(
        -object.width / 2 / clientFovMultiplier,
        -object.height / 2 / clientFovMultiplier,
        object.width / clientFovMultiplier,
        object.height / clientFovMultiplier
      );
      ctx.strokeRect(
        -object.width / 2 / clientFovMultiplier,
        -object.height / 2 / clientFovMultiplier,
        object.width / clientFovMultiplier,
        object.height / clientFovMultiplier
      );
      ctx.globalAlpha = 0.7; //transparency
      ctx.fillStyle = object.color2;
      ctx.fillRect(
        -object.width / clientFovMultiplier,
        -object.height / clientFovMultiplier,
        (object.width * 2) / clientFovMultiplier,
        (object.height * 2) / clientFovMultiplier
      );
      ctx.strokeRect(
        -object.width / clientFovMultiplier,
        -object.height / clientFovMultiplier,
        (object.width * 2) / clientFovMultiplier,
        (object.height * 2) / clientFovMultiplier
      );
      ctx.globalAlpha = 1.0; //reset transparency
      ctx.restore(); //restore after translating
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / 2 / clientFovMultiplier,
          0,
          2 * Math.PI
        );
        ctx.stroke();
      }
    } else if (object.type == "particle") {
      //draw particles
      if (object.timer <= 10){
        ctx.globalAlpha = object.timer / 10;
      }
      //ctx.globalAlpha = object.timer / object.maxtimer;
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.lineWidth = 3 / clientFovMultiplier;
      ctx.beginPath();
      ctx.arc(
        drawingX,
        drawingY,
        object.width / clientFovMultiplier,
        0,
        2 * Math.PI
      );
      ctx.fill();
      ctx.stroke();
      ctx.globalAlpha = 1.0;
    } else if (object.type == "wall") {
      //ctx.fillStyle = "#232323";
      ctx.fillStyle = "rgba(15, 15, 15, .5)";
      ctx.fillRect(
        drawingX,
        drawingY,
        object.w / clientFovMultiplier,
        object.h / clientFovMultiplier
      );
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.strokeRect(
          drawingX,
          drawingY,
          object.w / clientFovMultiplier,
          object.h / clientFovMultiplier
        );
      }
    } else if (object.type == "gate") {
      //ctx.fillStyle = "#232323";
      ctx.save();
      ctx.translate(drawingX, drawingY);
      ctx.rotate(object.angle/180*Math.PI);
      //draw white rectangle below
      ctx.fillStyle = "rgba(255,255,255,.7)";
      ctx.strokeStyle = "white";
      //FIRST WHITE RECTANGLE
      ctx.globalAlpha = 1.0 * (endGate - gateTimer) / (endGate - 1 - startGate);//gateTimer increases from 0.5 to 9, this equation makes the opacity decrease from 1 to 0
      ctx.fillRect(
        -(object.height / clientFovMultiplier * gateTimer)/2 + object.height / clientFovMultiplier/2,
         -object.width/2/clientFovMultiplier,
        object.height / clientFovMultiplier * gateTimer,
        object.width / clientFovMultiplier
      );
      ctx.strokeRect(
        -(object.height / clientFovMultiplier * gateTimer)/2 + object.height / clientFovMultiplier/2,
         -object.width/2/clientFovMultiplier,
        object.height / clientFovMultiplier * gateTimer,
        object.width / clientFovMultiplier
      );
      ctx.globalAlpha = 1.0;
      //SECOND WHITE RECTANGLE
      let gateTimer2 = gateTimer - endGate/2;
      if (gateTimer2 < startGate){
        gateTimer2 = endGate - (startGate - gateTimer2)
      }
      ctx.globalAlpha = 1.0 * (endGate - gateTimer2) / (endGate - 1 - startGate);//gateTimer increases from 1 to 7, this equation makes the opacity decrease from 1 to 0
      ctx.fillRect(
        -(object.height / clientFovMultiplier * gateTimer2)/2 + object.height / clientFovMultiplier/2,
         -object.width/2/clientFovMultiplier,
        object.height / clientFovMultiplier * gateTimer2,
        object.width / clientFovMultiplier
      );
      ctx.strokeRect(
        -(object.height / clientFovMultiplier * gateTimer2)/2 + object.height / clientFovMultiplier/2,
         -object.width/2/clientFovMultiplier,
        object.height / clientFovMultiplier * gateTimer2,
        object.width / clientFovMultiplier
      );
      ctx.globalAlpha = 1.0;
      //draw actual black gate
      ctx.fillStyle = "black";
      ctx.fillRect(0,
         -object.width/2/clientFovMultiplier,
        object.height / clientFovMultiplier,
        object.width / clientFovMultiplier
      );
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.strokeRect(0,
         -object.width/2/clientFovMultiplier,
          object.height / clientFovMultiplier,
          object.width / clientFovMultiplier
        );
      }
      ctx.restore();
      //spawn particles
      var choosing = Math.floor(Math.random() * 3); //choose if particle spawn. Lower number means more particles
      if (choosing == 1) {
        var dir = Math.floor(Math.random() * 2); //choose angle in degrees
        if (dir == 0){
          var angleRadians = (object.angle) * Math.PI / 180; //convert to radians
        }
        else{
          var angleRadians = (object.angle - 180) * Math.PI / 180;
        }
        let randX = 0;
        let randY = 0;
        //code currently does not support particles for gates that are tilted
        //i dont see a need to add that in the near future
        if (object.angle == 0 || object.angle == 180 || object.angle == 360){
          randY = Math.floor(Math.random() * object.width) - object.width/2;
        }
        else if (object.angle == 90 || object.angle == 270){
          randX = Math.floor(Math.random() * object.width) - object.width/2;
        }
        portalparticles[particleID] = {
          angle: angleRadians,
          x: object.x + randX,
          y: object.y + randY,
          width: 50,
          height: 50,
          speed: 10,
          timer: 30,
          maxtimer: 15, //difference between timer and maxtimer is the opacity change of the particle. Larger difference means more or less transparent
          color: "white",
          outline: "lightgrey",
          type: "particle",
        };
        particleID++;
      }
    } else if (object.type == "def") {
      //base defender in 2tdm
      ctx.save();
      ctx.translate(drawingX, drawingY);
      ctx.rotate(object.angle);
      ctx.lineJoin = "round"; //make corners of shape round
      ctx.lineWidth = 4 / clientFovMultiplier;

      //draw octagon base
      var octagonWidth = object.width/5*6;
      ctx.fillStyle = bodyColors.asset.col;
      ctx.strokeStyle = bodyColors.asset.outline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (octagonWidth / clientFovMultiplier) * Math.cos(0),
        0 + (octagonWidth / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= 8 + 1; i += 1) {
        ctx.lineTo(
          0 +
            (octagonWidth / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / 8),
          0 +
            (octagonWidth / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / 8)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };


      //draw barrels
      ctx.fillStyle = bodyColors.barrel.col;
      ctx.strokeStyle = bodyColors.barrel.outline;
      //trapezoid at the tip
      var barrelwidth = 70;
      var barrelheight = 20;
      //rectangle
      var barrelwidth2 = 90;
      var barrelheight2 = 20;
      //base trapezoid
      var barrelwidth3 = 70;
      var barrelheight3 = 60;
      //note that trapezoids and rectangles are drawn differently

      for (let i = 0; i < 4; i++) {//draw 4 barrels
        var barrelAngle = 360/4*i;
        var barrelX = Math.cos((barrelAngle * Math.PI) / 180) * object.width * 1.4;
        var barrelY = Math.sin((barrelAngle * Math.PI) / 180) * object.width * 1.4;
        var barrelX2 =
          Math.cos((barrelAngle * Math.PI) / 180) *
          (object.width * 1.4 - barrelheight); //move rectangle barrel downwards
        var barrelY2 =
          Math.sin((barrelAngle * Math.PI) / 180) *
          (object.width * 1.4 - barrelheight);
        var barrelX3 =
          Math.cos((barrelAngle * Math.PI) / 180) *
          (object.width * 1.4 - barrelheight - barrelheight2); //move base trapezoid barrel downwards
        var barrelY3 =
          Math.sin((barrelAngle * Math.PI) / 180) *
          (object.width * 1.4 - barrelheight - barrelheight2);
        //base trapezoid
        ctx.save();
        ctx.translate(
          barrelX3 / clientFovMultiplier,
          barrelY3 / clientFovMultiplier
        );
        ctx.rotate(((barrelAngle - 90) * Math.PI) / 180);
        ctx.beginPath();
        ctx.moveTo(
          ((-barrelwidth3 / 3) * 2) / clientFovMultiplier,
          -barrelheight3 / clientFovMultiplier
        );
        ctx.lineTo(-barrelwidth3 / clientFovMultiplier, 0);
        ctx.lineTo(barrelwidth3 / clientFovMultiplier, 0);
        ctx.lineTo(
          ((barrelwidth3 / 3) * 2) / clientFovMultiplier,
          -barrelheight3 / clientFovMultiplier
        );
        ctx.lineTo(
          ((-barrelwidth3 / 3) * 2) / clientFovMultiplier,
          -barrelheight3 / clientFovMultiplier
        );
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.restore();
        //rectangle
        ctx.save();
        ctx.translate(
          barrelX2 / clientFovMultiplier,
          barrelY2 / clientFovMultiplier
        );
        ctx.rotate(((barrelAngle - 90) * Math.PI) / 180);
        ctx.fillRect(
          -barrelwidth2 / 2 / clientFovMultiplier,
          -barrelheight2 / clientFovMultiplier,
          barrelwidth2 / clientFovMultiplier,
          barrelheight2 / clientFovMultiplier
        );
        if (CRTP != 'simplistic') {
        ctx.strokeRect(
          -barrelwidth2 / 2 / clientFovMultiplier,
          -barrelheight2 / clientFovMultiplier,
          barrelwidth2 / clientFovMultiplier,
          barrelheight2 / clientFovMultiplier
        );
        };
        ctx.restore();
        //trapezium at the tip
        ctx.save();
        ctx.translate(
          barrelX / clientFovMultiplier,
          barrelY / clientFovMultiplier
        );
        ctx.rotate(((barrelAngle - 90) * Math.PI) / 180);
        ctx.beginPath();
        ctx.moveTo(-barrelwidth / 2 / clientFovMultiplier, 0);
        ctx.lineTo(
          -barrelwidth / clientFovMultiplier,
          -barrelheight / clientFovMultiplier
        );
        ctx.lineTo(
          barrelwidth / clientFovMultiplier,
          -barrelheight / clientFovMultiplier
        );
        ctx.lineTo(barrelwidth / 2 / clientFovMultiplier, 0);
        ctx.lineTo(-barrelwidth / 2 / clientFovMultiplier, 0);
        ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
        ctx.restore();
      }

      //draw body
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.beginPath();
      ctx.arc(
        0,
        0,
        object.width / clientFovMultiplier,
        0,
        2 * Math.PI
      );
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      var octagonWidth = object.width/5*4;
      ctx.fillStyle = bodyColors.asset.col;
      ctx.strokeStyle = bodyColors.asset.outline;
      ctx.beginPath();
      ctx.moveTo(
        0 + (octagonWidth / clientFovMultiplier) * Math.cos(0),
        0 + (octagonWidth / clientFovMultiplier) * Math.sin(0)
      );
      for (var i = 1; i <= 8 + 1; i += 1) {
        ctx.lineTo(
          0 +
            (octagonWidth / clientFovMultiplier) *
              Math.cos((i * 2 * Math.PI) / 8),
          0 +
            (octagonWidth / clientFovMultiplier) *
              Math.sin((i * 2 * Math.PI) / 8)
        );
      }
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      ctx.fillStyle = object.color;
      ctx.strokeStyle = object.outline;
      ctx.beginPath();
      ctx.arc(
        0,
        0,
        object.width/2 / clientFovMultiplier,
        0,
        2 * Math.PI
      );
      ctx.fill();
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };

      ctx.lineJoin = "miter"; //change back
      ctx.restore();
      if (showHitBox == "yes") {
        //draw hitbox
        ctx.strokeStyle = "blue";
        ctx.lineWidth = 3;
        ctx.beginPath();
        ctx.arc(
          drawingX,
          drawingY,
          object.width / clientFovMultiplier,
          0,
          2 * Math.PI
        );
          if (CRTP != 'simplistic') {
          ctx.stroke();
          };
      }
    }
  };

    function newbulletbarrel(canvas, x,width,height,shootChange, fov){//shootchange is change in barrel height when shooting
    canvas.fillRect(
      (x - width / 2) / fov,
      -(height - shootChange) / fov,
      width / fov,
      (height - shootChange) / fov
    );
    if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (x - width / 2) / fov,
      -(height - shootChange) / fov,
      width / fov,
      (height - shootChange) / fov
    );
    };
  };
    function newdronebarrel(canvas, x,width,height,shootChange, fov){
    canvas.beginPath();
    canvas.moveTo(
      -width / 2 / fov +
        x / fov,
      0
    );
    canvas.lineTo(
      -width / fov +
        x / fov,
      -(height - shootChange) / fov
    );
    canvas.lineTo(
      width / fov +
        (x * 2) / fov,
      -(height - shootChange) / fov
    );
    canvas.lineTo(
      width / 2 / fov +
        (x * 2) / fov,
      0
    );
    canvas.fill();
        if (document.getElementById('theme').value != 'simplistic') {
    canvas.stroke();
        };
  };
    function newtrapbarrel(canvas, x,width,height,shootChange, fov){
    canvas.fillRect(
      (x - width / 2) / fov,
      -(height - shootChange) * 0.67 / fov,
      width / fov,
      (height - shootChange) * 0.67 / fov
    );
            if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (x - width / 2) / fov,
      -(height - shootChange) * 0.67 / fov,
      width / fov,
      (height - shootChange) * 0.67 / fov
    );
            };
    canvas.beginPath();
    canvas.moveTo(
      (x - width / 2) / fov,
      -(height - shootChange) * 0.67 / fov
    );
    canvas.lineTo(
      (x - width) / fov,
      -(height - shootChange) / fov
    );
    canvas.lineTo(
      (x + width) / fov,
      -(height - shootChange) / fov
    );
    canvas.lineTo(
      (x + width / 2) / fov,
      -(height - shootChange) * 0.67 / fov
    );
    canvas.fill();
        if (document.getElementById('theme').value != 'simplistic') {
    canvas.stroke();
        };
  };
    function newminebarrel(canvas, x,width,height,shootChange, fov){
    canvas.fillRect(
      (x - width / 2) / fov,
      -(height - shootChange) / fov,
      width / fov,
      (height - shootChange) / fov
    );
    if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (x - width / 2) / fov,
      -(height - shootChange) / fov,
      width / fov,
      (height - shootChange) / fov
    );
    };
    canvas.fillRect(
      (-width * 1.5) / 2 / fov + x / fov,
      -(height - shootChange) * 0.67 / fov,
      (width / fov) * 1.5,
      (height - shootChange) * 0.67 / fov
    );
     if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (-width * 1.5) / 2 / fov + x / fov,
      -(height - shootChange) * 0.67 / fov,
      (width / fov) * 1.5,
      (height - shootChange) * 0.67 / fov
    );
     };
  };
    function newminionbarrel(canvas, x,width,height,shootChange, fov){
    canvas.fillRect(
      (x - width / 2) / fov,
      -(height - shootChange) / fov,
      width / fov,
      (height - shootChange) / fov
    );
    if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (x - width / 2) / fov,
      -(height - shootChange) / fov,
      width / fov,
      (height - shootChange) / fov
    );
    };
    canvas.fillRect(
      (x - width * 0.75) / fov,
      -(height - shootChange) / 1.5 / fov,
      (width / fov) * 1.5,
      (height - shootChange) / 1.5 / fov
    );
    if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (x - width * 0.75) / fov,
      -(height - shootChange) / 1.5 / fov,
      (width / fov) * 1.5,
      (height - shootChange) / 1.5 / fov
    );
    };
    canvas.fillRect(
      (x - width * 0.75) / fov,
      -(height - shootChange) / fov,
      (width / fov) * 1.5,
      (height - shootChange) / 5 / fov
    );
    if (document.getElementById('theme').value != 'simplistic') {
    canvas.strokeRect(
      (x - width * 0.75) / fov,
      -(height - shootChange) / fov,
      (width / fov) * 1.5,
      (height - shootChange) / 5 /fov
    );
    };
  };

    drawobjects = newdraw;
    drawPlayer = newdrawplayer;
    drawBulletBarrel = newbulletbarrel; drawDroneBarrel = newdronebarrel; drawTrapBarrel = newtrapbarrel; drawMineBarrel = newminebarrel; drawMinionBarrel = newminionbarrel;

    let pd = false;
    const editloop = () => {
        if (player.health <= 1 && !pd && gameStart == -1) {
            if (autorespawn) {
                document.getElementById('continue').click();
                document.getElementById('play').click();
            };
            pd = true;
        } else if (player.health > 1) {
            pd = false;
        };
        requestAnimationFrame(editloop);
    };
    editloop();
})();