2021-01-10 02:02:23 -08:00
|
|
|
|
2021-03-30 23:44:56 -07:00
|
|
|
|
2021-08-07 16:15:44 -07:00
|
|
|
const classDefenseMultipliers = new Map([ ["relik",0.50], ["bow",0.60], ["wand", 0.80], ["dagger", 1.0], ["spear",1.20], ["sword", 1.10]]);
|
2021-01-10 02:02:23 -08:00
|
|
|
|
2021-01-18 07:31:20 +10:30
|
|
|
/**
|
|
|
|
* @description Error to catch items that don't exist.
|
|
|
|
* @module ItemNotFound
|
|
|
|
*/
|
|
|
|
class ItemNotFound {
|
|
|
|
/**
|
|
|
|
* @class
|
|
|
|
* @param {String} item the item name entered
|
|
|
|
* @param {String} type the type of item
|
|
|
|
* @param {Boolean} genElement whether to generate an element from inputs
|
|
|
|
* @param {String} override override for item type
|
|
|
|
*/
|
|
|
|
constructor(item, type, genElement, override) {
|
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
* @type {String}
|
|
|
|
*/
|
|
|
|
this.message = `Cannot find ${override||type} named ${item}`;
|
|
|
|
if (genElement)
|
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
* @type {Element}
|
|
|
|
*/
|
|
|
|
this.element = document.getElementById(`${type}-choice`).parentElement.querySelectorAll("p.error")[0];
|
|
|
|
else
|
|
|
|
this.element = document.createElement("div");
|
|
|
|
}
|
2021-01-06 18:08:19 -06:00
|
|
|
}
|
|
|
|
|
2021-01-18 07:31:20 +10:30
|
|
|
/**
|
|
|
|
* @description Error to catch incorrect input.
|
|
|
|
* @module IncorrectInput
|
|
|
|
*/
|
|
|
|
class IncorrectInput {
|
|
|
|
/**
|
|
|
|
* @class
|
|
|
|
* @param {String} input the inputted text
|
|
|
|
* @param {String} format the correct format
|
|
|
|
* @param {String} sibling the id of the error node's sibling
|
|
|
|
*/
|
|
|
|
constructor(input, format, sibling) {
|
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
* @type {String}
|
|
|
|
*/
|
|
|
|
this.message = `${input} is incorrect. Example: ${format}`;
|
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
* @type {String}
|
|
|
|
*/
|
|
|
|
this.id = sibling;
|
2021-01-06 18:08:19 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-18 07:31:20 +10:30
|
|
|
/**
|
|
|
|
* @description Error that inputs an array of items to generate errors of.
|
|
|
|
* @module ListError
|
|
|
|
* @extends Error
|
|
|
|
*/
|
|
|
|
class ListError extends Error {
|
|
|
|
/**
|
|
|
|
* @class
|
|
|
|
* @param {Array} errors array of errors
|
|
|
|
*/
|
|
|
|
constructor(errors) {
|
|
|
|
let ret = [];
|
|
|
|
if (typeof errors[0] == "string") {
|
|
|
|
super(errors[0]);
|
|
|
|
} else {
|
|
|
|
super(errors[0].message);
|
|
|
|
}
|
|
|
|
for (let i of errors) {
|
|
|
|
if (typeof i == "string") {
|
|
|
|
ret.push(new Error(i));
|
|
|
|
} else {
|
|
|
|
ret.push(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
* @type {Object[]}
|
|
|
|
*/
|
|
|
|
this.errors = ret;
|
2021-01-06 18:08:19 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-06 18:02:10 -06:00
|
|
|
/*Class that represents a wynn player's build.
|
|
|
|
*/
|
|
|
|
class Build{
|
|
|
|
|
2021-01-18 07:31:20 +10:30
|
|
|
/**
|
|
|
|
* @description Construct a build.
|
|
|
|
* @param {Number} level : Level of the player.
|
|
|
|
* @param {String[]} equipment : List of equipment names that make up the build.
|
2021-03-13 23:55:08 -08:00
|
|
|
* In order: boots, Chestplate, Leggings, Boots, Ring1, Ring2, Brace, Neck, Weapon.
|
2021-01-18 07:31:20 +10:30
|
|
|
* @param {Number[]} powders : Powder application. List of lists of integers (powder IDs).
|
2021-03-13 23:55:08 -08:00
|
|
|
* In order: boots, Chestplate, Leggings, Boots, Weapon.
|
2021-01-18 07:31:20 +10:30
|
|
|
* @param {Object[]} inputerrors : List of instances of error-like classes.
|
2022-06-19 00:42:49 -07:00
|
|
|
*
|
|
|
|
* @param {Object[]} tomes: List of tomes.
|
|
|
|
* In order: 2x Weapon Mastery Tome, 4x Armor Mastery Tome, 1x Guild Tome.
|
|
|
|
* 2x Slaying Mastery Tome, 2x Dungeoneering Mastery Tome, 2x Gathering Mastery Tome are in game, but do not have "useful" stats (those that affect damage calculations or building)
|
2021-01-09 02:52:58 -06:00
|
|
|
*/
|
2022-06-23 03:42:18 -07:00
|
|
|
constructor(level, items, weapon){
|
2021-01-18 07:31:20 +10:30
|
|
|
|
|
|
|
if (level < 1) { //Should these be constants?
|
2021-01-06 18:02:10 -06:00
|
|
|
this.level = 1;
|
2021-01-18 07:31:20 +10:30
|
|
|
} else if (level > 106) {
|
2021-01-06 18:02:10 -06:00
|
|
|
this.level = 106;
|
2021-01-18 07:31:20 +10:30
|
|
|
} else if (level <= 106 && level >= 1) {
|
2021-01-06 18:02:10 -06:00
|
|
|
this.level = level;
|
2021-01-18 07:31:20 +10:30
|
|
|
} else if (typeof level === "string") {
|
|
|
|
this.level = level;
|
|
|
|
errors.push(new IncorrectInput(level, "a number", "level-choice"));
|
|
|
|
} else {
|
|
|
|
errors.push("Level is not a string or number.");
|
2021-01-06 18:02:10 -06:00
|
|
|
}
|
2021-01-18 07:31:20 +10:30
|
|
|
document.getElementById("level-choice").value = this.level;
|
|
|
|
|
2021-01-07 00:41:41 -06:00
|
|
|
this.availableSkillpoints = levelToSkillPoints(this.level);
|
2022-06-19 00:42:49 -07:00
|
|
|
this.equipment = items;
|
|
|
|
this.weapon = weapon;
|
2022-06-23 03:42:18 -07:00
|
|
|
this.items = this.equipment.concat([this.weapon]);
|
2021-01-07 00:41:41 -06:00
|
|
|
// return [equip_order, best_skillpoints, final_skillpoints, best_total];
|
2022-06-19 09:49:04 -07:00
|
|
|
|
|
|
|
// calc skillpoints requires statmaps only
|
2022-06-23 03:42:18 -07:00
|
|
|
let result = calculate_skillpoints(this.equipment.map((x) => x.statMap), this.weapon.statMap);
|
2021-01-07 00:41:41 -06:00
|
|
|
this.equip_order = result[0];
|
2021-07-27 03:04:12 -07:00
|
|
|
// How many skillpoints the player had to assign (5 number)
|
2021-01-07 00:41:41 -06:00
|
|
|
this.base_skillpoints = result[1];
|
2021-07-27 03:04:12 -07:00
|
|
|
// How many skillpoints the build ended up with (5 number)
|
2021-01-07 00:41:41 -06:00
|
|
|
this.total_skillpoints = result[2];
|
2021-07-27 03:04:12 -07:00
|
|
|
// How many skillpoints assigned (1 number, sum of base_skillpoints)
|
2021-01-07 00:41:41 -06:00
|
|
|
this.assigned_skillpoints = result[3];
|
2021-01-09 22:29:07 -06:00
|
|
|
this.activeSetCounts = result[4];
|
2021-01-13 19:17:01 -08:00
|
|
|
|
2021-01-07 22:31:29 -06:00
|
|
|
this.initBuildStats();
|
2021-01-06 18:02:10 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/*Returns build in string format
|
2021-01-07 00:41:41 -06:00
|
|
|
*/
|
2021-01-06 18:02:10 -06:00
|
|
|
toString(){
|
2022-06-23 03:42:18 -07:00
|
|
|
return [this.equipment,this.weapon].flat();
|
2021-01-06 18:02:10 -06:00
|
|
|
}
|
|
|
|
|
2021-01-10 02:02:23 -08:00
|
|
|
|
2021-01-07 23:36:57 -06:00
|
|
|
/* Get all stats for this build. Stores in this.statMap.
|
2021-01-07 13:32:36 -08:00
|
|
|
@pre The build itself should be valid. No checking of validity of pieces is done here.
|
|
|
|
*/
|
2021-01-07 22:31:29 -06:00
|
|
|
initBuildStats(){
|
|
|
|
|
2022-06-26 04:08:54 -07:00
|
|
|
let staticIDs = ["hp", "eDef", "tDef", "wDef", "fDef", "aDef", "str", "dex", "int", "def", "agi", "damMobs", "defMobs"];
|
2021-01-07 22:31:29 -06:00
|
|
|
|
2022-06-24 03:35:03 -07:00
|
|
|
let must_ids = [
|
|
|
|
"eMdPct","eMdRaw","eSdPct","eSdRaw","eDamPct","eDamRaw","eDamAddMin","eDamAddMax",
|
|
|
|
"tMdPct","tMdRaw","tSdPct","tSdRaw","tDamPct","tDamRaw","tDamAddMin","tDamAddMax",
|
|
|
|
"wMdPct","wMdRaw","wSdPct","wSdRaw","wDamPct","wDamRaw","wDamAddMin","wDamAddMax",
|
|
|
|
"fMdPct","fMdRaw","fSdPct","fSdRaw","fDamPct","fDamRaw","fDamAddMin","fDamAddMax",
|
|
|
|
"aMdPct","aMdRaw","aSdPct","aSdRaw","aDamPct","aDamRaw","aDamAddMin","aDamAddMax",
|
|
|
|
"nMdPct","nMdRaw","nSdPct","nSdRaw","nDamPct","nDamRaw","nDamAddMin","nDamAddMax", // neutral which is now an element
|
|
|
|
"mdPct","mdRaw","sdPct","sdRaw","damPct","damRaw","damAddMin","damAddMax", // These are the old ids. Become proportional.
|
|
|
|
"rMdPct","rMdRaw","rSdPct","rSdRaw","rDamPct","rDamRaw","rDamAddMin","rDamAddMax" // rainbow (the "element" of all minus neutral). rSdRaw is rainraw
|
|
|
|
]
|
|
|
|
|
2021-01-07 13:32:36 -08:00
|
|
|
//Create a map of this build's stats
|
|
|
|
let statMap = new Map();
|
2022-06-23 02:23:56 -07:00
|
|
|
statMap.set("defMultiplier", 1);
|
2021-01-07 22:31:29 -06:00
|
|
|
|
|
|
|
for (const staticID of staticIDs) {
|
|
|
|
statMap.set(staticID, 0);
|
2021-01-07 13:32:36 -08:00
|
|
|
}
|
2022-06-24 03:35:03 -07:00
|
|
|
for (const staticID of must_ids) {
|
|
|
|
statMap.set(staticID, 0);
|
|
|
|
}
|
2021-01-30 00:50:25 -08:00
|
|
|
statMap.set("hp", levelToHPBase(this.level));
|
2021-01-13 22:58:32 -08:00
|
|
|
|
2021-01-30 06:03:40 -06:00
|
|
|
let major_ids = new Set();
|
2021-01-08 18:56:07 -06:00
|
|
|
for (const item of this.items){
|
2022-06-19 09:49:04 -07:00
|
|
|
const item_stats = item.statMap;
|
|
|
|
for (let [id, value] of item_stats.get("maxRolls")) {
|
2021-07-06 20:36:12 -07:00
|
|
|
if (staticIDs.includes(id)) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-01-07 22:31:29 -06:00
|
|
|
statMap.set(id,(statMap.get(id) || 0)+value);
|
|
|
|
}
|
2022-06-26 04:08:54 -07:00
|
|
|
console.log(item_stats);
|
2021-01-07 22:31:29 -06:00
|
|
|
for (const staticID of staticIDs) {
|
2022-06-19 09:49:04 -07:00
|
|
|
if (item_stats.get(staticID)) {
|
2022-06-23 02:23:56 -07:00
|
|
|
statMap.set(staticID, statMap.get(staticID) + item_stats.get(staticID));
|
2021-01-10 07:18:53 -06:00
|
|
|
}
|
2021-01-07 13:32:36 -08:00
|
|
|
}
|
2022-06-19 09:49:04 -07:00
|
|
|
if (item_stats.get("majorIds")) {
|
|
|
|
for (const major_id of item_stats.get("majorIds")) {
|
2021-07-23 00:47:13 -07:00
|
|
|
major_ids.add(major_id);
|
|
|
|
}
|
2021-01-30 06:03:40 -06:00
|
|
|
}
|
2021-01-07 13:32:36 -08:00
|
|
|
}
|
2022-06-26 04:12:04 -07:00
|
|
|
statMap.set('damageMultiplier', 1 + (statMap.get('damMobs') / 100));
|
|
|
|
statMap.set('defMultiplier', 1 - (statMap.get('defMobs') / 100));
|
2021-01-30 06:03:40 -06:00
|
|
|
statMap.set("activeMajorIDs", major_ids);
|
2021-01-09 22:29:07 -06:00
|
|
|
for (const [setName, count] of this.activeSetCounts) {
|
2022-06-19 09:49:04 -07:00
|
|
|
const bonus = sets.get(setName).bonuses[count-1];
|
2021-01-09 21:40:15 -06:00
|
|
|
for (const id in bonus) {
|
2021-01-09 22:29:07 -06:00
|
|
|
if (skp_order.includes(id)) {
|
|
|
|
// pass. Don't include skillpoints in ids
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
statMap.set(id,(statMap.get(id) || 0)+bonus[id]);
|
|
|
|
}
|
2021-01-09 21:40:15 -06:00
|
|
|
}
|
|
|
|
}
|
2021-01-18 08:18:14 -06:00
|
|
|
statMap.set("poisonPct", 100);
|
2021-01-07 00:41:41 -06:00
|
|
|
|
2021-01-07 22:31:29 -06:00
|
|
|
// The stuff relevant for damage calculation!!! @ferricles
|
2022-06-19 09:49:04 -07:00
|
|
|
statMap.set("atkSpd", this.weapon.statMap.get("atkSpd"));
|
2021-01-13 22:58:32 -08:00
|
|
|
|
2021-01-07 22:31:29 -06:00
|
|
|
this.statMap = statMap;
|
2021-01-18 08:18:14 -06:00
|
|
|
}
|
2021-01-06 18:02:10 -06:00
|
|
|
}
|