Refactor wynnatlas (normal) backend to use advanced search mechanics
This commit is contained in:
parent
ac87cf5112
commit
64e108bcb1
6 changed files with 82 additions and 128 deletions
|
@ -129,7 +129,8 @@
|
||||||
<script type="text/javascript" src="/js/damage_calc.js"></script>
|
<script type="text/javascript" src="/js/damage_calc.js"></script>
|
||||||
<script type="text/javascript" src="/js/display_constants.js"></script>
|
<script type="text/javascript" src="/js/display_constants.js"></script>
|
||||||
<script type="text/javascript" src="/js/display.js"></script>
|
<script type="text/javascript" src="/js/display.js"></script>
|
||||||
<script type="text/javascript" src="/js/query.js"></script>
|
<script type="text/javascript" src="/js/query_2.js"></script>
|
||||||
|
<script type="text/javascript" src="/js/expr_parser.js"></script>
|
||||||
<script type="text/javascript" src="/js/load.js"></script>
|
<script type="text/javascript" src="/js/load.js"></script>
|
||||||
<script type="text/javascript" src="/js/items.js"></script>
|
<script type="text/javascript" src="/js/items.js"></script>
|
||||||
</body>
|
</body>
|
||||||
|
|
|
@ -468,7 +468,6 @@ function displayExpandedItem(item, parent_id){
|
||||||
p_elem.textContent = "Set: " + item.get(id).toString();
|
p_elem.textContent = "Set: " + item.get(id).toString();
|
||||||
active_elem.appendChild(p_elem);
|
active_elem.appendChild(p_elem);
|
||||||
} else if (id === "majorIds") {
|
} else if (id === "majorIds") {
|
||||||
console.log(item.get(id));
|
|
||||||
for (let majorID of item.get(id)) {
|
for (let majorID of item.get(id)) {
|
||||||
let p_elem = document.createElement("p");
|
let p_elem = document.createElement("p");
|
||||||
p_elem.classList.add("itemp");
|
p_elem.classList.add("itemp");
|
||||||
|
|
|
@ -75,7 +75,7 @@ const ExprParser = (function() {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// parse a string literal
|
// parse a string literal
|
||||||
if ((m = /^"([^"]+)"/.exec(exprStr.substring(col))) !== null) { // with double-quotes
|
if ((m = /^"([^"]*)"/.exec(exprStr.substring(col))) !== null) { // with double-quotes
|
||||||
tokens.push({ type: 'sLit', value: m[1] });
|
tokens.push({ type: 'sLit', value: m[1] });
|
||||||
col += m[0].length;
|
col += m[0].length;
|
||||||
continue;
|
continue;
|
||||||
|
|
156
js/items.js
156
js/items.js
|
@ -1,5 +1,3 @@
|
||||||
|
|
||||||
|
|
||||||
const translate_mappings = {
|
const translate_mappings = {
|
||||||
//"Name": "name",
|
//"Name": "name",
|
||||||
//"Display Name": "displayName",
|
//"Display Name": "displayName",
|
||||||
|
@ -93,15 +91,15 @@ const translate_mappings = {
|
||||||
};
|
};
|
||||||
|
|
||||||
const special_mappings = {
|
const special_mappings = {
|
||||||
"Sum (skill points)": new SumQuery(["str", "dex", "int", "def", "agi"]),
|
"Sum (skill points)": "s:str+dex+int+def+agi",
|
||||||
"Sum (Mana Sustain)": new SumQuery(["mr", "ms"]),
|
"Sum (Mana Sustain)": "s:mr+ms",
|
||||||
"Sum (Life Sustain)": new SumQuery(["hpr", "ls"]),
|
"Sum (Life Sustain)": "s:hpr+ls",
|
||||||
"Sum (Health + Health Bonus)": new SumQuery(["hp", "hpBonus"]),
|
"Sum (Health + Health Bonus)": "s:hp+hpBonus",
|
||||||
"No Strength Req": new NegateQuery("strReq"),
|
"No Strength Req": "f:strReq==0",
|
||||||
"No Dexterity Req": new NegateQuery("dexReq"),
|
"No Dexterity Req": "f:dexReq==0",
|
||||||
"No Intelligence Req": new NegateQuery("intReq"),
|
"No Intelligence Req": "f:intReq==0",
|
||||||
"No Agility Req": new NegateQuery("agiReq"),
|
"No Agility Req": "f:agiReq==0",
|
||||||
"No Defense Req": new NegateQuery("defReq"),
|
"No Defense Req": "f:defReq==0",
|
||||||
};
|
};
|
||||||
|
|
||||||
let itemFilters = document.getElementById("filter-items");
|
let itemFilters = document.getElementById("filter-items");
|
||||||
|
@ -122,10 +120,10 @@ function applyQuery(items, query) {
|
||||||
return items.filter(query.filter, query).sort(query.compare);
|
return items.filter(query.filter, query).sort(query.compare);
|
||||||
}
|
}
|
||||||
|
|
||||||
function displayItems(items_copy) {
|
function displayItems(results) {
|
||||||
let items_parent = document.getElementById("main");
|
let items_parent = document.getElementById("main");
|
||||||
for (let i in items_copy) {
|
for (let i in results) {
|
||||||
let item = items_copy[i];
|
let item = results[i].itemExp;
|
||||||
let box = document.createElement("div");
|
let box = document.createElement("div");
|
||||||
box.classList.add("box");
|
box.classList.add("box");
|
||||||
box.id = "item"+i;
|
box.id = "item"+i;
|
||||||
|
@ -134,110 +132,42 @@ function displayItems(items_copy) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let items_expanded;
|
let searchDb;
|
||||||
|
|
||||||
// updates the current search state from the search query input boxes
|
|
||||||
function updateSearch() {
|
|
||||||
// compile query expressions, aborting if nothing has changed or either fails to compile
|
|
||||||
const changed = searchFilterField.compile() | searchSortField.compile();
|
|
||||||
if (!changed || searchFilterField.output === null || searchSortField.output === null) return;
|
|
||||||
|
|
||||||
// update url query string
|
|
||||||
const newUrl = `${window.location.protocol}//${window.location.host}${window.location.pathname}`
|
|
||||||
+ `?f=${encodeURIComponent(searchFilterField.value)}&s=${encodeURIComponent(searchSortField.value)}`;
|
|
||||||
window.history.pushState({ path: newUrl }, '', newUrl);
|
|
||||||
|
|
||||||
// hide old search results
|
|
||||||
itemListFooter.innerText = '';
|
|
||||||
for (const itemEntry of itemEntries) itemEntry.classList.remove('visible');
|
|
||||||
|
|
||||||
// index and sort search results
|
|
||||||
const searchResults = [];
|
|
||||||
try {
|
|
||||||
for (let i = 0; i < searchDb.length; i++) {
|
|
||||||
const item = searchDb[i][0], itemExp = searchDb[i][1];
|
|
||||||
if (checkBool(searchFilterField.output.resolve(item, itemExp))) {
|
|
||||||
searchResults.push({ item, itemExp, sortKeys: searchSortField.output.resolve(item, itemExp) });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
searchFilterField.errorText.innerText = e.message;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (searchResults.length === 0) {
|
|
||||||
itemListFooter.innerText = 'No results!';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
searchResults.sort((a, b) => {
|
|
||||||
try {
|
|
||||||
return compareLexico(a.item, a.sortKeys, b.item, b.sortKeys);
|
|
||||||
} catch (e) {
|
|
||||||
console.log(a.item, b.item);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (e) {
|
|
||||||
searchSortField.errorText.innerText = e.message;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// display search results
|
|
||||||
const searchMax = Math.min(searchResults.length, ITEM_LIST_SIZE);
|
|
||||||
for (let i = 0; i < searchMax; i++) {
|
|
||||||
const result = searchResults[i];
|
|
||||||
itemEntries[i].classList.add('visible');
|
|
||||||
displayExpandedItem(result.itemExp, `item-entry-${i}`);
|
|
||||||
if (result.sortKeys.length > 0) {
|
|
||||||
const sortKeyListContainer = document.createElement('div');
|
|
||||||
sortKeyListContainer.classList.add('itemleft');
|
|
||||||
const sortKeyList = document.createElement('ul');
|
|
||||||
sortKeyList.classList.add('item-entry-sort-key', 'itemp', 'T0');
|
|
||||||
sortKeyListContainer.append(sortKeyList);
|
|
||||||
for (let j = 0; j < result.sortKeys.length; j++) {
|
|
||||||
const sortKeyElem = document.createElement('li');
|
|
||||||
sortKeyElem.innerText = stringify(result.sortKeys[j]);
|
|
||||||
sortKeyList.append(sortKeyElem);
|
|
||||||
}
|
|
||||||
itemEntries[i].append(sortKeyListContainer);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (searchMax < searchResults.length) {
|
|
||||||
itemListFooter.innerText = `${searchResults.length - searchMax} more...`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function doItemSearch() {
|
function doItemSearch() {
|
||||||
window.scrollTo(0, 0);
|
window.scrollTo(0, 0);
|
||||||
let queries = [];
|
let queries = [];
|
||||||
queries.push(new NameQuery(document.getElementById("name-choice").value.trim()));
|
queries.push('f:name?="'+document.getElementById("name-choice").value.trim()+'"');
|
||||||
|
|
||||||
let categoryOrType = document.getElementById("category-choice").value;
|
let categoryOrType = document.getElementById("category-choice").value;
|
||||||
if (itemTypes.includes(categoryOrType)) {
|
if (itemTypes.includes(categoryOrType)) {
|
||||||
queries.push(new IdMatchQuery("type", categoryOrType));
|
queries.push('f:type="'+categoryOrType+'"');
|
||||||
}
|
}
|
||||||
else if (itemCategories.includes(categoryOrType)) {
|
else if (itemCategories.includes(categoryOrType)) {
|
||||||
queries.push(new IdMatchQuery("category", categoryOrType));
|
queries.push('f:cat="'+categoryOrType+'"');
|
||||||
}
|
}
|
||||||
|
|
||||||
let rarity = document.getElementById("rarity-choice").value;
|
let rarity = document.getElementById("rarity-choice").value;
|
||||||
if (rarity) {
|
if (rarity) {
|
||||||
if (rarity === "ANY") {
|
if (rarity === "ANY") {
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (rarity === "Sane") {
|
||||||
|
queries.push('f:tiername!="mythic"');
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
queries.push(new IdMatchQuery("tier", rarity));
|
queries.push('f:tiername="'+rarity+'"');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let level_dat = document.getElementById("level-choice").value.split("-");
|
let level_dat = document.getElementById("level-choice").value.split("-");
|
||||||
queries.push(new LevelRangeQuery(parseInt(level_dat[0]), parseInt(level_dat[1])));
|
queries.push('f:(lvl>='+parseInt(level_dat[0])+'&lvl<='+parseInt(level_dat[1])+')');
|
||||||
|
|
||||||
for (let i = 1; i <= 4; ++i) {
|
for (let i = 1; i <= 4; ++i) {
|
||||||
let raw_dat = document.getElementById("filter"+i+"-choice").value;
|
let raw_dat = document.getElementById("filter"+i+"-choice").value;
|
||||||
let filter_dat = translate_mappings[raw_dat];
|
let filter_dat = translate_mappings[raw_dat];
|
||||||
if (filter_dat !== undefined) {
|
if (filter_dat !== undefined) {
|
||||||
queries.push(new IdQuery(filter_dat));
|
queries.push("s:"+filter_dat);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
filter_dat = special_mappings[raw_dat];
|
filter_dat = special_mappings[raw_dat];
|
||||||
|
@ -247,21 +177,45 @@ function doItemSearch() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let items_copy = items_expanded.slice();
|
|
||||||
document.getElementById("main").textContent = "";
|
document.getElementById("main").textContent = "";
|
||||||
|
let filterQuery = "true";
|
||||||
|
let sortQueries = [];
|
||||||
|
console.log(queries);
|
||||||
for (const query of queries) {
|
for (const query of queries) {
|
||||||
console.log(items_copy.length);
|
if (query.startsWith("s:")) {
|
||||||
console.log(query);
|
sortQueries.push(query.slice(2));
|
||||||
console.log(query.filter);
|
}
|
||||||
items_copy = applyQuery(items_copy, query);
|
else if (query.startsWith("f:")) {
|
||||||
console.log(items_copy.length);
|
filterQuery = filterQuery + "&" + query.slice(2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
document.getElementById("summary").textContent = items_copy.length + " results."
|
console.log(filterQuery);
|
||||||
displayItems(items_copy);
|
console.log(sortQueries);
|
||||||
|
let results = [];
|
||||||
|
try {
|
||||||
|
const filterExpr = exprParser.parse(filterQuery);
|
||||||
|
const sortExprs = sortQueries.map(q => exprParser.parse(q));
|
||||||
|
for (let i = 0; i < searchDb.length; ++i) {
|
||||||
|
const item = searchDb[i][0];
|
||||||
|
const itemExp = searchDb[i][1];
|
||||||
|
if (checkBool(filterExpr.resolve(item, itemExp))) {
|
||||||
|
results.push({ item, itemExp, sortKeys: sortExprs.map(e => e.resolve(item, itemExp)) });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
results.sort((a, b) => {
|
||||||
|
return compareLexico(a.item, a.sortKeys, b.item, b.sortKeys);
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
document.getElementById("summary").textContent = e.message;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
document.getElementById("summary").textContent = results.length + " results."
|
||||||
|
displayItems(results);
|
||||||
}
|
}
|
||||||
|
|
||||||
function init_items() {
|
function init_items() {
|
||||||
items_expanded = items.filter( (i) => !("remapID" in i) ).map( (i) => expandItem(i, []) );
|
searchDb = items.filter( i => ! i.remapID ).map( i => [i, expandItem(i, [])] );
|
||||||
|
exprParser = new ExprParser(itemQueryProps, itemQueryFuncs);
|
||||||
}
|
}
|
||||||
|
|
||||||
load_init(init_items);
|
load_init(init_items);
|
||||||
|
|
|
@ -230,30 +230,6 @@ class ExprField {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function compareLexico(ia, keysA, ib, keysB) {
|
|
||||||
for (let i = 0; i < keysA.length; i++) { // assuming keysA and keysB are the same length
|
|
||||||
let aKey = keysA[i], bKey = keysB[i];
|
|
||||||
if (typeof aKey !== typeof bKey) throw new Error(`Incomparable types ${typeof aKey} and ${typeof bKey}`); // can this even happen?
|
|
||||||
switch (typeof aKey) {
|
|
||||||
case 'string':
|
|
||||||
aKey = aKey.toLowerCase();
|
|
||||||
bKey = bKey.toLowerCase();
|
|
||||||
if (aKey < bKey) return -1;
|
|
||||||
if (aKey > bKey) return 1;
|
|
||||||
break;
|
|
||||||
case 'number': // sort numeric stuff in reverse order
|
|
||||||
aKey = isNaN(aKey) ? 0 : aKey;
|
|
||||||
bKey = isNaN(bKey) ? 0 : bKey;
|
|
||||||
if (aKey < bKey) return 1;
|
|
||||||
if (aKey > bKey) return -1;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new Error(`Incomparable type ${typeof aKey}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ib.lvl - ia.lvl;
|
|
||||||
}
|
|
||||||
|
|
||||||
function stringify(v) {
|
function stringify(v) {
|
||||||
return typeof v === 'number' ? (Math.round(v * 100) / 100).toString() : v;
|
return typeof v === 'number' ? (Math.round(v * 100) / 100).toString() : v;
|
||||||
}
|
}
|
||||||
|
|
|
@ -523,3 +523,27 @@ class PropTerm extends Term {
|
||||||
return this.prop.resolve(item, itemExt);
|
return this.prop.resolve(item, itemExt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function compareLexico(ia, keysA, ib, keysB) {
|
||||||
|
for (let i = 0; i < keysA.length; i++) { // assuming keysA and keysB are the same length
|
||||||
|
let aKey = keysA[i], bKey = keysB[i];
|
||||||
|
if (typeof aKey !== typeof bKey) throw new Error(`Incomparable types ${typeof aKey} and ${typeof bKey}`); // can this even happen?
|
||||||
|
switch (typeof aKey) {
|
||||||
|
case 'string':
|
||||||
|
aKey = aKey.toLowerCase();
|
||||||
|
bKey = bKey.toLowerCase();
|
||||||
|
if (aKey < bKey) return -1;
|
||||||
|
if (aKey > bKey) return 1;
|
||||||
|
break;
|
||||||
|
case 'number': // sort numeric stuff in reverse order
|
||||||
|
aKey = isNaN(aKey) ? 0 : aKey;
|
||||||
|
bKey = isNaN(bKey) ? 0 : bKey;
|
||||||
|
if (aKey < bKey) return 1;
|
||||||
|
if (aKey > bKey) return -1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error(`Incomparable type ${typeof aKey}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ib.lvl - ia.lvl;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue