2022-07-13 07:37:31 +00:00
|
|
|
let player_build;
|
|
|
|
let build_powders;
|
|
|
|
|
|
|
|
function getItemNameFromID(id) { return idMap.get(id); }
|
|
|
|
function getTomeNameFromID(id) { return tomeIDMap.get(id); }
|
|
|
|
|
2022-06-19 16:49:04 +00:00
|
|
|
function parsePowdering(powder_info) {
|
|
|
|
// TODO: Make this run in linear instead of quadratic time... ew
|
|
|
|
let powdering = [];
|
|
|
|
for (let i = 0; i < 5; ++i) {
|
|
|
|
let powders = "";
|
|
|
|
let n_blocks = Base64.toInt(powder_info.charAt(0));
|
|
|
|
// console.log(n_blocks + " blocks");
|
|
|
|
powder_info = powder_info.slice(1);
|
|
|
|
for (let j = 0; j < n_blocks; ++j) {
|
|
|
|
let block = powder_info.slice(0,5);
|
|
|
|
let six_powders = Base64.toInt(block);
|
|
|
|
for (let k = 0; k < 6 && six_powders != 0; ++k) {
|
|
|
|
powders += powderNames.get((six_powders & 0x1f) - 1);
|
|
|
|
six_powders >>>= 5;
|
|
|
|
}
|
|
|
|
powder_info = powder_info.slice(5);
|
|
|
|
}
|
|
|
|
powdering[i] = powders;
|
|
|
|
}
|
|
|
|
return [powdering, powder_info];
|
|
|
|
}
|
2022-06-19 07:42:49 +00:00
|
|
|
|
2022-06-30 15:03:41 +00:00
|
|
|
let atree_data = null;
|
2022-12-16 10:29:01 +00:00
|
|
|
const wynn_version_names = [
|
|
|
|
'2.0.1.1',
|
2023-02-17 11:40:23 +00:00
|
|
|
'2.0.1.2',
|
|
|
|
'2.0.2.1'
|
2022-12-16 10:29:01 +00:00
|
|
|
];
|
|
|
|
const WYNN_VERSION_LATEST = wynn_version_names.length - 1;
|
|
|
|
// Default to the newest version.
|
|
|
|
let wynn_version_id = WYNN_VERSION_LATEST;
|
2022-06-30 15:03:41 +00:00
|
|
|
|
2022-06-19 07:42:49 +00:00
|
|
|
/*
|
|
|
|
* Populate fields based on url, and calculate build.
|
2022-12-16 10:29:01 +00:00
|
|
|
* TODO: THIS CODE IS GOD AWFUL result of being lazy
|
|
|
|
* fix all the slice() and break into functions or do something about it... its inefficient, ugly and error prone
|
2022-06-19 07:42:49 +00:00
|
|
|
*/
|
2022-12-16 10:29:01 +00:00
|
|
|
async function parse_hash(url_tag) {
|
|
|
|
const default_load_promises = [ load_atree_data(wynn_version_names[WYNN_VERSION_LATEST]),
|
|
|
|
load_init(), load_ing_init(), load_tome_init() ];
|
|
|
|
if (!url_tag) {
|
|
|
|
await Promise.all(default_load_promises);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
//default values
|
|
|
|
let equipment = [null, null, null, null, null, null, null, null, null];
|
|
|
|
let tomes = [null, null, null, null, null, null, null];
|
|
|
|
let powdering = ["", "", "", "", ""];
|
|
|
|
let info = url_tag.split("_");
|
|
|
|
let version = info[0];
|
2022-12-20 21:34:11 +00:00
|
|
|
// Whether skillpoints are manually updated. True if they should be set to something other than default values
|
2022-12-16 10:29:01 +00:00
|
|
|
let save_skp = false;
|
|
|
|
let skillpoints = [0, 0, 0, 0, 0];
|
|
|
|
let level = 106;
|
|
|
|
|
|
|
|
let version_number = parseInt(version);
|
|
|
|
let data_str = info[1];
|
|
|
|
if (version_number >= 8) {
|
|
|
|
// parse query parameters
|
|
|
|
// https://stackoverflow.com/questions/901115/how-can-i-get-query-string-values-in-javascript
|
|
|
|
const url_params = new URLSearchParams(window.location.search);
|
|
|
|
const version_id = url_params.get('v');
|
|
|
|
wynn_version_id = parseInt(version_id);
|
|
|
|
if (isNaN(wynn_version_id) || wynn_version_id > WYNN_VERSION_LATEST || wynn_version_id < 0) {
|
|
|
|
// TODO: maybe make the NAN try to use the human readable version?
|
|
|
|
// NOTE: Failing silently... do we want to raise a loud error?
|
|
|
|
console.log("Explicit version not found or invalid, using latest version");
|
|
|
|
wynn_version_id = WYNN_VERSION_LATEST;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
console.log(`Build link for wynn version ${wynn_version_id} (${wynn_version_names[wynn_version_id]})`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Change the default to oldest. (A time before v8)
|
|
|
|
wynn_version_id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// the deal with this is because old versions should default to 0 (oldest wynn item version), and v8+ defaults to latest.
|
|
|
|
// its ugly... but i think this is the behavior we want...
|
|
|
|
if (wynn_version_id != WYNN_VERSION_LATEST) {
|
|
|
|
// force reload item database and such.
|
|
|
|
// TODO MUST: display a warning showing older version!
|
|
|
|
const msg = 'This build was created in an older version of wynncraft '
|
|
|
|
+ `(${wynn_version_names[wynn_version_id]} < ${wynn_version_names[WYNN_VERSION_LATEST]}). `
|
|
|
|
+ 'Would you like to update to the latest version? Updating may break the build and ability tree.';
|
|
|
|
|
|
|
|
if (confirm(msg)) {
|
|
|
|
wynn_version_id = WYNN_VERSION_LATEST;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
version_name = wynn_version_names[wynn_version_id];
|
|
|
|
const load_promises = [ load_atree_data(version_name),
|
|
|
|
load_old_version(version_name),
|
|
|
|
load_ings_old_version(version_name),
|
|
|
|
load_tome_old_version(version_name) ];
|
|
|
|
console.log("Loading old version data...", version_name)
|
|
|
|
await Promise.all(load_promises);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (wynn_version_id == WYNN_VERSION_LATEST) {
|
|
|
|
await Promise.all(default_load_promises);
|
|
|
|
}
|
|
|
|
|
|
|
|
//equipment (items)
|
|
|
|
// TODO: use filters
|
|
|
|
if (version_number < 4) {
|
|
|
|
let equipments = info[1];
|
|
|
|
for (let i = 0; i < 9; ++i ) {
|
|
|
|
let equipment_str = equipments.slice(i*3,i*3+3);
|
|
|
|
equipment[i] = getItemNameFromID(Base64.toInt(equipment_str));
|
|
|
|
}
|
|
|
|
data_str = equipments.slice(27);
|
|
|
|
}
|
|
|
|
else if (version_number == 4) {
|
|
|
|
let info_str = data_str;
|
|
|
|
let start_idx = 0;
|
|
|
|
for (let i = 0; i < 9; ++i ) {
|
|
|
|
if (info_str.charAt(start_idx) === "-") {
|
|
|
|
equipment[i] = "CR-"+info_str.slice(start_idx+1, start_idx+18);
|
|
|
|
start_idx += 18;
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
else {
|
|
|
|
let equipment_str = info_str.slice(start_idx, start_idx+3);
|
|
|
|
equipment[i] = getItemNameFromID(Base64.toInt(equipment_str));
|
|
|
|
start_idx += 3;
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
}
|
|
|
|
data_str = info_str.slice(start_idx);
|
|
|
|
}
|
|
|
|
else if (version_number <= 8) {
|
|
|
|
let info_str = data_str;
|
|
|
|
let start_idx = 0;
|
|
|
|
for (let i = 0; i < 9; ++i ) {
|
|
|
|
if (info_str.slice(start_idx,start_idx+3) === "CR-") {
|
|
|
|
equipment[i] = info_str.slice(start_idx, start_idx+20);
|
|
|
|
start_idx += 20;
|
|
|
|
} else if (info_str.slice(start_idx+3,start_idx+6) === "CI-") {
|
|
|
|
let len = Base64.toInt(info_str.slice(start_idx,start_idx+3));
|
|
|
|
equipment[i] = info_str.slice(start_idx+3,start_idx+3+len);
|
|
|
|
start_idx += (3+len);
|
|
|
|
} else {
|
|
|
|
let equipment_str = info_str.slice(start_idx, start_idx+3);
|
|
|
|
equipment[i] = getItemNameFromID(Base64.toInt(equipment_str));
|
|
|
|
start_idx += 3;
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
}
|
|
|
|
data_str = info_str.slice(start_idx);
|
|
|
|
}
|
|
|
|
//constant in all versions
|
|
|
|
for (let i in equipment) {
|
|
|
|
setValue(equipment_inputs[i], equipment[i]);
|
|
|
|
}
|
2022-06-19 07:42:49 +00:00
|
|
|
|
2022-12-16 10:29:01 +00:00
|
|
|
//level, skill point assignments, and powdering
|
2022-12-20 21:34:11 +00:00
|
|
|
if (version_number == 0) {
|
|
|
|
// do nothing! lol
|
|
|
|
} else if (version_number == 1) {
|
2022-12-16 10:29:01 +00:00
|
|
|
let powder_info = data_str;
|
|
|
|
let res = parsePowdering(powder_info);
|
|
|
|
powdering = res[0];
|
|
|
|
} else if (version_number == 2) {
|
|
|
|
save_skp = true;
|
|
|
|
let skillpoint_info = data_str.slice(0, 10);
|
|
|
|
for (let i = 0; i < 5; ++i ) {
|
|
|
|
skillpoints[i] = Base64.toIntSigned(skillpoint_info.slice(i*2,i*2+2));
|
|
|
|
}
|
2022-06-19 07:42:49 +00:00
|
|
|
|
2022-12-16 10:29:01 +00:00
|
|
|
let powder_info = data_str.slice(10);
|
|
|
|
let res = parsePowdering(powder_info);
|
|
|
|
powdering = res[0];
|
|
|
|
} else if (version_number <= 8){
|
|
|
|
level = Base64.toInt(data_str.slice(10,12));
|
|
|
|
setValue("level-choice",level);
|
|
|
|
save_skp = true;
|
|
|
|
let skillpoint_info = data_str.slice(0, 10);
|
|
|
|
for (let i = 0; i < 5; ++i ) {
|
|
|
|
skillpoints[i] = Base64.toIntSigned(skillpoint_info.slice(i*2,i*2+2));
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
|
|
|
|
let powder_info = data_str.slice(12);
|
|
|
|
|
|
|
|
let res = parsePowdering(powder_info);
|
|
|
|
powdering = res[0];
|
|
|
|
data_str = res[1];
|
|
|
|
}
|
|
|
|
// Tomes.
|
|
|
|
if (version_number >= 6) {
|
|
|
|
//tome values do not appear in anything before v6.
|
|
|
|
if (version_number < 8) {
|
2022-06-23 10:42:18 +00:00
|
|
|
for (let i in tomes) {
|
2022-12-16 10:29:01 +00:00
|
|
|
let tome_str = data_str.charAt(i);
|
2022-06-23 10:42:18 +00:00
|
|
|
let tome_name = getTomeNameFromID(Base64.toInt(tome_str));
|
|
|
|
setValue(tomeInputs[i], tome_name);
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
data_str = data_str.slice(7);
|
2022-06-30 15:03:41 +00:00
|
|
|
}
|
|
|
|
else {
|
2022-12-16 10:29:01 +00:00
|
|
|
// 2chr tome encoding to allow for more tomes.
|
|
|
|
for (let i in tomes) {
|
|
|
|
let tome_str = data_str.slice(2*i, 2*i+2);
|
|
|
|
let tome_name = getTomeNameFromID(Base64.toInt(tome_str));
|
|
|
|
setValue(tomeInputs[i], tome_name);
|
|
|
|
}
|
|
|
|
data_str = data_str.slice(14);
|
2022-06-30 15:03:41 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
}
|
2022-06-30 15:03:41 +00:00
|
|
|
|
2022-12-16 10:29:01 +00:00
|
|
|
if (version_number >= 7) {
|
|
|
|
// ugly af. only works since its the last thing. will be fixed with binary decode
|
|
|
|
atree_data = new BitVector(data_str);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
atree_data = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i in powder_inputs) {
|
|
|
|
setValue(powder_inputs[i], powdering[i]);
|
|
|
|
}
|
|
|
|
for (let i in skillpoints) {
|
|
|
|
setValue(skp_order[i] + "-skp", skillpoints[i]);
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-20 21:34:11 +00:00
|
|
|
|
|
|
|
return save_skp;
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Stores the entire build in a string using B64 encoding and adds it to the URL.
|
|
|
|
*/
|
2022-06-30 15:03:41 +00:00
|
|
|
function encodeBuild(build, powders, skillpoints, atree, atree_state) {
|
2022-06-19 07:42:49 +00:00
|
|
|
|
|
|
|
if (build) {
|
|
|
|
let build_string;
|
|
|
|
|
|
|
|
//V6 encoding - Tomes
|
2022-06-30 15:03:41 +00:00
|
|
|
//V7 encoding - ATree
|
2022-12-16 10:29:01 +00:00
|
|
|
//V8 encoding - wynn version
|
|
|
|
build_version = 8;
|
2022-06-19 07:42:49 +00:00
|
|
|
build_string = "";
|
|
|
|
tome_string = "";
|
|
|
|
|
|
|
|
for (const item of build.items) {
|
2022-06-19 20:44:02 +00:00
|
|
|
if (item.statMap.get("custom")) {
|
|
|
|
let custom = "CI-"+encodeCustom(item, true);
|
2022-06-19 07:42:49 +00:00
|
|
|
build_string += Base64.fromIntN(custom.length, 3) + custom;
|
2022-12-16 10:29:01 +00:00
|
|
|
//build_version = Math.max(build_version, 5);
|
2022-06-19 20:44:02 +00:00
|
|
|
} else if (item.statMap.get("crafted")) {
|
|
|
|
build_string += "CR-"+encodeCraft(item);
|
|
|
|
} else if (item.statMap.get("category") === "tome") {
|
2022-06-23 09:23:56 +00:00
|
|
|
let tome_id = item.statMap.get("id");
|
2022-06-19 07:42:49 +00:00
|
|
|
if (tome_id <= 60) {
|
|
|
|
// valid normal tome. ID 61-63 is for NONE tomes.
|
2022-12-16 10:29:01 +00:00
|
|
|
//build_version = Math.max(build_version, 6);
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
2022-12-16 10:29:01 +00:00
|
|
|
tome_string += Base64.fromIntN(tome_id, 2);
|
2022-06-19 07:42:49 +00:00
|
|
|
} else {
|
2022-06-19 20:44:02 +00:00
|
|
|
build_string += Base64.fromIntN(item.statMap.get("id"), 3);
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-20 13:12:22 +00:00
|
|
|
for (const skp of skillpoints) {
|
|
|
|
build_string += Base64.fromIntN(skp, 2); // Maximum skillpoints: 2048
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
|
|
|
build_string += Base64.fromIntN(build.level, 2);
|
2022-06-19 20:44:02 +00:00
|
|
|
for (const _powderset of powders) {
|
2022-06-19 07:42:49 +00:00
|
|
|
let n_bits = Math.ceil(_powderset.length / 6);
|
|
|
|
build_string += Base64.fromIntN(n_bits, 1); // Hard cap of 378 powders.
|
|
|
|
// Slice copy.
|
|
|
|
let powderset = _powderset.slice();
|
|
|
|
while (powderset.length != 0) {
|
|
|
|
let firstSix = powderset.slice(0,6).reverse();
|
|
|
|
let powder_hash = 0;
|
|
|
|
for (const powder of firstSix) {
|
|
|
|
powder_hash = (powder_hash << 5) + 1 + powder; // LSB will be extracted first.
|
|
|
|
}
|
|
|
|
build_string += Base64.fromIntN(powder_hash, 5);
|
|
|
|
powderset = powderset.slice(6);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
build_string += tome_string;
|
|
|
|
|
2022-07-05 07:51:02 +00:00
|
|
|
if (atree.length > 0 && atree_state.get(atree[0].ability.id).active) {
|
2022-12-16 10:29:01 +00:00
|
|
|
//build_version = Math.max(build_version, 7);
|
2022-06-30 15:03:41 +00:00
|
|
|
const bitvec = encode_atree(atree, atree_state);
|
|
|
|
build_string += bitvec.toB64();
|
|
|
|
}
|
|
|
|
|
2022-06-19 07:42:49 +00:00
|
|
|
return build_version.toString() + "_" + build_string;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-16 10:29:01 +00:00
|
|
|
function get_full_url() {
|
|
|
|
return `${url_base}?v=${wynn_version_id.toString()}${location.hash}`
|
|
|
|
}
|
|
|
|
|
2022-06-19 20:44:02 +00:00
|
|
|
function copyBuild() {
|
2022-12-16 10:48:00 +00:00
|
|
|
copyTextToClipboard(get_full_url());
|
2022-06-19 20:44:02 +00:00
|
|
|
document.getElementById("copy-button").textContent = "Copied!";
|
2022-06-19 07:42:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function shareBuild(build) {
|
|
|
|
if (build) {
|
2022-12-16 10:48:00 +00:00
|
|
|
let text = get_full_url()+"\n"+
|
2022-06-19 07:42:49 +00:00
|
|
|
"WynnBuilder build:\n"+
|
2022-06-22 05:39:10 +00:00
|
|
|
"> "+build.items[0].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[1].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[2].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[3].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[4].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[5].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[6].statMap.get("displayName")+"\n"+
|
|
|
|
"> "+build.items[7].statMap.get("displayName")+"\n"+
|
2022-07-29 04:27:49 +00:00
|
|
|
"> "+build.items[15].statMap.get("displayName")+" ["+build_powders[4].map(x => powderNames.get(x)).join("")+"]\n";
|
|
|
|
for (let tomeslots = 8; tomeslots < 15; tomeslots++) {
|
|
|
|
if (!build.items[tomeslots].statMap.has('NONE')) {
|
|
|
|
text += ">"+' (Has Tomes)' ;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-06-19 07:42:49 +00:00
|
|
|
copyTextToClipboard(text);
|
|
|
|
document.getElementById("share-button").textContent = "Copied!";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-30 15:03:41 +00:00
|
|
|
/**
|
|
|
|
* Ability tree encode and decode functions
|
|
|
|
*
|
|
|
|
* Based on a traversal, basically only uses bits to represent the nodes that are on (and "dark" outgoing edges).
|
|
|
|
* credit: SockMower
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return: BitVector
|
|
|
|
*/
|
|
|
|
function encode_atree(atree, atree_state) {
|
|
|
|
let ret_vec = new BitVector(0, 0);
|
|
|
|
|
|
|
|
function traverse(head, atree_state, visited, ret) {
|
|
|
|
for (const child of head.children) {
|
|
|
|
if (visited.has(child.ability.id)) { continue; }
|
|
|
|
visited.set(child.ability.id, true);
|
|
|
|
if (atree_state.get(child.ability.id).active) {
|
|
|
|
ret.append(1, 1);
|
|
|
|
traverse(child, atree_state, visited, ret);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ret.append(0, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
traverse(atree[0], atree_state, new Map(), ret_vec);
|
|
|
|
return ret_vec;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return: List of active nodes
|
|
|
|
*/
|
|
|
|
function decode_atree(atree, bits) {
|
|
|
|
let i = 0;
|
|
|
|
let ret = [];
|
|
|
|
ret.push(atree[0]);
|
|
|
|
function traverse(head, visited, ret) {
|
|
|
|
for (const child of head.children) {
|
|
|
|
if (visited.has(child.ability.id)) { continue; }
|
|
|
|
visited.set(child.ability.id, true);
|
|
|
|
if (bits.read_bit(i)) {
|
|
|
|
i += 1;
|
|
|
|
ret.push(child);
|
|
|
|
traverse(child, visited, ret);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
traverse(atree[0], new Map(), ret);
|
|
|
|
return ret;
|
|
|
|
}
|