let nonRolledIDs = ["name", "displayName", "tier", "set", "slots", "type", "material", "drop", "quest", "restrict", "nDam", "fDam", "wDam", "aDam", "tDam", "eDam", "atkSpd", "hp", "fDef", "wDef", "aDef", "tDef", "eDef", "lvl", "classReq", "strReq", "dexReq", "intReq", "defReq", "agiReq","str", "dex", "int", "agi", "def", "fixID", "category", "id", "skillpoints", "reqs", "nDam_", "fDam_", "wDam_", "aDam_", "tDam_", "eDam_"]; let rolledIDs = ["hprPct", "mr", "sdPct", "mdPct", "ls", "ms", "xpb", "lb", "ref", "thorns", "expd", "spd", "atkTier", "poison", "hpBonus", "spRegen", "eSteal", "hprRaw", "sdRaw", "mdRaw", "fDamPct", "wDamPct", "aDamPct", "tDamPct", "eDamPct", "fDefPct", "wDefPct", "aDefPct", "tDefPct", "eDefPct", "spPct1", "spRaw1", "spPct2", "spRaw2", "spPct3", "spRaw3", "spPct4", "spRaw4", "rainbowRaw", "sprint", "sprintReg", "jh", "lq", "gXp", "gSpd"]; let damageClasses = ["Neutral","Earth","Thunder","Water","Fire","Air"]; let reversedIDs = [ "spPct1", "spRaw1", "spPct2", "spRaw2", "spPct3", "spRaw3", "spPct4", "spRaw4" ]; function expandItem(item, powders){ let minRolls = new Map(); let maxRolls = new Map(); let expandedItem = new Map(); if(item.fixID){ //The item has fixed IDs. expandedItem.set("fixID",true); for (const id of rolledIDs){ //all rolled IDs are numerical let val = (item[id] || 0); //if(item[id]) { minRolls.set(id,val); maxRolls.set(id,val); //} } }else{ //The item does not have fixed IDs. for (const id of rolledIDs){ let val = (item[id] || 0); if(val > 0){ // positive rolled IDs if (reversedIDs.includes(id)) { maxRolls.set(id,idRound(val*0.3)); minRolls.set(id,idRound(val*1.3)); } else { maxRolls.set(id,idRound(val*1.3)); minRolls.set(id,idRound(val*0.3)); } }else if(val < 0){ //negative rolled IDs if (reversedIDs.includes(id)) { maxRolls.set(id,idRound(val*1.3)); minRolls.set(id,idRound(val*0.7)); } else { minRolls.set(id,idRound(val*1.3)); maxRolls.set(id,idRound(val*0.7)); } }else{//Id = 0 minRolls.set(id,0); maxRolls.set(id,0); } } } for (const id of nonRolledIDs){ expandedItem.set(id,item[id]); } expandedItem.set("minRolls",minRolls); expandedItem.set("maxRolls",maxRolls); expandedItem.set("powders", powders); if(expandedItem.get("category") == "armor"){ //item is armor for(const id of powders){ //console.log(powderStats[id]); let powder = powderStats[id]; let name = powderNames.get(id); expandedItem.set(name.charAt(0) + "Def", expandedItem.get(name.charAt(0)+"Def") + powder["defPlus"]); expandedItem.set(skp_elements[(skp_elements.indexOf(name.charAt(0)) + 6 )% 5] + "Def", expandedItem.get(skp_elements[(skp_elements.indexOf(name.charAt(0)) + 6 )% 5]+"Def") - powder["defMinus"]); } } return expandedItem; } /*An independent helper function that rounds a rolled ID to the nearest integer OR brings the roll away from 0. * @param id */ function idRound(id){ rounded = Math.round(id); if(rounded == 0){ return 1; //this is a hack, will need changing along w/ rest of ID system if anything changes }else{ return rounded; } } let idPrefixes = {"displayName": "", "lvl":"Combat Level Min: ", "classReq":"Class Req: ","strReq":"Strength Min: ","dexReq":"Dexterity Min: ","intReq":"Intelligence Min: ","defReq":"Defense Min: ","agiReq":"Agility Min: ", "nDam_":"Neutral Damage: ", "eDam_":"Earth Damage: ", "tDam_":"Thunder Damage: ", "wDam_":"Water Damage: ", "fDam_":"Fire Damage: ", "aDam_":"Air Damage: ", "atkSpd":"Attack Speed: ", "hp":"Health : ", "eDef":"Earth Defense: ", "tDef":"Thunder Defense: ", "wDef":"Water Defense: ", "fDef":"Fire Defense: ", "aDef":"Air Defense: ", "str":"Strength: ", "dex":"Dexterity: ", "int":"Intelligence: ", "def":"Defense: ","agi":"Agility: ", "hpBonus":"Health Bonus: ", "hprRaw":"Health Regen Raw: ", "hprPct":"Health Regen %: ", "sdRaw":"Raw Spell Damage: ", "sdPct":"Spell Damage %: ", "mdRaw":"Raw Melee Damage: ", "mdPct":"Melee Damage %: ", "mr":"Mana Regen: ", "ms":"Mana Steal: ", "ref":"Reflection: ", "ls":"Life Steal: ", "poison":"Poison: ", "thorns":"Thorns: ", "expd":"Exploding: ", "spd":"Walk Speed Bonus: ", "atkTier":"Attack Speed Bonus: ", "eDamPct":"Earth Damage %: ", "tDamPct":"Thunder Damage %: ", "wDamPct":"Water Damage %: ", "fDamPct":"Fire Damage %: ", "aDamPct":"Air Damage %: ", "eDefPct":"Earth Defense %: ", "tDefPct":"Thunder Defense %: ", "wDefPct":"Water Defense %: ", "fDefPct":"Fire Defense %: ", "aDefPct":"Air Defense %: ", "spPct1":"1st Spell Cost %: ", "spRaw1":"1st Spell Cost Raw: ", "spPct2":"2nd Spell Cost %: ", "spRaw2":"2nd Spell Cost Raw: ", "spPct3":"3rd Spell Cost %: ", "spRaw3":"3rd Spell Cost Raw: ", "spPct4":"4th Spell Cost %: ", "spRaw4":"4th Spell Cost Raw: ", "rainbowRaw":"Rainbow Spell Damage Raw: ", "sprint":"Sprint Bonus: ", "sprintReg":"Sprint Regen Bonus: ", "jh":"Jump Height: ", "xpb":"Combat XP Bonus: ", "lb":"Loot Bonus: ", "lq":"Loot Quality: ", "spRegen":"Soul Point Regen: ", "eSteal":"Stealing: ", "gXp":"Gathering XP Bonus: ", "gSpd":"Gathering Speed Bonus: ", "slots":"Powder Slots: ", "set":"Set: ", "quest":"Quest Req: ", "restrict":""}; let idSuffixes = {"displayName": "", "lvl":"", "classReq":"","strReq":"","dexReq":"","intReq":"","defReq":"","agiReq":"", "nDam_":"", "eDam_":"", "tDam_":"", "wDam_":"", "fDam_":"", "aDam_":"", "atkSpd":"", "hp":"", "eDef":"", "tDef":"", "wDef":"", "fDef":"", "aDef":"", "str":"", "dex":"", "int":"", "def":"","agi":"", "hpBonus":"", "hprRaw":"", "hprPct":"%", "sdRaw":"", "sdPct":"%", "mdRaw":"", "mdPct":"%", "mr":"/4s", "ms":"/4s", "ref":"%", "ls":"/4s", "poison":"/3s", "thorns":"%", "expd":"%", "spd":"%", "atkTier":" tier", "eDamPct":"%", "tDamPct":"%", "wDamPct":"%", "fDamPct":"%", "aDamPct":"%", "eDefPct":"%", "tDefPct":"%", "wDefPct":"%", "fDefPct":"%", "aDefPct":"%", "spPct1":"%", "spRaw1":"", "spPct2":"%", "spRaw2":"", "spPct3":"%", "spRaw3":"", "spPct4":"%", "spRaw4":"", "rainbowRaw":"", "sprint":"%", "sprintReg":"%", "jh":"", "xpb":"%", "lb":"%", "lq":"%", "spRegen":"%", "eSteal":"%", "gXp":"%", "gSpd":"%", "slots":"", "set":" set.", "quest":"", "restrict":""}; function apply_elemental_format(p_elem, id, suffix) { suffix = (typeof suffix !== 'undefined') ? suffix : ""; // THIS IS SO JANK BUT IM TOO LAZY TO FIX IT TODO let parts = idPrefixes[id].split(/ (.*)/); let element_prefix = parts[0]; let desc = parts[1]; let i_elem = document.createElement('b'); i_elem.classList.add(element_prefix); i_elem.textContent = element_prefix; p_elem.appendChild(i_elem); let i_elem2 = document.createElement('b'); i_elem2.textContent = " " + desc + suffix; p_elem.appendChild(i_elem2); } function displaySetBonuses(build, parent_id) { setHTML(parent_id, ""); let parent_div = document.getElementById(parent_id); let set_summary_elem = document.createElement('p'); set_summary_elem.classList.add('itemcenter'); set_summary_elem.textContent = "Set Bonuses:"; parent_div.append(set_summary_elem); if (build.activeSetCounts.size) { parent_div.parentElement.style.visibility = "visible"; } else { parent_div.parentElement.style.visibility = "hidden"; } for (const [setName, count] of build.activeSetCounts) { let set_elem = document.createElement('p'); set_elem.id = "set-"+setName; set_summary_elem.append(set_elem); const bonus = sets[setName].bonuses[count-1]; let mock_item = new Map(); mock_item.set("fixID", true); mock_item.set("displayName", setName+" Set: "+count+"/"+sets[setName].items.length); let mock_minRolls = new Map(); mock_item.set("minRolls", mock_minRolls); for (const id in bonus) { if (rolledIDs.includes(id)) { mock_minRolls.set(id, bonus[id]); } else { mock_item.set(id, bonus[id]); } } mock_item.set("powders", []); displayExpandedItem(mock_item, set_elem.id); } } function displayBuildStats(build, parent_id){ // Commands to "script" the creation of nice formatting. // #commands create a new element. // !elemental is some janky hack for elemental damage. // normals just display a thing. let display_commands = [ // "#ldiv", // "!elemental", // "hp", // "fDef", "wDef", "aDef", "tDef", "eDef", // "!elemental", "#table", "mr", "ms", // "hprRaw", "hprPct", "#table", "sdRaw", "sdPct", "mdRaw", "mdPct", "ref", "thorns", "ls", "poison", "expd", "spd", "atkTier", "!elemental", "fDamPct", "wDamPct", "aDamPct", "tDamPct", "eDamPct", // "fDefPct", "wDefPct", "aDefPct", "tDefPct", "eDefPct", "!elemental", "spPct1", "spRaw1", "spPct2", "spRaw2", "spPct3", "spRaw3", "spPct4", "spRaw4", "rainbowRaw", "sprint", "sprintReg", "jh", "xpb", "lb", "lq", "spRegen", "eSteal", "gXp", "gSpd", ]; // Clear the parent div. setHTML(parent_id, ""); let parent_div = document.getElementById(parent_id); let title = document.createElement("p"); title.classList.add("itemcenter"); title.classList.add("itemp"); title.classList.add("title"); title.classList.add("Normal"); title.textContent = "Overall Build Stats"; parent_div.append(title); parent_div.append(document.createElement("br")); if (build.activeSetCounts.size > 0) { let set_summary_elem = document.createElement('p'); set_summary_elem.classList.add('itemp'); set_summary_elem.classList.add('left'); set_summary_elem.textContent = "Set Summary:"; parent_div.append(set_summary_elem); for (const [setName, count] of build.activeSetCounts) { let set_elem = document.createElement('p'); set_elem.classList.add('itemp'); set_elem.classList.add('left'); set_elem.textContent = " "+setName+" Set: "+count+"/"+sets[setName].items.length; set_summary_elem.append(set_elem); } } displayDefenseStats(parent_div, build, true); let stats = build.statMap; //console.log(build.statMap); let active_elem; let elemental_format = false; //TODO this is put here for readability, consolidate with definition in build.js let staticIDs = ["hp", "eDef", "tDef", "wDef", "fDef", "aDef"]; for (const command of display_commands) { if (command.charAt(0) === "#") { if (command === "#cdiv") { active_elem = document.createElement('div'); active_elem.classList.add('itemcenter'); } else if (command === "#ldiv") { active_elem = document.createElement('div'); active_elem.classList.add('itemleft'); } else if (command === "#table") { active_elem = document.createElement('table'); active_elem.classList.add('itemtable'); } parent_div.appendChild(active_elem); } else if (command.charAt(0) === "!") { // TODO: This is sooo incredibly janky..... if (command === "!elemental") { elemental_format = !elemental_format; } } else { let id = command; if (stats.get(id)) { let style = null; if (!staticIDs.includes(id)) { style = "positive"; if (stats.get(id) < 0) { style = "negative"; } } let id_val = stats.get(id); if (reversedIDs.filter(e => e !== "atkTier").includes(id)) { style === "positive" ? style = "negative" : style = "positive"; } if (id === "poison" && id_val > 0) { id_val = Math.round(id_val*(build.statMap.get("poisonPct") + build.externalStats.get("poisonPct"))/100); } displayFixedID(active_elem, id, id_val, elemental_format, style); if (id === "poison" && id_val > 0) { let style = "positive"; let row = document.createElement('tr'); let value_elem = document.createElement('td'); value_elem.classList.add('right'); value_elem.setAttribute("colspan", "2"); let prefix_elem = document.createElement('b'); prefix_elem.textContent = "-> With Strength: "; let number_elem = document.createElement('b'); number_elem.classList.add(style); number_elem.textContent = (id_val * (1+skillPointsToPercentage(build.total_skillpoints[0])) ).toFixed(0) + idSuffixes[id]; value_elem.append(prefix_elem); value_elem.append(number_elem); row.appendChild(value_elem); active_elem.appendChild(row); } } } } } function displayExpandedItem(item, parent_id){ // Commands to "script" the creation of nice formatting. // #commands create a new element. // !elemental is some janky hack for elemental damage. // normals just display a thing. if (item.get("category") === "weapon") { let stats = new Map(); stats.set("atkSpd", item.get("atkSpd")); stats.set("damageBonus", [0, 0, 0, 0, 0]); stats.set("damageRaw", [item.get("nDam"), item.get("eDam"), item.get("tDam"), item.get("wDam"), item.get("fDam"), item.get("aDam")]); let results = calculateSpellDamage(stats, [100, 0, 0, 0, 0, 0], 0, 0, 0, item, [0, 0, 0, 0, 0], 1, undefined); let damages = results[2]; let damage_keys = [ "nDam_", "eDam_", "tDam_", "wDam_", "fDam_", "aDam_" ]; for (const i in damage_keys) { item.set(damage_keys[i], damages[i][0]+"-"+damages[i][1]); } } let display_commands = [ "#cdiv", "displayName", //"type", //REPLACE THIS WITH SKIN "#ldiv", "atkSpd", "#ldiv", "!elemental", "hp", "nDam_", "fDam_", "wDam_", "aDam_", "tDam_", "eDam_", "fDef", "wDef", "aDef", "tDef", "eDef", "!elemental", "#ldiv", "classReq", "lvl", "strReq", "dexReq", "intReq", "defReq","agiReq", "#ldiv", "str", "dex", "int", "def", "agi", "#table", "hpBonus", "hprRaw", "hprPct", "sdRaw", "sdPct", "mdRaw", "mdPct", "mr", "ms", "ref", "thorns", "ls", "poison", "expd", "spd", "atkTier", "!elemental", "fDamPct", "wDamPct", "aDamPct", "tDamPct", "eDamPct", "fDefPct", "wDefPct", "aDefPct", "tDefPct", "eDefPct", "!elemental", "spPct1", "spRaw1", "spPct2", "spRaw2", "spPct3", "spRaw3", "spPct4", "spRaw4", "rainbowRaw", "sprint", "sprintReg", "jh", "xpb", "lb", "lq", "spRegen", "eSteal", "gXp", "gSpd", "#ldiv", "!elemental", "slots", "!elemental", "set", "quest", "restrict"]; // Clear the parent div. setHTML(parent_id, ""); let parent_div = document.getElementById(parent_id); let active_elem; let fix_id = item.has("fixID") && item.get("fixID"); let elemental_format = false; for (const command of display_commands) { if (command.charAt(0) === "#") { if (command === "#cdiv") { active_elem = document.createElement('div'); active_elem.classList.add('itemcenter'); } else if (command === "#ldiv") { active_elem = document.createElement('div'); active_elem.classList.add('itemleft'); } else if (command === "#table") { active_elem = document.createElement('table'); active_elem.classList.add('itemtable'); } parent_div.appendChild(active_elem); } else if (command.charAt(0) === "!") { // TODO: This is sooo incredibly janky..... if (command === "!elemental") { elemental_format = !elemental_format; } } else { let id = command; if(nonRolledIDs.includes(id) && item.get(id)){//nonRolledID & non-0/non-null/non-und ID if (id === "slots") { let p_elem = document.createElement("p"); // PROPER POWDER DISPLAYING EZ CLAP let numerals = new Map([[1, "I"], [2, "II"], [3, "III"], [4, "IV"], [5, "V"], [6, "VI"]]); /*p_elem.textContent = idPrefixes[id].concat(item.get(id), idSuffixes[id]) + " [ " + item.get("powders").map(x => powderNames.get(x)) + " ]";*/ let powderPrefix = document.createElement("b"); powderPrefix.classList.add("itemp"); powderPrefix.classList.add("left"); powderPrefix.textContent = "Powder Slots: " + item.get(id) + " ["; p_elem.appendChild(powderPrefix); let powders = item.get("powders"); //console.log(powders); for (let i = 0; i < powders.length; i++) { let powder = document.createElement("b"); powder.textContent = numerals.get((powders[i]%6)+1)+" "; powder.classList.add(damageClasses[Math.floor(powders[i]/6)+1]+"_powder"); p_elem.appendChild(powder); } let powderSuffix = document.createElement("b"); powderSuffix.classList.add("itemp"); powderSuffix.classList.add("left"); powderSuffix.textContent = "]"; p_elem.appendChild(powderSuffix); active_elem.appendChild(p_elem); } else { let p_elem = displayFixedID(active_elem, id, item.get(id), elemental_format); if (id === "displayName") { p_elem.classList.add("title"); if (item.get("tier") !== " ") { p_elem.classList.add(item.get("tier")); } /*let validTypes = ["helmet", "chestplate", "leggings", "boots", "relik", "wand", "bow", "spear", "dagger", "ring", "bracelet", "necklace"]; if (item.has("type") && validTypes.includes(item.get("type"))) { p = document.createElement("p"); img = document.createElement("img"); img.src = "./media/items/generic-"+item.get("type")+".png"; img.alt = "image no display :("; img.classList.add("center"); p.append(img); p.classList.add("itemp"); p_elem.append(p); }*/ } else if (skp_order.includes(id)) { //id = str, dex, int, def, or agi p_elem.textContent = ""; p_elem.classList.add("itemtable"); let row = document.createElement("tr"); let title = document.createElement("td"); title.textContent = idPrefixes[id] + " "; let boost = document.createElement("td"); if (item.get(id) < 0) { boost.classList.add("negative"); } else { //boost = 0 SHOULD not come up boost.classList.add("positive"); } boost.classList.add("spaceLeft"); boost.textContent = item.get(id); row.appendChild(title); row.appendChild(boost); p_elem.appendChild(row); } else if (id === "restrict") { p_elem.classList.add("restrict"); } } } else if (rolledIDs.includes(id) && item.get("minRolls").get(id)){ // && item.get("maxRolls").get(id) ){//rolled ID & non-0/non-null/non-und ID let style = "positive"; if (item.get("minRolls").get(id) < 0) { style = "negative"; } //let flipPosNeg = ["spRaw1","spRaw2","spRaw3","spRaw4","spPct1","spPct2","spPct3","spPct4"]; if(reversedIDs.filter(e => e !== "atkTier").includes(id)){ style === "positive" ? style = "negative" : style = "positive"; } if (fix_id) { displayFixedID(active_elem, id, item.get("minRolls").get(id), elemental_format, style); } else { let row = document.createElement('tr'); let min_elem = document.createElement('td'); min_elem.classList.add('left'); min_elem.classList.add(style); min_elem.textContent = item.get("minRolls").get(id) + idSuffixes[id]; row.appendChild(min_elem); let desc_elem = document.createElement('td'); desc_elem.classList.add('center'); //TODO elemental format jank if (elemental_format) { apply_elemental_format(desc_elem, id); } else { desc_elem.textContent = idPrefixes[id]; } row.appendChild(desc_elem); let max_elem = document.createElement('td'); max_elem.classList.add('right'); max_elem.classList.add(style); max_elem.textContent = item.get("maxRolls").get(id) + idSuffixes[id]; row.appendChild(max_elem); active_elem.appendChild(row); } }//Just don't do anything if else } } //Show powder specials ;-; let powder_special = document.createElement("p"); powder_special.classList.add("left"); let powders = item.get("powders"); let element = ""; let power = 0; for (let i = 0; i < powders.length; i++) { let firstPowderType = skp_elements[Math.floor(powders[i]/6)]; if (element !== "") break; else if (powders[i]%6 > 2) { //t4+ for (let j = i+1; j < powders.length; j++) { let currentPowderType = skp_elements[Math.floor(powders[j]/6)] if (powders[j] % 6 > 2 && firstPowderType === currentPowderType) { element = currentPowderType; power = Math.round(((powders[i] % 6 + powders[j] % 6 + 2) / 2 - 4) * 2); break; } } } } if (element !== "") {//powder special is "[e,t,w,f,a]+[0,1,2,3,4]" let powderSpecial = powderSpecialStats[ skp_elements.indexOf(element)]; let specialSuffixes = new Map([ ["Duration", " sec"], ["Radius", " blocks"], ["Chains", ""], ["Damage", "%"], ["Damage Boost", "%"], ["Knockback", " blocks"] ]); let specialTitle = document.createElement("p"); let specialEffects = document.createElement("p"); specialTitle.classList.add("left"); specialTitle.classList.add("itemp"); specialTitle.classList.add(damageClasses[skp_elements.indexOf(element) + 1]); specialEffects.classList.add("left"); specialEffects.classList.add("itemp"); specialEffects.classList.add("nocolor"); let effects; if (item.get("category") === "weapon") {//weapon effects = powderSpecial["weaponSpecialEffects"]; specialTitle.textContent = powderSpecial["weaponSpecialName"]; }else if (item.get("category") === "armor") {//armor effects = powderSpecial["armorSpecialEffects"]; specialTitle.textContent += powderSpecial["armorSpecialName"] + ": "; } for (const [key,value] of effects) { if (key !== "Description") { let effect = document.createElement("p"); effect.classList.add("itemp"); effect.textContent += key + ": " + value[power] + specialSuffixes.get(key); if(key === "Damage"){ effect.textContent += elementIcons[skp_elements.indexOf(element)]; } if (element === "w") { effect.textContent += " / Mana Used"; } specialEffects.appendChild(effect); }else{ specialTitle.textContent += "[ " + effects.get("Description") + " ]"; } } specialTitle.append(specialEffects); powder_special.appendChild(specialTitle); parent_div.append(powder_special); } //Show item tier if (item.get("tier") && item.get("tier") !== " ") { let item_desc_elem = document.createElement('p'); item_desc_elem.classList.add('left'); item_desc_elem.classList.add(item.get("tier")); item_desc_elem.textContent = item.get("tier")+" "+item.get("type"); parent_div.append(item_desc_elem); } } function displayFixedID(active, id, value, elemental_format, style) { if (style) { /*if(reversedIDs.filter(e => e !== "atkTier").includes(id)){ style === "positive" ? style = "negative" : style = "positive"; }*/ let row = document.createElement('tr'); let desc_elem = document.createElement('td'); desc_elem.classList.add('left'); if (elemental_format) { apply_elemental_format(desc_elem, id); } else { desc_elem.textContent = idPrefixes[id]; } row.appendChild(desc_elem); let value_elem = document.createElement('td'); value_elem.classList.add('right'); value_elem.classList.add(style); value_elem.textContent = value + idSuffixes[id]; row.appendChild(value_elem); active.appendChild(row); return row; } else { // HACK TO AVOID DISPLAYING ZERO DAMAGE! TODO if (value === "0-0") { return; } let p_elem = document.createElement('p'); p_elem.classList.add('itemp'); if (elemental_format) { apply_elemental_format(p_elem, id, value); } else { p_elem.textContent = idPrefixes[id].concat(value, idSuffixes[id]); } active.appendChild(p_elem); return p_elem; } } function displayEquipOrder(parent_elem,buildOrder){ parent_elem.textContent = ""; const order = buildOrder.slice(); let title_elem = document.createElement("p"); title_elem.textContent = "Equip order "; title_elem.classList.add("title"); title_elem.classList.add("Normal"); title_elem.classList.add("itemp"); parent_elem.append(title_elem); parent_elem.append(document.createElement("br")); for (const item of order) { let p_elem = document.createElement("p"); p_elem.classList.add("itemp"); p_elem.classList.add("left"); p_elem.textContent = item.get("displayName"); parent_elem.append(p_elem); } } function displayPoisonDamage(overallparent_elem, build) { overallparent_elem.textContent = ""; //Title let title_elemavg = document.createElement("p"); title_elemavg.classList.add("smalltitle"); title_elemavg.classList.add("Normal"); title_elemavg.textContent = "Poison Stats"; overallparent_elem.append(title_elemavg); let overallpoisonDamage = document.createElement("p"); overallpoisonDamage.classList.add("itemp"); let overallpoisonDamageFirst = document.createElement("b"); let overallpoisonDamageSecond = document.createElement("b"); let poison_tick = Math.round(build.statMap.get("poison") * (1+skillPointsToPercentage(build.total_skillpoints[0])) * (build.statMap.get("poisonPct") + build.externalStats.get("poisonPct"))/100 /3); overallpoisonDamageFirst.textContent = "Poison Tick: "; overallpoisonDamageSecond.textContent = Math.max(poison_tick,0); overallpoisonDamageSecond.classList.add("Damage"); overallpoisonDamage.appendChild(overallpoisonDamageFirst); overallpoisonDamage.appendChild(overallpoisonDamageSecond); overallparent_elem.append(overallpoisonDamage); overallparent_elem.append(document.createElement("br")); } function displayMeleeDamage(parent_elem, overallparent_elem, meleeStats){ let attackSpeeds = ["Super Slow", "Very Slow", "Slow", "Normal", "Fast", "Very Fast", "Super Fast"]; //let damagePrefixes = ["Neutral Damage: ","Earth Damage: ","Thunder Damage: ","Water Damage: ","Fire Damage: ","Air Damage: "]; parent_elem.textContent = ""; overallparent_elem.textContent = ""; const stats = meleeStats.slice(); for (let i = 0; i < 6; ++i) { for (let j in stats[i]) { stats[i][j] = stats[i][j].toFixed(2); } } for (let i = 6; i < 8; ++i) { for (let j = 0; j < 2; j++) { stats[i][j] = stats[i][j].toFixed(2); } } for (let i = 8; i < 11; ++i){ stats[i] = stats[i].toFixed(2); } //title let title_elem = document.createElement("p"); title_elem.classList.add("title"); title_elem.classList.add("Normal"); title_elem.classList.add("itemp"); title_elem.textContent = "Melee Stats"; parent_elem.append(title_elem); parent_elem.append(document.createElement("br")); //overall title let title_elemavg = document.createElement("p"); title_elemavg.classList.add("smalltitle"); title_elemavg.classList.add("Normal"); title_elemavg.textContent = "Melee Stats"; overallparent_elem.append(title_elemavg); //average DPS let averageDamage = document.createElement("p"); averageDamage.classList.add("left"); averageDamage.classList.add("itemp"); averageDamage.textContent = "Average DPS: " + stats[10]; parent_elem.append(averageDamage); //overall average DPS let overallaverageDamage = document.createElement("p"); overallaverageDamage.classList.add("itemp"); let overallaverageDamageFirst = document.createElement("b"); overallaverageDamageFirst.textContent = "Average DPS: " let overallaverageDamageSecond = document.createElement("b"); overallaverageDamageSecond.classList.add("Damage"); overallaverageDamageSecond.textContent = stats[10]; overallaverageDamage.appendChild(overallaverageDamageFirst); overallaverageDamage.appendChild(overallaverageDamageSecond); overallparent_elem.append(overallaverageDamage); overallparent_elem.append(document.createElement("br")); //attack speed let atkSpd = document.createElement("p"); atkSpd.classList.add("left"); atkSpd.classList.add("itemp"); atkSpd.textContent = "Attack Speed: " + attackSpeeds[stats[11]]; parent_elem.append(atkSpd); parent_elem.append(document.createElement("br")); //overall attack speed let overallatkSpd = document.createElement("p"); overallatkSpd.classList.add("center"); overallatkSpd.classList.add("itemp"); let overallatkSpdFirst = document.createElement("b"); overallatkSpdFirst.textContent = "Attack Speed: "; let overallatkSpdSecond = document.createElement("b"); overallatkSpdSecond.classList.add("Damage"); overallatkSpdSecond.textContent = attackSpeeds[stats[11]]; overallatkSpd.appendChild(overallatkSpdFirst); overallatkSpd.appendChild(overallatkSpdSecond); overallparent_elem.append(overallatkSpd); overallparent_elem.append(document.createElement("br")); //Non-Crit: n->elem, total dmg, DPS let nonCritStats = document.createElement("p"); nonCritStats.classList.add("left"); nonCritStats.classList.add("itemp"); nonCritStats.textContent = "Non-Crit Stats: "; nonCritStats.append(document.createElement("br")); for (let i = 0; i < 6; i++){ if(stats[i][0] > 0){ let dmg = document.createElement("p"); dmg.textContent = stats[i][0] + "-" + stats[i][1]; dmg.classList.add(damageClasses[i]); dmg.classList.add("itemp"); nonCritStats.append(dmg); } } let normalDamage = document.createElement("p"); normalDamage.textContent = "Total: " + stats[6][0] + "-" + stats[6][1]; normalDamage.classList.add("itemp"); nonCritStats.append(normalDamage); let normalDPS = document.createElement("p"); normalDPS.textContent = "Normal DPS: " + stats[8]; normalDPS.classList.add("itemp"); nonCritStats.append(normalDPS); //overall average DPS let singleHitDamage = document.createElement("p"); singleHitDamage.classList.add("itemp"); let singleHitDamageFirst = document.createElement("b"); singleHitDamageFirst.textContent = "Single Hit Average: "; let singleHitDamageSecond = document.createElement("b"); singleHitDamageSecond.classList.add("Damage"); singleHitDamageSecond.textContent = stats[12].toFixed(2); singleHitDamage.appendChild(singleHitDamageFirst); singleHitDamage.appendChild(singleHitDamageSecond); overallparent_elem.append(singleHitDamage); overallparent_elem.append(document.createElement("br")); let normalChance = document.createElement("p"); normalChance.textContent = "Non-Crit Chance: " + (stats[6][2]*100).toFixed(2) + "%"; normalChance.classList.add("itemp"); normalChance.append(document.createElement("br")); normalChance.append(document.createElement("br")); nonCritStats.append(normalChance); parent_elem.append(nonCritStats); parent_elem.append(document.createElement("br")); //Crit: n->elem, total dmg, DPS let critStats = document.createElement("p"); critStats.classList.add("left"); critStats.classList.add("itemp"); critStats.textContent = "Crit Stats: "; critStats.append(document.createElement("br")); for (let i = 0; i < 6; i++){ if(stats[i][2] > 0){ dmg = document.createElement("p"); dmg.textContent = stats[i][2] + "-" + stats[i][3]; dmg.classList.add(damageClasses[i]); dmg.classList.add("itemp"); critStats.append(dmg); } } let critDamage = document.createElement("p"); critDamage.textContent = "Total: " + stats[7][0] + "-" + stats[7][1]; critDamage.classList.add("itemp"); critStats.append(critDamage); let critDPS = document.createElement("p"); critDPS.textContent = "Crit DPS: " + stats[9]; critDPS.classList.add("itemp"); critStats.append(critDPS); let critChance = document.createElement("p"); critChance.textContent = "Crit Chance: " + (stats[7][2]*100).toFixed(2) + "%"; critChance.classList.add("itemp"); critChance.append(document.createElement("br")); critChance.append(document.createElement("br")); critStats.append(critChance); parent_elem.append(critStats); } function displayDefenseStats(parent_elem, build, insertSummary){ let defenseStats = build.getDefenseStats(); insertSummary = (typeof insertSummary !== 'undefined') ? insertSummary : false; if (!insertSummary) { parent_elem.textContent = ""; } const stats = defenseStats.slice(); if (!insertSummary) { let title_elem = document.createElement("p"); title_elem.textContent = "Defense Stats"; title_elem.classList.add("title"); title_elem.classList.add("Normal"); title_elem.classList.add("itemp"); parent_elem.append(title_elem); let base_stat_elem = document.createElement("p"); base_stat_elem.id = "base-stat"; parent_elem.append(base_stat_elem); let mock_item = new Map(); mock_item.set("fixID", true); let mock_minRolls = new Map(); mock_item.set("minRolls", mock_minRolls); const stats = ["hp", "hpBonus", "hprRaw", "hprPct", "fDef", "wDef", "aDef", "tDef", "eDef", "fDefPct", "wDefPct", "aDefPct", "tDefPct", "eDefPct"]; for (const stat of stats) { if (rolledIDs.includes(stat)) { mock_minRolls.set(stat, build.statMap.get(stat)); } else { mock_item.set(stat, build.statMap.get(stat)); } } mock_item.set("powders", []); displayExpandedItem(mock_item, base_stat_elem.id); } parent_elem.append(document.createElement("br")); let statsTable = document.createElement("table"); statsTable.classList.add("itemtable"); //[total hp, ehp, total hpr, ehpr, [def%, agi%], [edef,tdef,wdef,fdef,adef]] for(const i in stats){ if(typeof stats[i] === "number"){ stats[i] = stats[i].toFixed(2); }else{ for(const j in stats[i]){ stats[i][j] = stats[i][j].toFixed(2); } } } //total HP let hpRow = document.createElement("tr"); let hp = document.createElement("td"); hp.classList.add("Health"); hp.classList.add("left"); hp.textContent = "Total HP:"; let boost = document.createElement("td"); boost.textContent = stats[0]; boost.classList.add("right"); hpRow.appendChild(hp); hpRow.append(boost); statsTable.appendChild(hpRow); //EHP let ehpRow = document.createElement("tr"); let ehp = document.createElement("td"); ehp.classList.add("left"); ehp.textContent = "Effective HP:"; boost = document.createElement("td"); boost.textContent = stats[1][0]; boost.classList.add("right"); ehpRow.appendChild(ehp); ehpRow.append(boost); statsTable.append(ehpRow); ehpRow = document.createElement("tr"); ehp = document.createElement("td"); ehp.classList.add("left"); ehp.textContent = "Effective HP (no agi):"; boost = document.createElement("td"); boost.textContent = stats[1][1]; boost.classList.add("right"); ehpRow.appendChild(ehp); ehpRow.append(boost); statsTable.append(ehpRow); //total HPR let hprRow = document.createElement("tr"); let hpr = document.createElement("td"); hpr.classList.add("Health"); hpr.classList.add("left"); hpr.textContent = "HP Regen (Total):"; boost = document.createElement("td"); boost.textContent = stats[2]; boost.classList.add("right"); hprRow.appendChild(hpr); hprRow.appendChild(boost); statsTable.appendChild(hprRow); /*//EHPR let ehprRow = document.createElement("tr"); let ehpr = document.createElement("td"); ehpr.classList.add("left"); ehpr.textContent = "Effective HP Regen:"; boost = document.createElement("td"); boost.textContent = stats[3][0]; boost.classList.add("right"); ehprRow.appendChild(ehpr); ehprRow.append(boost); statsTable.append(ehprRow); ehprRow = document.createElement("tr"); ehpr = document.createElement("td"); ehpr.classList.add("left"); ehpr.textContent = "Effective HP Regen (no agi):"; boost = document.createElement("td"); boost.textContent = stats[3][1]; boost.classList.add("right"); ehprRow.appendChild(ehpr); ehprRow.append(boost); statsTable.append(ehprRow);*/ //eledefs let eledefs = stats[5]; for (let i = 0; i < eledefs.length; i++){ let eledefElemRow = document.createElement("tr"); let eledef = document.createElement("td"); eledef.classList.add("left") let eledefTitle = document.createElement("b"); eledefTitle.textContent = damageClasses[i+1]; eledefTitle.classList.add(damageClasses[i+1]); let defense = document.createElement("b"); defense.textContent = " Def (Total): "; eledef.appendChild(eledefTitle); eledef.appendChild(defense); eledefElemRow.appendChild(eledef); let boost = document.createElement("td"); boost.textContent = eledefs[i]; boost.classList.add("right"); eledefElemRow.appendChild(boost); statsTable.appendChild(eledefElemRow); } if (!insertSummary) { //skp let defRow = document.createElement("tr"); let defElem = document.createElement("td"); defElem.classList.add("left"); defElem.textContent = "Damage Absorbed %:"; boost = document.createElement("td"); boost.classList.add("right"); boost.textContent = stats[4][0] + "%"; defRow.appendChild(defElem); defRow.appendChild(boost); statsTable.append(defRow); let agiRow = document.createElement("tr"); let agiElem = document.createElement("td"); agiElem.classList.add("left"); agiElem.textContent = "Dodge Chance %:"; boost = document.createElement("td"); boost.classList.add("right"); boost.textContent = stats[4][1] + "%"; agiRow.appendChild(agiElem); agiRow.appendChild(boost); statsTable.append(agiRow); } parent_elem.append(statsTable); } function displayPowderSpecials(parent_elem, powderSpecials, build) { parent_elem.textContent = "Powder Specials"; let specials = powderSpecials.slice(); let stats = build.statMap; let expandedStats = new Map(); //each entry of powderSpecials is [ps, power] for (special of specials) { //iterate through the special and display its effects. let powder_special = document.createElement("p"); powder_special.classList.add("left"); let specialSuffixes = new Map([ ["Duration", " sec"], ["Radius", " blocks"], ["Chains", ""], ["Damage", "%"], ["Damage Boost", "%"], ["Knockback", " blocks"] ]); let specialTitle = document.createElement("p"); let specialEffects = document.createElement("p"); specialTitle.classList.add("left"); specialTitle.classList.add("itemp"); specialTitle.classList.add(damageClasses[powderSpecialStats.indexOf(special[0]) + 1]); specialEffects.classList.add("left"); specialEffects.classList.add("itemp"); specialEffects.classList.add("nocolor"); let effects = special[0]["weaponSpecialEffects"]; let power = special[1]; specialTitle.textContent = special[0]["weaponSpecialName"] + " " + Math.floor((power-1)*0.5 + 4) + (power % 2 == 0 ? ".5" : ""); for (const [key,value] of effects) { let effect = document.createElement("p"); effect.classList.add("itemp"); effect.textContent += key + ": " + value[power-1] + specialSuffixes.get(key); if(key === "Damage"){ effect.textContent += elementIcons[powderSpecialStats.indexOf(special[0])]; } if(special[0]["weaponSpecialName"] === "Air Prison" && key === "Damage Boost") { effect.textContent += " (only 1st hit)"; } specialEffects.appendChild(effect); } powder_special.appendChild(specialTitle); powder_special.appendChild(specialEffects); //if this special is an instant-damage special (Quake, Chain Lightning, Courage Burst), display the damage. let specialDamage = document.createElement("p"); let spells = spell_table["powder"]; if (powderSpecialStats.indexOf(special[0]) == 0 || powderSpecialStats.indexOf(special[0]) == 1 || powderSpecialStats.indexOf(special[0]) == 3) { //Quake, Chain Lightning, or Courage let spell = (powderSpecialStats.indexOf(special[0]) == 3 ? spells[2] : spells[powderSpecialStats.indexOf(special[0])]); let part = spell["parts"][0]; let _results = calculateSpellDamage(stats, part.conversion, stats.get("mdRaw"), stats.get("mdPct") + build.externalStats.get("mdPct"), 0, build.weapon, build.total_skillpoints, build.damageMultiplier * ((part.multiplier[power-1] / 100)), build.externalStats);//part.multiplier[power] / 100 let critChance = skillPointsToPercentage(build.total_skillpoints[1]); let save_damages = []; let totalDamNormal = _results[0]; let totalDamCrit = _results[1]; let results = _results[2]; for (let i = 0; i < 6; ++i) { for (let j in results[i]) { results[i][j] = results[i][j].toFixed(2); } } let nonCritAverage = (totalDamNormal[0]+totalDamNormal[1])/2 || 0; let critAverage = (totalDamCrit[0]+totalDamCrit[1])/2 || 0; let averageDamage = (1-critChance)*nonCritAverage+critChance*critAverage || 0; let averageLabel = document.createElement("p"); averageLabel.textContent = "Average: "+averageDamage.toFixed(2); averageLabel.classList.add("damageSubtitle"); specialDamage.append(averageLabel); let nonCritLabel = document.createElement("p"); nonCritLabel.textContent = "Non-Crit Average: "+nonCritAverage.toFixed(2); nonCritLabel.classList.add("damageSubtitle"); specialDamage.append(nonCritLabel); for (let i = 0; i < 6; i++){ if (results[i][1] > 0){ let p = document.createElement("p"); p.classList.add("damagep"); p.classList.add(damageClasses[i]); p.textContent = results[i][0]+"-"+results[i][1]; specialDamage.append(p); } } let normalDamage = document.createElement("p"); normalDamage.textContent = "Total: " + totalDamNormal[0].toFixed(2) + "-" + totalDamNormal[1].toFixed(2); normalDamage.classList.add("itemp"); specialDamage.append(normalDamage); let nonCritChanceLabel = document.createElement("p"); nonCritChanceLabel.textContent = "Non-Crit Chance: " + ((1-critChance)*100).toFixed(2) + "%"; specialDamage.append(nonCritChanceLabel); let critLabel = document.createElement("p"); critLabel.textContent = "Crit Average: "+critAverage.toFixed(2); critLabel.classList.add("damageSubtitle"); specialDamage.append(critLabel); for (let i = 0; i < 6; i++){ if (results[i][1] > 0){ let p = document.createElement("p"); p.classList.add("damagep"); p.classList.add(damageClasses[i]); p.textContent = results[i][2]+"-"+results[i][3]; specialDamage.append(p); } } let critDamage = document.createElement("p"); critDamage.textContent = "Total: " + totalDamCrit[0].toFixed(2) + "-" + totalDamCrit[1].toFixed(2); critDamage.classList.add("itemp"); specialDamage.append(critDamage); let critChanceLabel = document.createElement("p"); critChanceLabel.textContent = "Crit Chance: " + (critChance*100).toFixed(2) + "%"; specialDamage.append(critChanceLabel); save_damages.push(averageDamage); powder_special.append(specialDamage); } parent_elem.appendChild(powder_special); } } function displaySpellDamage(parent_elem, overallparent_elem, build, spell, spellIdx) { parent_elem.textContent = ""; const stats = build.statMap; let title_elem = document.createElement("p"); title_elem.classList.add("smalltitle"); title_elem.classList.add("Normal"); overallparent_elem.textContent = ""; let title_elemavg = document.createElement("p"); title_elemavg.classList.add('smalltitle'); title_elemavg.classList.add('Normal'); if (spellIdx != 0) { title_elem.textContent = spell.title + " (" + build.getSpellCost(spellIdx, spell.cost) + ")"; let first = document.createElement("b"); first.classList.add("space"); first.textContent = spell.title + " ("; title_elemavg.appendChild(first); let second = document.createElement("b"); second.textContent = build.getSpellCost(spellIdx, spell.cost); second.classList.add("Legendary"); title_elemavg.appendChild(second); let third = document.createElement("b"); third.classList.add("Water"); title_elemavg.appendChild(third); let fourth = document.createElement("b"); fourth.textContent = ")"; title_elemavg.appendChild(fourth); //title_elemavg.textContent = spell.title + " (" + build.getSpellCost(spellIdx, spell.cost) + ")"; } else { title_elem.textContent = spell.title; title_elemavg.textContent = spell.title; } parent_elem.append(title_elem); overallparent_elem.append(title_elemavg); let critChance = skillPointsToPercentage(build.total_skillpoints[1]); let save_damages = []; for (const part of spell.parts) { parent_elem.append(document.createElement("br")); let part_div = document.createElement("p"); parent_elem.append(part_div); let part_divavg = document.createElement("p"); //part_divavg.classList.add("Normal"); overallparent_elem.append(part_divavg); let subtitle_elem = document.createElement("p"); subtitle_elem.textContent = part.subtitle; part_div.append(subtitle_elem); if (part.summary == true) { let subtitle_elemavg = document.createElement("p"); subtitle_elemavg.textContent = part.subtitle; part_divavg.append(subtitle_elemavg); } if (part.type === "damage") { //console.log(build.expandedStats); let _results = calculateSpellDamage(stats, part.conversion, stats.get("sdRaw"), stats.get("sdPct") + build.externalStats.get("sdPct"), part.multiplier / 100, build.weapon, build.total_skillpoints, build.damageMultiplier, build.externalStats); let totalDamNormal = _results[0]; let totalDamCrit = _results[1]; let results = _results[2]; for (let i = 0; i < 6; ++i) { for (let j in results[i]) { results[i][j] = results[i][j].toFixed(2); } } let nonCritAverage = (totalDamNormal[0]+totalDamNormal[1])/2 || 0; let critAverage = (totalDamCrit[0]+totalDamCrit[1])/2 || 0; let averageDamage = (1-critChance)*nonCritAverage+critChance*critAverage || 0; let averageLabel = document.createElement("p"); averageLabel.textContent = "Average: "+averageDamage.toFixed(2); averageLabel.classList.add("damageSubtitle"); part_div.append(averageLabel); if (part.summary == true) { let overallaverageLabel = document.createElement("p"); let first = document.createElement("b"); let second = document.createElement("b"); first.textContent = "Average: "; second.textContent = averageDamage.toFixed(2); overallaverageLabel.appendChild(first); overallaverageLabel.appendChild(second); second.classList.add("Damage"); overallaverageLabel.classList.add("damageSubtitle"); part_divavg.append(overallaverageLabel); } let nonCritLabel = document.createElement("p"); nonCritLabel.textContent = "Non-Crit Average: "+nonCritAverage.toFixed(2); nonCritLabel.classList.add("damageSubtitle"); part_div.append(nonCritLabel); for (let i = 0; i < 6; i++){ if (results[i][1] > 0){ let p = document.createElement("p"); p.classList.add("damagep"); p.classList.add(damageClasses[i]); p.textContent = results[i][0]+"-"+results[i][1]; part_div.append(p); } } //part_div.append(document.createElement("br")); let critLabel = document.createElement("p"); critLabel.textContent = "Crit Average: "+critAverage.toFixed(2); critLabel.classList.add("damageSubtitle"); part_div.append(critLabel); for (let i = 0; i < 6; i++){ if (results[i][1] > 0){ let p = document.createElement("p"); p.classList.add("damagep"); p.classList.add(damageClasses[i]); p.textContent = results[i][2]+"-"+results[i][3]; part_div.append(p); } } save_damages.push(averageDamage); } else if (part.type === "heal") { let heal_amount = (part.strength * build.getDefenseStats()[0] * Math.max(0.5,Math.min(1.75, 1 + 0.5 * stats.get("wDamPct")/100))).toFixed(2); let healLabel = document.createElement("p"); healLabel.textContent = heal_amount; healLabel.classList.add("damagep"); part_div.append(healLabel); if (part.summary == true) { let overallhealLabel = document.createElement("p"); overallhealLabel.textContent = heal_amount; overallhealLabel.classList.add("damagep"); overallhealLabel.classList.add("Set"); part_divavg.append(overallhealLabel); } } else if (part.type === "total") { let total_damage = 0; for (let i in part.factors) { total_damage += save_damages[i] * part.factors[i]; } let averageLabel = document.createElement("p"); averageLabel.textContent = "Average: "+total_damage.toFixed(2); averageLabel.classList.add("damageSubtitle"); part_div.append(averageLabel); let overallaverageLabel = document.createElement("p"); overallaverageLabel.textContent = "Average: "+total_damage.toFixed(2); overallaverageLabel.classList.add("damageSubtitle"); part_divavg.append(overallaverageLabel); } } }