Merge branch 'dev' into UI_test
probably broken somewhere
This commit is contained in:
commit
757a6aab35
14 changed files with 190 additions and 29 deletions
4
.gitignore
vendored
4
.gitignore
vendored
|
@ -4,6 +4,4 @@ sets/
|
||||||
|
|
||||||
.idea/
|
.idea/
|
||||||
*.iml
|
*.iml
|
||||||
node_modules/
|
.editor_log.txt
|
||||||
package.json
|
|
||||||
package-lock.json
|
|
||||||
|
|
|
@ -1,6 +1,22 @@
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<meta name="HandheldFriendly" content="true" />
|
||||||
|
<meta name="MobileOptimized" content="320" />
|
||||||
|
<meta name="viewport" content="initial-scale=1.0, maximum-scale=1.0, width=device-width, user-scalable=no" />
|
||||||
|
|
||||||
|
<!--OGP suite
|
||||||
|
<meta property="og:title" content="Wynnbuilder" />
|
||||||
|
<meta property="og:type" content="website" />
|
||||||
|
<meta property="og:image" content="https://wynnbuilder.github.io/media/icons/new/builder.png" />
|
||||||
|
<meta property="og:image:width" content="420" />
|
||||||
|
<meta property="og:image:height" content="420" />
|
||||||
|
<meta property="og:image:type" content="image/png" />
|
||||||
|
<meta property="og:description" id = "ogp-build-list" content = "">
|
||||||
|
<meta property="og:url" id = "ogp-url" content="" />
|
||||||
|
-->
|
||||||
|
|
||||||
<title>WynnBuilder</title>
|
<title>WynnBuilder</title>
|
||||||
<link rel="icon" href="../media/icons/new/builder.png" type="image/icon type">
|
<link rel="icon" href="../media/icons/new/builder.png" type="image/icon type">
|
||||||
|
|
||||||
|
|
12
clean.json
12
clean.json
|
@ -54944,7 +54944,8 @@
|
||||||
"wDamPct": 6,
|
"wDamPct": 6,
|
||||||
"type": "ring",
|
"type": "ring",
|
||||||
"fixID": true,
|
"fixID": true,
|
||||||
"id": 2611
|
"id": 2611,
|
||||||
|
"set": "Wynnterfest 2016"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "North Pole",
|
"name": "North Pole",
|
||||||
|
@ -55061,7 +55062,8 @@
|
||||||
"eDamPct": 6,
|
"eDamPct": 6,
|
||||||
"type": "necklace",
|
"type": "necklace",
|
||||||
"fixID": true,
|
"fixID": true,
|
||||||
"id": 2615
|
"id": 2615,
|
||||||
|
"set": "Wynnterfest 2016"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Saint's Leggings",
|
"name": "Saint's Leggings",
|
||||||
|
@ -55095,7 +55097,8 @@
|
||||||
"fDamPct": 6,
|
"fDamPct": 6,
|
||||||
"type": "bracelet",
|
"type": "bracelet",
|
||||||
"fixID": true,
|
"fixID": true,
|
||||||
"id": 2616
|
"id": 2616,
|
||||||
|
"set": "Wynnterfest 2016"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Saint's Shawl",
|
"name": "Saint's Shawl",
|
||||||
|
@ -55190,7 +55193,8 @@
|
||||||
"tDamPct": 6,
|
"tDamPct": 6,
|
||||||
"type": "ring",
|
"type": "ring",
|
||||||
"fixID": true,
|
"fixID": true,
|
||||||
"id": 2621
|
"id": 2621,
|
||||||
|
"set": "Wynnterfest 2016"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Sleigh Bell",
|
"name": "Sleigh Bell",
|
||||||
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1,3 +1,4 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
<html>
|
<html>
|
||||||
<meta http-equiv="refresh" content="0; URL=./builder" />
|
<meta http-equiv="refresh" content="0; URL=./builder" />
|
||||||
</html>
|
</html>
|
||||||
|
|
|
@ -588,14 +588,14 @@ function updateStats() {
|
||||||
let delta_total = 0;
|
let delta_total = 0;
|
||||||
for (let i in skp_order) {
|
for (let i in skp_order) {
|
||||||
let value = document.getElementById(skp_order[i] + "-skp").value;
|
let value = document.getElementById(skp_order[i] + "-skp").value;
|
||||||
if (value === ""){value = 0; setValue(skp_order[i] + "-skp", value)}
|
if (value === ""){value = "0"; setValue(skp_order[i] + "-skp", value)}
|
||||||
let manual_assigned = 0;
|
let manual_assigned = 0;
|
||||||
if (value.includes("+")) {
|
if (value.includes("+")) {
|
||||||
let skp = value.split("+");
|
let skp = value.split("+");
|
||||||
for (const s of skp) {
|
for (const s of skp) {
|
||||||
manual_assigned += parseInt(s,10);
|
manual_assigned += parseInt(s,10);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
manual_assigned = parseInt(value,10);
|
manual_assigned = parseInt(value,10);
|
||||||
}
|
}
|
||||||
let delta = manual_assigned - skillpoints[i];
|
let delta = manual_assigned - skillpoints[i];
|
||||||
|
@ -861,6 +861,7 @@ function calculateBuildStats() {
|
||||||
|
|
||||||
location.hash = encodeBuild();
|
location.hash = encodeBuild();
|
||||||
clear_highlights();
|
clear_highlights();
|
||||||
|
updateOGP();
|
||||||
}
|
}
|
||||||
|
|
||||||
function copyBuild() {
|
function copyBuild() {
|
||||||
|
@ -1076,3 +1077,4 @@ function init2() {
|
||||||
load_ing_init(init);
|
load_ing_init(init);
|
||||||
}
|
}
|
||||||
load_init(init2);
|
load_init(init2);
|
||||||
|
updateOGP();
|
|
@ -5,9 +5,10 @@
|
||||||
*/
|
*/
|
||||||
function applyArmorPowders(expandedItem, powders) {
|
function applyArmorPowders(expandedItem, powders) {
|
||||||
applyArmorPowdersOnce(expandedItem, powders);
|
applyArmorPowdersOnce(expandedItem, powders);
|
||||||
if (expandedItem.get("crafted")) {
|
// NOTE: armor powder only applies once!
|
||||||
applyArmorPowdersOnce(expandedItem, powders);
|
//if (expandedItem.get("crafted")) {
|
||||||
}
|
// applyArmorPowdersOnce(expandedItem, powders);
|
||||||
|
//}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -202,7 +202,7 @@ function redraw(data) {
|
||||||
dps_getter_func = tmp;
|
dps_getter_func = tmp;
|
||||||
prepowder = tmp2;
|
prepowder = tmp2;
|
||||||
let _bbox = bbox();
|
let _bbox = bbox();
|
||||||
let x = d3.scaleLinear([70, 105], [margin.left, bbox().width - margin.right]);
|
let x = d3.scaleLinear([70, 110], [margin.left, bbox().width - margin.right]);
|
||||||
let y = d3.scaleLinear([0, max_dps_base * 1.1], [bbox().height - margin.bottom, margin.top]);
|
let y = d3.scaleLinear([0, max_dps_base * 1.1], [bbox().height - margin.bottom, margin.top]);
|
||||||
|
|
||||||
let type_mod = weapon_type_mods.get(current_type);
|
let type_mod = weapon_type_mods.get(current_type);
|
||||||
|
|
27
js/items.js
27
js/items.js
|
@ -71,14 +71,14 @@ const translate_mappings = {
|
||||||
"Custom Skin": "skin",
|
"Custom Skin": "skin",
|
||||||
//"Item Category": "category",
|
//"Item Category": "category",
|
||||||
|
|
||||||
"1st Spell Cost %": "spPct1",
|
"1st Spell Cost %": "-spPct1",
|
||||||
"1st Spell Cost Raw": "spRaw1",
|
"1st Spell Cost Raw": "-spRaw1",
|
||||||
"2nd Spell Cost %": "spPct2",
|
"2nd Spell Cost %": "-spPct2",
|
||||||
"2nd Spell Cost Raw": "spRaw2",
|
"2nd Spell Cost Raw": "-spRaw2",
|
||||||
"3rd Spell Cost %": "spPct3",
|
"3rd Spell Cost %": "-spPct3",
|
||||||
"3rd Spell Cost Raw": "spRaw3",
|
"3rd Spell Cost Raw": "-spRaw3",
|
||||||
"4th Spell Cost %": "spPct4",
|
"4th Spell Cost %": "-spPct4",
|
||||||
"4th Spell Cost Raw": "spRaw4",
|
"4th Spell Cost Raw": "-spRaw4",
|
||||||
|
|
||||||
"Rainbow Spell Damage": "rainbowRaw",
|
"Rainbow Spell Damage": "rainbowRaw",
|
||||||
"Sprint": "sprint",
|
"Sprint": "sprint",
|
||||||
|
@ -95,11 +95,11 @@ const special_mappings = {
|
||||||
"Sum (Mana Sustain)": "s:mr+ms",
|
"Sum (Mana Sustain)": "s:mr+ms",
|
||||||
"Sum (Life Sustain)": "s:hpr+ls",
|
"Sum (Life Sustain)": "s:hpr+ls",
|
||||||
"Sum (Health + Health Bonus)": "s:hp+hpBonus",
|
"Sum (Health + Health Bonus)": "s:hp+hpBonus",
|
||||||
"No Strength Req": "f:strReq==0",
|
"No Strength Req": "f:strReq=0",
|
||||||
"No Dexterity Req": "f:dexReq==0",
|
"No Dexterity Req": "f:dexReq=0",
|
||||||
"No Intelligence Req": "f:intReq==0",
|
"No Intelligence Req": "f:intReq=0",
|
||||||
"No Agility Req": "f:agiReq==0",
|
"No Agility Req": "f:agiReq=0",
|
||||||
"No Defense Req": "f:defReq==0",
|
"No Defense Req": "f:defReq=0",
|
||||||
};
|
};
|
||||||
|
|
||||||
let itemFilters = document.getElementById("filter-items");
|
let itemFilters = document.getElementById("filter-items");
|
||||||
|
@ -168,6 +168,7 @@ function doItemSearch() {
|
||||||
let filter_dat = translate_mappings[raw_dat];
|
let filter_dat = translate_mappings[raw_dat];
|
||||||
if (filter_dat !== undefined) {
|
if (filter_dat !== undefined) {
|
||||||
queries.push("s:"+filter_dat);
|
queries.push("s:"+filter_dat);
|
||||||
|
queries.push("f:"+filter_dat+"!=0");
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
filter_dat = special_mappings[raw_dat];
|
filter_dat = special_mappings[raw_dat];
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
const DB_VERSION = 89;
|
const DB_VERSION = 90;
|
||||||
// @See https://github.com/mdn/learning-area/blob/master/javascript/apis/client-side-storage/indexeddb/video-store/index.jsA
|
// @See https://github.com/mdn/learning-area/blob/master/javascript/apis/client-side-storage/indexeddb/video-store/index.jsA
|
||||||
|
|
||||||
let db;
|
let db;
|
||||||
|
|
25
js/utils.js
25
js/utils.js
|
@ -3,6 +3,31 @@ const url_base = getUrl.protocol + "//" + getUrl.host + "/" + getUrl.pathname.sp
|
||||||
|
|
||||||
const zip = (a, b) => a.map((k, i) => [k, b[i]]);
|
const zip = (a, b) => a.map((k, i) => [k, b[i]]);
|
||||||
|
|
||||||
|
//updates all the OGP tags for a webpage. Should be called when build changes
|
||||||
|
function updateOGP() {
|
||||||
|
//update the embed URL
|
||||||
|
let url_elem = document.getElementById("ogp-url");
|
||||||
|
if (url_elem) {
|
||||||
|
url_elem.content = url_base+location.hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
//update the embed text content
|
||||||
|
let build_elem = document.getElementById("ogp-build-list");
|
||||||
|
if (build_elem && player_build) {
|
||||||
|
let text = "WynnBuilder build:\n"+
|
||||||
|
"> "+player_build.helmet.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.chestplate.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.leggings.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.boots.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.ring1.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.ring2.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.bracelet.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.necklace.get("displayName")+"\n"+
|
||||||
|
"> "+player_build.weapon.get("displayName")+" ["+player_build.weapon.get("powders").map(x => powderNames.get(x)).join("")+"]";
|
||||||
|
build_elem.content = text;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function clamp(num, low, high){
|
function clamp(num, low, high){
|
||||||
return Math.min(Math.max(num, low), high);
|
return Math.min(Math.max(num, low), high);
|
||||||
}
|
}
|
||||||
|
|
48
testing/ms_linalg.py
Normal file
48
testing/ms_linalg.py
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
import numpy as np
|
||||||
|
import numpy.linalg as la
|
||||||
|
import matplotlib.pyplot as plt
|
||||||
|
|
||||||
|
# Attack speed independent. Fast speed losses not accounted for
|
||||||
|
mana_consumption = 3
|
||||||
|
mana_steal = 10 # /3s
|
||||||
|
mana_regen = 0 # /5s
|
||||||
|
#mana_steal = 5 # /3s
|
||||||
|
#mana_regen = 5 # /5s
|
||||||
|
natural_regen = 1
|
||||||
|
|
||||||
|
weight_natural = 8/15 # 4/5 * 2/3
|
||||||
|
weight_mr = 2/15 # 1/5 * 2/3
|
||||||
|
weight_ms = 4/15 # 4/5 * 1/3
|
||||||
|
weight_mr_ms = 1/15 # 1/5 * 1/3
|
||||||
|
|
||||||
|
MAX_MANA = 20
|
||||||
|
transition_matrix = np.zeros((MAX_MANA, MAX_MANA))
|
||||||
|
for i in range(MAX_MANA):
|
||||||
|
natural_state = max(0, i - mana_consumption + natural_regen)
|
||||||
|
mr_state = min(19, natural_state + mana_regen)
|
||||||
|
ms_state = min(19, natural_state + mana_steal)
|
||||||
|
mr_ms_state = min(19, natural_state + mana_regen + mana_steal)
|
||||||
|
transition_matrix[natural_state, i] = weight_natural
|
||||||
|
transition_matrix[mr_state, i] += weight_mr
|
||||||
|
transition_matrix[ms_state, i] += weight_ms
|
||||||
|
transition_matrix[mr_ms_state, i] += weight_mr_ms
|
||||||
|
|
||||||
|
eigval, eigvec = la.eig(transition_matrix)
|
||||||
|
print(eigval)
|
||||||
|
eps = 0.00001
|
||||||
|
ind = np.argwhere(abs(eigval - 1) < eps)
|
||||||
|
steady_state = abs(eigvec[:, ind])
|
||||||
|
steady_state /= np.sum(steady_state)
|
||||||
|
cumulative = np.cumsum(steady_state)
|
||||||
|
print("mana\tprob cumsum")
|
||||||
|
for i in range(MAX_MANA):
|
||||||
|
print(f"{i+1}\t{cumulative[i]}")
|
||||||
|
|
||||||
|
plt.figure()
|
||||||
|
plt.scatter(range(len(steady_state)), steady_state)
|
||||||
|
plt.xlim(0, 19)
|
||||||
|
plt.ylim(0, 0.2)
|
||||||
|
plt.xlabel("Mana Value")
|
||||||
|
plt.ylabel("Probability at t=infty")
|
||||||
|
plt.title(f"Build={mana_regen}mr,{mana_steal}ms,{mana_consumption}mana/sec")
|
||||||
|
plt.show()
|
65
testing/ms_sslow.py
Normal file
65
testing/ms_sslow.py
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
import numpy as np
|
||||||
|
import numpy.linalg as la
|
||||||
|
import matplotlib.pyplot as plt
|
||||||
|
|
||||||
|
# Super slow attack speed. (Idealized to 1 hit/2s, 2/3 chance of proc
|
||||||
|
mana_consumption = 6
|
||||||
|
mana_steal = 14 # /3s
|
||||||
|
mana_regen = 6 # /5s
|
||||||
|
#mana_steal = 5 # /3s
|
||||||
|
#mana_regen = 5 # /5s
|
||||||
|
natural_regen = 1
|
||||||
|
|
||||||
|
ms_period = 2
|
||||||
|
ms_chance = ms_period / 3
|
||||||
|
no_ms_chance = 1 - ms_chance
|
||||||
|
|
||||||
|
MAX_MANA = 20
|
||||||
|
TIME_CYCLE = 10
|
||||||
|
transition_matrix = np.zeros((MAX_MANA * TIME_CYCLE, MAX_MANA * TIME_CYCLE))
|
||||||
|
for j in range(TIME_CYCLE):
|
||||||
|
for i in range(MAX_MANA):
|
||||||
|
natural_state = max(0, i - mana_consumption + natural_regen)
|
||||||
|
if j % 5 == 0: # mr activation
|
||||||
|
natural_state = min(19, natural_state + mana_regen)
|
||||||
|
next_ind = ((j+1) % TIME_CYCLE) * MAX_MANA
|
||||||
|
if j % ms_period == 0: # ms activation
|
||||||
|
ms_state = min(19, natural_state + mana_steal)
|
||||||
|
transition_matrix[next_ind + natural_state, i+j*MAX_MANA] = no_ms_chance
|
||||||
|
transition_matrix[next_ind + ms_state, i+j*MAX_MANA] += ms_chance
|
||||||
|
else:
|
||||||
|
transition_matrix[next_ind + natural_state, i+j*MAX_MANA] = 1
|
||||||
|
|
||||||
|
eigval, eigvec = la.eig(transition_matrix)
|
||||||
|
print(eigval)
|
||||||
|
eps = 0.00001
|
||||||
|
ind = np.argwhere(abs(eigval - 1) < eps)
|
||||||
|
steady_state = np.sum(abs(eigvec[:, ind]).reshape((TIME_CYCLE, MAX_MANA)), axis=0)
|
||||||
|
steady_state /= np.sum(steady_state)
|
||||||
|
cumulative = np.cumsum(steady_state)
|
||||||
|
print("mana\tcumulative probability")
|
||||||
|
for i in range(MAX_MANA):
|
||||||
|
print(f"{i+1}\t{cumulative[i]}")
|
||||||
|
|
||||||
|
mana_limit = 6+mana_consumption
|
||||||
|
|
||||||
|
x_ticks = list(range(len(steady_state)))
|
||||||
|
plt.figure()
|
||||||
|
plt.scatter(x_ticks, steady_state, label="mana values")
|
||||||
|
plt.xlim(0, 19)
|
||||||
|
plt.ylim(0, 0.3)
|
||||||
|
plt.axvline(x=mana_limit, color="red")
|
||||||
|
plt.xlabel("Mana Value")
|
||||||
|
plt.xticks(x_ticks)
|
||||||
|
plt.ylabel("Probability at t=infty")
|
||||||
|
plt.legend()
|
||||||
|
ax2 = plt.gca().twinx()
|
||||||
|
ax2.plot(x_ticks, cumulative, label="cumulative probability", color="pink")
|
||||||
|
|
||||||
|
plt.text(mana_limit - 0.2, cumulative[mana_limit] + 0.03, f"time with sprint loss: {cumulative[mana_limit]*100:.2f}%", horizontalalignment='right')
|
||||||
|
plt.scatter((mana_limit,), (cumulative[mana_limit],), color="red")
|
||||||
|
ax2.set_ylim(0, 1)
|
||||||
|
ax2.set_ylabel("Cumulative probability at t=infty")
|
||||||
|
plt.title(f"Super Slow Speed: Build={mana_regen}mr,{mana_steal}ms,{mana_consumption}mana/sec")
|
||||||
|
plt.legend()
|
||||||
|
plt.show()
|
Loading…
Add table
Reference in a new issue