Add cool visualization thingers
This commit is contained in:
parent
afd7d0601f
commit
17a19cd4e6
6 changed files with 126 additions and 13 deletions
|
@ -44847,7 +44847,7 @@
|
||||||
"slots": 2,
|
"slots": 2,
|
||||||
"drop": "NORMAL",
|
"drop": "NORMAL",
|
||||||
"nDam": "0-0",
|
"nDam": "0-0",
|
||||||
"fDam": "320-260",
|
"fDam": "320-360",
|
||||||
"wDam": "0-0",
|
"wDam": "0-0",
|
||||||
"aDam": "0-0",
|
"aDam": "0-0",
|
||||||
"tDam": "0-0",
|
"tDam": "0-0",
|
||||||
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
120
dps_vis.js
120
dps_vis.js
|
@ -75,14 +75,18 @@ let strDex = false;
|
||||||
|
|
||||||
let dps_data = {};
|
let dps_data = {};
|
||||||
let current_data = null;
|
let current_data = null;
|
||||||
|
let current_type = "wand";
|
||||||
|
let baseline_x = [];
|
||||||
|
let baseline_y = [];
|
||||||
|
|
||||||
let baseUrl = getUrl.protocol + "//" + getUrl.host + "/";// + getUrl.pathname.split('/')[1];
|
let baseUrl = getUrl.protocol + "//" + getUrl.host + "/";// + getUrl.pathname.split('/')[1];
|
||||||
(async function() {
|
(async function() {
|
||||||
dps_data = await (await fetch(baseUrl + "/dps_data_compress.json")).json()
|
dps_data = await (await fetch(baseUrl + "/dps_data_compress.json")).json();
|
||||||
console.log(dps_data)
|
|
||||||
|
|
||||||
dps_getter_func = d => d[4];
|
dps_getter_func = d => d[4];
|
||||||
current_data = dps_data.wand;
|
current_data = dps_data.wand;
|
||||||
|
baseline_x = dps_data.baseline_xs;
|
||||||
|
baseline_y = dps_data.baseline_ys;
|
||||||
|
|
||||||
d3.select(window)
|
d3.select(window)
|
||||||
.on("resize", function() {
|
.on("resize", function() {
|
||||||
|
@ -104,9 +108,87 @@ let colorMap = new Map(
|
||||||
["Set","#5f5"]
|
["Set","#5f5"]
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
|
let tiers_mod = new Map(
|
||||||
|
[
|
||||||
|
["Normal", 0.8],
|
||||||
|
["Unique", 1.0],
|
||||||
|
["Rare", 1.1],
|
||||||
|
["Legendary", 1.3],
|
||||||
|
["Fabled", 1.5],
|
||||||
|
["Mythic", 1.7],
|
||||||
|
["Set", 1.05]
|
||||||
|
]
|
||||||
|
);
|
||||||
|
let weapon_type_mods = new Map(
|
||||||
|
[
|
||||||
|
["wand", 0.6],
|
||||||
|
["spear", 0.8],
|
||||||
|
["dagger", 1.0],
|
||||||
|
["bow", 1.2],
|
||||||
|
["relik", 1.2]
|
||||||
|
]
|
||||||
|
);
|
||||||
|
let tier_baselines = new Map()
|
||||||
|
for (let tier of tiers_mod.keys()) {
|
||||||
|
let line_top = graph.append("path");
|
||||||
|
let line_bot = graph.append("path");
|
||||||
|
tier_baselines.set(tier, [line_top, line_bot]);
|
||||||
|
}
|
||||||
|
|
||||||
|
let circles = graph.append("g");
|
||||||
|
|
||||||
|
let details = graph.append("g")
|
||||||
|
.attr("fill", "#444");
|
||||||
|
details.append("rect").attr("z", "100");
|
||||||
|
|
||||||
|
function showDetails(data, i, xfunc, yfunc) {
|
||||||
|
let xpos = xfunc(i[1]);
|
||||||
|
let ypos = yfunc(dps_getter_func(i));
|
||||||
|
let texts = [[1, i[0]],
|
||||||
|
[3, "Prepowder: "+i[4].toFixed(2)],
|
||||||
|
[4, "Postpowder: "+i[5].toFixed(2)],
|
||||||
|
[5, "Prepowder (1.20.3): "+i[6].toFixed(2)],
|
||||||
|
[6, "Postpowder (1.20.3): "+i[7].toFixed(2)],
|
||||||
|
[7, "Prepowder drop: "+((i[4]-i[6])/i[4]*100).toFixed(2) + "%"],
|
||||||
|
[8, "Postpowder drop: "+((i[5]-i[7])/i[5]*100).toFixed(2) + "%"],
|
||||||
|
[10, "Design Modifier: "+i[9].toFixed(2)],
|
||||||
|
[11, "Explained D. Mod: "+i[8].toFixed(2)]];
|
||||||
|
let idx = 12;
|
||||||
|
for (const explanation of i[10]) {
|
||||||
|
texts.push([idx, explanation[0] + ": " + explanation[1]]);
|
||||||
|
idx += 1;
|
||||||
|
}
|
||||||
|
details.select("rect")
|
||||||
|
.attr("x", xpos)
|
||||||
|
.attr("y", ypos)
|
||||||
|
.attr("width", 200)
|
||||||
|
.attr("height", (14 * idx) - 4)
|
||||||
|
.attr("opacity", 0.8);
|
||||||
|
details.selectAll("text")
|
||||||
|
.data(texts, d => d[0])
|
||||||
|
.join("text")
|
||||||
|
.style("font-size", "12px")
|
||||||
|
.attr("x", xpos+5)
|
||||||
|
.attr("y", d => d[0]*14 + ypos)
|
||||||
|
.attr("text-anchor", "start")
|
||||||
|
.attr("fill", "#fff")
|
||||||
|
.attr("opacity", 1)
|
||||||
|
.text(d => d[1]);
|
||||||
|
}
|
||||||
|
function hideDetails() {
|
||||||
|
details.select("rect")
|
||||||
|
.attr("x", 0)
|
||||||
|
.attr("y", 0)
|
||||||
|
.attr("width", 0)
|
||||||
|
.attr("height", 0)
|
||||||
|
.attr("opacity", 0);
|
||||||
|
details.selectAll("text")
|
||||||
|
.data([])
|
||||||
|
.join("text");
|
||||||
|
}
|
||||||
|
|
||||||
function redraw(data) {
|
function redraw(data) {
|
||||||
|
hideDetails();
|
||||||
let max_dps_base = 0;
|
let max_dps_base = 0;
|
||||||
let tmp = dps_getter_func;
|
let tmp = dps_getter_func;
|
||||||
let tmp2 = prepowder;
|
let tmp2 = prepowder;
|
||||||
|
@ -119,14 +201,38 @@ function redraw(data) {
|
||||||
}
|
}
|
||||||
dps_getter_func = tmp;
|
dps_getter_func = tmp;
|
||||||
prepowder = tmp2;
|
prepowder = tmp2;
|
||||||
|
let _bbox = bbox();
|
||||||
let x = d3.scaleLinear([70, 105], [margin.left, bbox().width - margin.right]);
|
let x = d3.scaleLinear([70, 105], [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 _bbox = bbox();
|
|
||||||
|
let type_mod = weapon_type_mods.get(current_type);
|
||||||
|
for (let tier of tiers_mod.keys()) {
|
||||||
|
let res = tier_baselines.get(tier);
|
||||||
|
let line_top = res[0];
|
||||||
|
let line_bot = res[1];
|
||||||
|
let tier_mod = tiers_mod.get(tier);
|
||||||
|
let y_max = baseline_y.map(x => 2.1*x*tier_mod*type_mod);
|
||||||
|
let y_min = baseline_y.map(x => 2.0*x*tier_mod*type_mod);
|
||||||
|
line_top.datum(zip(baseline_x, y_max))
|
||||||
|
.attr("fill", "none")
|
||||||
|
.attr("stroke", d => colorMap.get(tier))
|
||||||
|
.attr("d", d3.line()
|
||||||
|
.x(function(d) { return x(d[0]) })
|
||||||
|
.y(function(d) { return y(d[1]) })
|
||||||
|
)
|
||||||
|
line_bot.datum(zip(baseline_x, y_min))
|
||||||
|
.attr("fill", "none")
|
||||||
|
.attr("stroke", d => colorMap.get(tier))
|
||||||
|
.attr("d", d3.line()
|
||||||
|
.x(function(d) { return x(d[0]) })
|
||||||
|
.y(function(d) { return y(d[1]) })
|
||||||
|
)
|
||||||
|
}
|
||||||
graph.attr("viewBox", [0, 0, _bbox.width, _bbox.height]);
|
graph.attr("viewBox", [0, 0, _bbox.width, _bbox.height]);
|
||||||
xAxis(_xAxis, x);
|
xAxis(_xAxis, x);
|
||||||
yAxis(_yAxis, y);
|
yAxis(_yAxis, y);
|
||||||
grid(_grid1, _grid2, x, y);
|
grid(_grid1, _grid2, x, y);
|
||||||
graph.selectAll('circle')
|
circles.selectAll('circle')
|
||||||
.data(data, d => d[0])
|
.data(data, d => d[0])
|
||||||
.join(
|
.join(
|
||||||
function(enter) {
|
function(enter) {
|
||||||
|
@ -135,11 +241,14 @@ function redraw(data) {
|
||||||
.attr("cx", d => x(d[1]))
|
.attr("cx", d => x(d[1]))
|
||||||
.attr("cy", d => y(dps_getter_func(d)))
|
.attr("cy", d => y(dps_getter_func(d)))
|
||||||
.attr("r", d => 5)
|
.attr("r", d => 5)
|
||||||
|
.on("click", (d, i) => showDetails(d, i, x, y))
|
||||||
|
.on("mouseover", function() { d3.select(this).raise(); })
|
||||||
.call(circle => circle.append("title")
|
.call(circle => circle.append("title")
|
||||||
.text(d => [d[0], "DPS: "+dps_getter_func(d)].join("\n")));
|
.text(d => [d[0], "DPS: "+dps_getter_func(d)].join("\n")));
|
||||||
},
|
},
|
||||||
update => update.attr("cx", d => x(d[1]))
|
update => update.attr("cx", d => x(d[1]))
|
||||||
.attr("cy", d => y(dps_getter_func(d)))
|
.attr("cy", d => y(dps_getter_func(d)))
|
||||||
|
.on("click", (d, i) => showDetails(d, i, x, y))
|
||||||
.select("title")
|
.select("title")
|
||||||
.text(d => [d[0], "DPS: "+dps_getter_func(d)].join("\n")),
|
.text(d => [d[0], "DPS: "+dps_getter_func(d)].join("\n")),
|
||||||
exit => exit.remove()
|
exit => exit.remove()
|
||||||
|
@ -147,6 +256,7 @@ function redraw(data) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function setData(type) {
|
function setData(type) {
|
||||||
|
current_type = type;
|
||||||
d3.select("#info").text("SELECTED ITEM TYPE: " + type);
|
d3.select("#info").text("SELECTED ITEM TYPE: " + type);
|
||||||
current_data = dps_data[type];
|
current_data = dps_data[type];
|
||||||
redraw(current_data);
|
redraw(current_data);
|
||||||
|
@ -169,6 +279,8 @@ function togglePowder() {
|
||||||
|
|
||||||
function toggleStrDex() {
|
function toggleStrDex() {
|
||||||
strDex = !strDex;
|
strDex = !strDex;
|
||||||
|
if (strDex) baseline_y = dps_data.baseline_ys_new
|
||||||
|
else baseline_y = dps_data.baseline_ys
|
||||||
d3.select("#strDexToggle").text("1.20.3 ("+strDex+")");
|
d3.select("#strDexToggle").text("1.20.3 ("+strDex+")");
|
||||||
setGetterFunc();
|
setGetterFunc();
|
||||||
redraw(current_data);
|
redraw(current_data);
|
||||||
|
|
|
@ -71,8 +71,7 @@ for k in attack_speed_mods:
|
||||||
dps_to_baseline[k] = 1/mult
|
dps_to_baseline[k] = 1/mult
|
||||||
|
|
||||||
weapon_type_mods = {"wand": 0.6, "spear": 0.8, "dagger": 1.0, "bow": 1.2, "relik": 1.2}
|
weapon_type_mods = {"wand": 0.6, "spear": 0.8, "dagger": 1.0, "bow": 1.2, "relik": 1.2}
|
||||||
min_mult
|
print((min_mult, max_mult))
|
||||||
max_mult
|
|
||||||
|
|
||||||
tiers_mod = {"Normal": 0.8, "Unique": 1.0, "Rare": 1.1, "Legendary": 1.3, "Fabled": 1.5, "Mythic": 1.7, "Set": 1.05}
|
tiers_mod = {"Normal": 0.8, "Unique": 1.0, "Rare": 1.1, "Legendary": 1.3, "Fabled": 1.5, "Mythic": 1.7, "Set": 1.05}
|
||||||
tiers_colors = {"Normal": (0.9, 0.9, 0.9), "Unique": (1, 1, 1/3), "Rare": (1, 1/3, 1), "Legendary": (1/3, 1, 1), "Fabled": (1, 1/3, 1/3), "Mythic": (2/3, 0, 2/3), "Set": (1/3, 1, 1/3)}
|
tiers_colors = {"Normal": (0.9, 0.9, 0.9), "Unique": (1, 1, 1/3), "Rare": (1, 1/3, 1), "Legendary": (1/3, 1, 1), "Fabled": (1, 1/3, 1/3), "Mythic": (2/3, 0, 2/3), "Set": (1/3, 1, 1/3)}
|
||||||
|
@ -126,13 +125,13 @@ def guess_design_modifier(item, base_dps):
|
||||||
actual_modifier = (item_baseline - nominal_baseline) / nominal_baseline
|
actual_modifier = (item_baseline - nominal_baseline) / nominal_baseline
|
||||||
explained_baseline = nominal_baseline * (1 + total_modifier)
|
explained_baseline = nominal_baseline * (1 + total_modifier)
|
||||||
delta = (item_baseline - explained_baseline) / nominal_baseline
|
delta = (item_baseline - explained_baseline) / nominal_baseline
|
||||||
if delta >= 0.04 and delta <= 0.05:
|
if delta >= 0.04 and delta <= 0.06:
|
||||||
total_modifier += 0.05
|
total_modifier += 0.05
|
||||||
explanation.append(("offensive", 0.05))
|
explanation.append(("offensive", 0.05))
|
||||||
elif delta >= 0.08:
|
elif delta >= 0.08:
|
||||||
total_modifier += 0.1
|
total_modifier += 0.1
|
||||||
explanation.append(("hyper_offensive", 0.1))
|
explanation.append(("hyper_offensive", 0.1))
|
||||||
elif delta <= -0.04 and delta >= -0.05:
|
elif delta <= -0.04 and delta >= -0.06:
|
||||||
total_modifier -= 0.05
|
total_modifier -= 0.05
|
||||||
explanation.append(("defensive", -0.05))
|
explanation.append(("defensive", -0.05))
|
||||||
elif delta <= -0.08:
|
elif delta <= -0.08:
|
||||||
|
|
2
utils.js
2
utils.js
|
@ -1,6 +1,8 @@
|
||||||
let getUrl = window.location;
|
let getUrl = window.location;
|
||||||
const url_base = getUrl.protocol + "//" + getUrl.host + "/" + getUrl.pathname.split('/')[1];
|
const url_base = getUrl.protocol + "//" + getUrl.host + "/" + getUrl.pathname.split('/')[1];
|
||||||
|
|
||||||
|
const zip = (a, b) => a.map((k, i) => [k, b[i]]);
|
||||||
|
|
||||||
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);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue