mirror of
https://github.com/hrfee/jfa-go.git
synced 2024-12-22 17:10:10 +00:00
accounts: modularize search
now part of ts/modules/search.ts, UI of the activity page is gonna be very similar so it made sense to.
This commit is contained in:
parent
3cad30a8e5
commit
4fa0630aef
@ -3,6 +3,7 @@ import { templateEmail } from "../modules/settings.js";
|
|||||||
import { Marked } from "@ts-stack/markdown";
|
import { Marked } from "@ts-stack/markdown";
|
||||||
import { stripMarkdown } from "../modules/stripmd.js";
|
import { stripMarkdown } from "../modules/stripmd.js";
|
||||||
import { DiscordUser, newDiscordSearch } from "../modules/discord.js";
|
import { DiscordUser, newDiscordSearch } from "../modules/discord.js";
|
||||||
|
import { Search, SearchConfiguration, QueryType, SearchableItem } from "../modules/search.js";
|
||||||
const dateParser = require("any-date-parser");
|
const dateParser = require("any-date-parser");
|
||||||
|
|
||||||
interface User {
|
interface User {
|
||||||
@ -39,7 +40,7 @@ interface announcementTemplate {
|
|||||||
|
|
||||||
var addDiscord: (passData: string) => void;
|
var addDiscord: (passData: string) => void;
|
||||||
|
|
||||||
class user implements User {
|
class user implements User, SearchableItem {
|
||||||
private _id = "";
|
private _id = "";
|
||||||
private _row: HTMLTableRowElement;
|
private _row: HTMLTableRowElement;
|
||||||
private _check: HTMLInputElement;
|
private _check: HTMLInputElement;
|
||||||
@ -780,13 +781,13 @@ export class accountsList {
|
|||||||
private _userSelect = document.getElementById("modify-user-users") as HTMLSelectElement;
|
private _userSelect = document.getElementById("modify-user-users") as HTMLSelectElement;
|
||||||
private _referralsProfileSelect = document.getElementById("enable-referrals-user-profiles") as HTMLSelectElement;
|
private _referralsProfileSelect = document.getElementById("enable-referrals-user-profiles") as HTMLSelectElement;
|
||||||
private _referralsInviteSelect = document.getElementById("enable-referrals-user-invites") as HTMLSelectElement;
|
private _referralsInviteSelect = document.getElementById("enable-referrals-user-invites") as HTMLSelectElement;
|
||||||
private _search = document.getElementById("accounts-search") as HTMLInputElement;
|
private _searchBox = document.getElementById("accounts-search") as HTMLInputElement;
|
||||||
|
private _search: Search;
|
||||||
|
|
||||||
private _selectAll = document.getElementById("accounts-select-all") as HTMLInputElement;
|
private _selectAll = document.getElementById("accounts-select-all") as HTMLInputElement;
|
||||||
private _users: { [id: string]: user };
|
private _users: { [id: string]: user };
|
||||||
private _ordering: string[] = [];
|
private _ordering: string[] = [];
|
||||||
private _checkCount: number = 0;
|
private _checkCount: number = 0;
|
||||||
private _inSearch = false;
|
|
||||||
// Whether the enable/disable button should enable or not.
|
// Whether the enable/disable button should enable or not.
|
||||||
private _shouldEnable = false;
|
private _shouldEnable = false;
|
||||||
|
|
||||||
@ -836,7 +837,7 @@ export class accountsList {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _queries: { [field: string]: { name: string, getter: string, bool: boolean, string: boolean, date: boolean, dependsOnTableHeader?: string, show?: boolean }} = {
|
private _queries: { [field: string]: QueryType } = {
|
||||||
"id": {
|
"id": {
|
||||||
// We don't use a translation here to circumvent the name substitution feature.
|
// We don't use a translation here to circumvent the name substitution feature.
|
||||||
name: "Jellyfin/Emby ID",
|
name: "Jellyfin/Emby ID",
|
||||||
@ -887,7 +888,7 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: false,
|
string: false,
|
||||||
date: false,
|
date: false,
|
||||||
dependsOnTableHeader: "accounts-header-access-jfa"
|
dependsOnElement: ".accounts-header-access-jfa"
|
||||||
},
|
},
|
||||||
"email": {
|
"email": {
|
||||||
name: window.lang.strings("emailAddress"),
|
name: window.lang.strings("emailAddress"),
|
||||||
@ -895,7 +896,7 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: true,
|
string: true,
|
||||||
date: false,
|
date: false,
|
||||||
dependsOnTableHeader: "accounts-header-email"
|
dependsOnElement: ".accounts-header-email"
|
||||||
},
|
},
|
||||||
"telegram": {
|
"telegram": {
|
||||||
name: "Telegram",
|
name: "Telegram",
|
||||||
@ -903,7 +904,7 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: true,
|
string: true,
|
||||||
date: false,
|
date: false,
|
||||||
dependsOnTableHeader: "accounts-header-telegram"
|
dependsOnElement: ".accounts-header-telegram"
|
||||||
},
|
},
|
||||||
"matrix": {
|
"matrix": {
|
||||||
name: "Matrix",
|
name: "Matrix",
|
||||||
@ -911,7 +912,7 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: true,
|
string: true,
|
||||||
date: false,
|
date: false,
|
||||||
dependsOnTableHeader: "accounts-header-matrix"
|
dependsOnElement: ".accounts-header-matrix"
|
||||||
},
|
},
|
||||||
"discord": {
|
"discord": {
|
||||||
name: "Discord",
|
name: "Discord",
|
||||||
@ -919,7 +920,7 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: true,
|
string: true,
|
||||||
date: false,
|
date: false,
|
||||||
dependsOnTableHeader: "accounts-header-discord"
|
dependsOnElement: ".accounts-header-discord"
|
||||||
},
|
},
|
||||||
"expiry": {
|
"expiry": {
|
||||||
name: window.lang.strings("expiry"),
|
name: window.lang.strings("expiry"),
|
||||||
@ -927,7 +928,7 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: false,
|
string: false,
|
||||||
date: true,
|
date: true,
|
||||||
dependsOnTableHeader: "accounts-header-expiry"
|
dependsOnElement: ".accounts-header-expiry"
|
||||||
},
|
},
|
||||||
"last-active": {
|
"last-active": {
|
||||||
name: window.lang.strings("lastActiveTime"),
|
name: window.lang.strings("lastActiveTime"),
|
||||||
@ -942,229 +943,12 @@ export class accountsList {
|
|||||||
bool: true,
|
bool: true,
|
||||||
string: false,
|
string: false,
|
||||||
date: false,
|
date: false,
|
||||||
dependsOnTableHeader: "accounts-header-referrals"
|
dependsOnElement: ".accounts-header-referrals"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _notFoundPanel: HTMLElement = document.getElementById("accounts-not-found");
|
private _notFoundPanel: HTMLElement = document.getElementById("accounts-not-found");
|
||||||
|
|
||||||
search = (query: String): string[] => {
|
|
||||||
console.log(this._queries);
|
|
||||||
this._filterArea.textContent = "";
|
|
||||||
|
|
||||||
query = query.toLowerCase();
|
|
||||||
let result: string[] = [...this._ordering];
|
|
||||||
// console.log("initial:", result);
|
|
||||||
|
|
||||||
// const words = query.split(" ");
|
|
||||||
let words: string[] = [];
|
|
||||||
|
|
||||||
let quoteSymbol = ``;
|
|
||||||
let queryStart = -1;
|
|
||||||
let lastQuote = -1;
|
|
||||||
for (let i = 0; i < query.length; i++) {
|
|
||||||
if (queryStart == -1 && query[i] != " " && query[i] != `"` && query[i] != `'`) {
|
|
||||||
queryStart = i;
|
|
||||||
}
|
|
||||||
if ((query[i] == `"` || query[i] == `'`) && (quoteSymbol == `` || query[i] == quoteSymbol)) {
|
|
||||||
if (lastQuote != -1) {
|
|
||||||
lastQuote = -1;
|
|
||||||
quoteSymbol = ``;
|
|
||||||
} else {
|
|
||||||
lastQuote = i;
|
|
||||||
quoteSymbol = query[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query[i] == " " || i == query.length-1) {
|
|
||||||
if (lastQuote != -1) {
|
|
||||||
continue;
|
|
||||||
} else {
|
|
||||||
let end = i+1;
|
|
||||||
if (query[i] == " ") {
|
|
||||||
end = i;
|
|
||||||
while (i+1 < query.length && query[i+1] == " ") {
|
|
||||||
i += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
words.push(query.substring(queryStart, end).replace(/['"]/g, ""));
|
|
||||||
console.log("pushed", words);
|
|
||||||
queryStart = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
query = "";
|
|
||||||
for (let word of words) {
|
|
||||||
if (!word.includes(":")) {
|
|
||||||
let cachedResult = [...result];
|
|
||||||
for (let id of cachedResult) {
|
|
||||||
const u = this._users[id];
|
|
||||||
if (!u.matchesSearch(word)) {
|
|
||||||
result.splice(result.indexOf(id), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const split = [word.substring(0, word.indexOf(":")), word.substring(word.indexOf(":")+1)];
|
|
||||||
|
|
||||||
if (!(split[0] in this._queries)) continue;
|
|
||||||
|
|
||||||
const queryFormat = this._queries[split[0]];
|
|
||||||
|
|
||||||
if (queryFormat.bool) {
|
|
||||||
let isBool = false;
|
|
||||||
let boolState = false;
|
|
||||||
if (split[1] == "true" || split[1] == "yes" || split[1] == "t" || split[1] == "y") {
|
|
||||||
isBool = true;
|
|
||||||
boolState = true;
|
|
||||||
} else if (split[1] == "false" || split[1] == "no" || split[1] == "f" || split[1] == "n") {
|
|
||||||
isBool = true;
|
|
||||||
boolState = false;
|
|
||||||
}
|
|
||||||
if (isBool) {
|
|
||||||
const filterCard = document.createElement("span");
|
|
||||||
filterCard.ariaLabel = window.lang.strings("clickToRemoveFilter");
|
|
||||||
filterCard.classList.add("button", "~" + (boolState ? "positive" : "critical"), "@high", "center", "mx-2", "h-full");
|
|
||||||
filterCard.innerHTML = `
|
|
||||||
<span class="font-bold mr-2">${queryFormat.name}</span>
|
|
||||||
<i class="text-2xl ri-${boolState? "checkbox" : "close"}-circle-fill"></i>
|
|
||||||
`;
|
|
||||||
|
|
||||||
filterCard.addEventListener("click", () => {
|
|
||||||
for (let quote of [`"`, `'`, ``]) {
|
|
||||||
this._search.value = this._search.value.replace(split[0] + ":" + quote + split[1] + quote, "");
|
|
||||||
}
|
|
||||||
this._search.oninput((null as Event));
|
|
||||||
})
|
|
||||||
|
|
||||||
this._filterArea.appendChild(filterCard);
|
|
||||||
|
|
||||||
// console.log("is bool, state", boolState);
|
|
||||||
// So removing elements doesn't affect us
|
|
||||||
let cachedResult = [...result];
|
|
||||||
for (let id of cachedResult) {
|
|
||||||
const u = this._users[id];
|
|
||||||
const value = Object.getOwnPropertyDescriptor(user.prototype, queryFormat.getter).get.call(u);
|
|
||||||
// console.log("got", queryFormat.getter + ":", value);
|
|
||||||
// Remove from result if not matching query
|
|
||||||
if (!((value && boolState) || (!value && !boolState))) {
|
|
||||||
// console.log("not matching, result is", result);
|
|
||||||
result.splice(result.indexOf(id), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (queryFormat.string) {
|
|
||||||
const filterCard = document.createElement("span");
|
|
||||||
filterCard.ariaLabel = window.lang.strings("clickToRemoveFilter");
|
|
||||||
filterCard.classList.add("button", "~neutral", "@low", "center", "mx-2", "h-full");
|
|
||||||
filterCard.innerHTML = `
|
|
||||||
<span class="font-bold mr-2">${queryFormat.name}:</span> "${split[1]}"
|
|
||||||
`;
|
|
||||||
|
|
||||||
filterCard.addEventListener("click", () => {
|
|
||||||
for (let quote of [`"`, `'`, ``]) {
|
|
||||||
let regex = new RegExp(split[0] + ":" + quote + split[1] + quote, "ig");
|
|
||||||
this._search.value = this._search.value.replace(regex, "");
|
|
||||||
}
|
|
||||||
this._search.oninput((null as Event));
|
|
||||||
})
|
|
||||||
|
|
||||||
this._filterArea.appendChild(filterCard);
|
|
||||||
|
|
||||||
let cachedResult = [...result];
|
|
||||||
for (let id of cachedResult) {
|
|
||||||
const u = this._users[id];
|
|
||||||
const value = Object.getOwnPropertyDescriptor(user.prototype, queryFormat.getter).get.call(u);
|
|
||||||
if (!(value.includes(split[1]))) {
|
|
||||||
result.splice(result.indexOf(id), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (queryFormat.date) {
|
|
||||||
// -1 = Before, 0 = On, 1 = After, 2 = No symbol, assume 0
|
|
||||||
let compareType = (split[1][0] == ">") ? 1 : ((split[1][0] == "<") ? -1 : ((split[1][0] == "=") ? 0 : 2));
|
|
||||||
let unmodifiedValue = split[1];
|
|
||||||
if (compareType != 2) {
|
|
||||||
split[1] = split[1].substring(1);
|
|
||||||
}
|
|
||||||
if (compareType == 2) compareType = 0;
|
|
||||||
|
|
||||||
let attempt: { year?: number, month?: number, day?: number, hour?: number, minute?: number } = dateParser.attempt(split[1]);
|
|
||||||
// Month in Date objects is 0-based, so make our parsed date that way too
|
|
||||||
if ("month" in attempt) attempt.month -= 1;
|
|
||||||
|
|
||||||
let date: Date = (Date as any).fromString(split[1]) as Date;
|
|
||||||
console.log("Read", attempt, "and", date);
|
|
||||||
if ("invalid" in (date as any)) continue;
|
|
||||||
|
|
||||||
const filterCard = document.createElement("span");
|
|
||||||
filterCard.ariaLabel = window.lang.strings("clickToRemoveFilter");
|
|
||||||
filterCard.classList.add("button", "~neutral", "@low", "center", "m-2", "h-full");
|
|
||||||
filterCard.innerHTML = `
|
|
||||||
<span class="font-bold mr-2">${queryFormat.name}:</span> ${(compareType == 1) ? window.lang.strings("after")+" " : ((compareType == -1) ? window.lang.strings("before")+" " : "")}${split[1]}
|
|
||||||
`;
|
|
||||||
|
|
||||||
filterCard.addEventListener("click", () => {
|
|
||||||
for (let quote of [`"`, `'`, ``]) {
|
|
||||||
let regex = new RegExp(split[0] + ":" + quote + unmodifiedValue + quote, "ig");
|
|
||||||
this._search.value = this._search.value.replace(regex, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
this._search.oninput((null as Event));
|
|
||||||
})
|
|
||||||
|
|
||||||
this._filterArea.appendChild(filterCard);
|
|
||||||
|
|
||||||
let cachedResult = [...result];
|
|
||||||
for (let id of cachedResult) {
|
|
||||||
const u = this._users[id];
|
|
||||||
const unixValue = Object.getOwnPropertyDescriptor(user.prototype, queryFormat.getter).get.call(u);
|
|
||||||
if (unixValue == 0) {
|
|
||||||
result.splice(result.indexOf(id), 1);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
let value = new Date(unixValue*1000);
|
|
||||||
|
|
||||||
const getterPairs: [string, () => number][] = [["year", Date.prototype.getFullYear], ["month", Date.prototype.getMonth], ["day", Date.prototype.getDate], ["hour", Date.prototype.getHours], ["minute", Date.prototype.getMinutes]];
|
|
||||||
|
|
||||||
// When doing > or < <time> with no date, we need to ignore the rest of the Date object
|
|
||||||
if (compareType != 0 && Object.keys(attempt).length == 2 && "hour" in attempt && "minute" in attempt) {
|
|
||||||
const temp = new Date(date.valueOf());
|
|
||||||
temp.setHours(value.getHours(), value.getMinutes());
|
|
||||||
value = temp;
|
|
||||||
console.log("just hours/minutes workaround, value set to", value);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
let match = true;
|
|
||||||
if (compareType == 0) {
|
|
||||||
for (let pair of getterPairs) {
|
|
||||||
if (pair[0] in attempt) {
|
|
||||||
if (compareType == 0 && attempt[pair[0]] != pair[1].call(value)) {
|
|
||||||
match = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (compareType == -1) {
|
|
||||||
match = (value < date);
|
|
||||||
} else if (compareType == 1) {
|
|
||||||
match = (value > date);
|
|
||||||
}
|
|
||||||
if (!match) {
|
|
||||||
result.splice(result.indexOf(id), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
get selectAll(): boolean { return this._selectAll.checked; }
|
get selectAll(): boolean { return this._selectAll.checked; }
|
||||||
set selectAll(state: boolean) {
|
set selectAll(state: boolean) {
|
||||||
let count = 0;
|
let count = 0;
|
||||||
@ -2014,34 +1798,22 @@ export class accountsList {
|
|||||||
this._deleteNotify.checked = false;
|
this._deleteNotify.checked = false;
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
const onchange = () => {
|
let conf: SearchConfiguration = {
|
||||||
const query = this._search.value;
|
filterArea: this._filterArea,
|
||||||
if (!query) {
|
sortingByButton: this._sortingByButton,
|
||||||
// this.setVisibility(this._ordering, true);
|
searchOptionsHeader: this._searchOptionsHeader,
|
||||||
this._inSearch = false;
|
notFoundPanel: this._notFoundPanel,
|
||||||
} else {
|
search: this._searchBox,
|
||||||
this._inSearch = true;
|
queries: this._queries,
|
||||||
// this.setVisibility(this.search(query), true);
|
setVisibility: this.setVisibility,
|
||||||
}
|
clearSearchButtonSelector: ".accounts-search-clear",
|
||||||
const results = this.search(query);
|
onSearchCallback: () => {
|
||||||
this.setVisibility(results, true);
|
|
||||||
this._checkCheckCount();
|
this._checkCheckCount();
|
||||||
this.showHideSearchOptionsHeader();
|
|
||||||
if (results.length == 0) {
|
|
||||||
this._notFoundPanel.classList.remove("unfocused");
|
|
||||||
} else {
|
|
||||||
this._notFoundPanel.classList.add("unfocused");
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
this._search.oninput = onchange;
|
this._search = new Search(conf);
|
||||||
|
this._search.items = this._users;
|
||||||
|
|
||||||
const clearSearchButtons = Array.from(document.getElementsByClassName("accounts-search-clear")) as Array<HTMLSpanElement>;
|
|
||||||
for (let b of clearSearchButtons) {
|
|
||||||
b.addEventListener("click", () => {
|
|
||||||
this._search.value = "";
|
|
||||||
onchange();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
this._announceTextarea.onkeyup = this.loadPreview;
|
this._announceTextarea.onkeyup = this.loadPreview;
|
||||||
addDiscord = newDiscordSearch(window.lang.strings("linkDiscord"), window.lang.strings("searchDiscordUser"), window.lang.strings("add"), (user: DiscordUser, id: string) => {
|
addDiscord = newDiscordSearch(window.lang.strings("linkDiscord"), window.lang.strings("searchDiscordUser"), window.lang.strings("add"), (user: DiscordUser, id: string) => {
|
||||||
@ -2088,15 +1860,16 @@ export class accountsList {
|
|||||||
|
|
||||||
document.addEventListener("header-click", (event: CustomEvent) => {
|
document.addEventListener("header-click", (event: CustomEvent) => {
|
||||||
this._ordering = this._columns[event.detail].sort(this._users);
|
this._ordering = this._columns[event.detail].sort(this._users);
|
||||||
|
this._search.ordering = this._ordering;
|
||||||
this._activeSortColumn = event.detail;
|
this._activeSortColumn = event.detail;
|
||||||
this._sortingByButton.innerHTML = this._columns[event.detail].buttonContent;
|
this._sortingByButton.innerHTML = this._columns[event.detail].buttonContent;
|
||||||
this._sortingByButton.parentElement.classList.remove("hidden");
|
this._sortingByButton.parentElement.classList.remove("hidden");
|
||||||
// console.log("ordering by", event.detail, ": ", this._ordering);
|
// console.log("ordering by", event.detail, ": ", this._ordering);
|
||||||
if (!(this._inSearch)) {
|
if (!(this._search.inSearch)) {
|
||||||
this.setVisibility(this._ordering, true);
|
this.setVisibility(this._ordering, true);
|
||||||
this._notFoundPanel.classList.add("unfocused");
|
this._notFoundPanel.classList.add("unfocused");
|
||||||
} else {
|
} else {
|
||||||
const results = this.search(this._search.value);
|
const results = this._search.search(this._searchBox.value);
|
||||||
this.setVisibility(results, true);
|
this.setVisibility(results, true);
|
||||||
if (results.length == 0) {
|
if (results.length == 0) {
|
||||||
this._notFoundPanel.classList.remove("unfocused");
|
this._notFoundPanel.classList.remove("unfocused");
|
||||||
@ -2113,21 +1886,21 @@ export class accountsList {
|
|||||||
const filterList = document.getElementById("accounts-filter-list");
|
const filterList = document.getElementById("accounts-filter-list");
|
||||||
|
|
||||||
const fillInFilter = (name: string, value: string, offset?: number) => {
|
const fillInFilter = (name: string, value: string, offset?: number) => {
|
||||||
this._search.value = name + ":" + value + " " + this._search.value;
|
this._searchBox.value = name + ":" + value + " " + this._searchBox.value;
|
||||||
this._search.focus();
|
this._searchBox.focus();
|
||||||
let newPos = name.length + 1 + value.length;
|
let newPos = name.length + 1 + value.length;
|
||||||
if (typeof offset !== 'undefined')
|
if (typeof offset !== 'undefined')
|
||||||
newPos += offset;
|
newPos += offset;
|
||||||
this._search.setSelectionRange(newPos, newPos);
|
this._searchBox.setSelectionRange(newPos, newPos);
|
||||||
this._search.oninput(null as any);
|
this._searchBox.oninput(null as any);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Generate filter buttons
|
// Generate filter buttons
|
||||||
for (let queryName of Object.keys(this._queries)) {
|
for (let queryName of Object.keys(this._queries)) {
|
||||||
const query = this._queries[queryName];
|
const query = this._queries[queryName];
|
||||||
if ("show" in query && !query.show) continue;
|
if ("show" in query && !query.show) continue;
|
||||||
if ("dependsOnTableHeader" in query && query.dependsOnTableHeader) {
|
if ("dependsOnElement" in query && query.dependsOnElement) {
|
||||||
const el = document.querySelector("."+query.dependsOnTableHeader);
|
const el = document.querySelector(query.dependsOnElement);
|
||||||
if (el === null) continue;
|
if (el === null) continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2210,11 +1983,12 @@ export class accountsList {
|
|||||||
}
|
}
|
||||||
// console.log("reload, so sorting by", this._activeSortColumn);
|
// console.log("reload, so sorting by", this._activeSortColumn);
|
||||||
this._ordering = this._columns[this._activeSortColumn].sort(this._users);
|
this._ordering = this._columns[this._activeSortColumn].sort(this._users);
|
||||||
if (!(this._inSearch)) {
|
this._search.ordering = this._ordering;
|
||||||
|
if (!(this._search.inSearch)) {
|
||||||
this.setVisibility(this._ordering, true);
|
this.setVisibility(this._ordering, true);
|
||||||
this._notFoundPanel.classList.add("unfocused");
|
this._notFoundPanel.classList.add("unfocused");
|
||||||
} else {
|
} else {
|
||||||
const results = this.search(this._search.value);
|
const results = this._search.search(this._searchBox.value);
|
||||||
if (results.length == 0) {
|
if (results.length == 0) {
|
||||||
this._notFoundPanel.classList.remove("unfocused");
|
this._notFoundPanel.classList.remove("unfocused");
|
||||||
} else {
|
} else {
|
||||||
|
299
ts/modules/search.ts
Normal file
299
ts/modules/search.ts
Normal file
@ -0,0 +1,299 @@
|
|||||||
|
const dateParser = require("any-date-parser");
|
||||||
|
|
||||||
|
export interface QueryType {
|
||||||
|
name: string;
|
||||||
|
getter: string;
|
||||||
|
bool: boolean;
|
||||||
|
string: boolean;
|
||||||
|
date: boolean;
|
||||||
|
dependsOnElement?: string; // Format for querySelector
|
||||||
|
show?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SearchConfiguration {
|
||||||
|
filterArea: HTMLElement;
|
||||||
|
sortingByButton: HTMLButtonElement;
|
||||||
|
searchOptionsHeader: HTMLElement;
|
||||||
|
notFoundPanel: HTMLElement;
|
||||||
|
clearSearchButtonSelector: string;
|
||||||
|
search: HTMLInputElement;
|
||||||
|
queries: { [field: string]: QueryType };
|
||||||
|
setVisibility: (items: string[], visible: boolean) => void;
|
||||||
|
onSearchCallback: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SearchableItem {
|
||||||
|
matchesSearch: (query: string) => boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class Search {
|
||||||
|
private _c: SearchConfiguration;
|
||||||
|
private _ordering: string[] = [];
|
||||||
|
private _items: { [id: string]: SearchableItem };
|
||||||
|
inSearch: boolean;
|
||||||
|
|
||||||
|
search = (query: String): string[] => {
|
||||||
|
this._c.filterArea.textContent = "";
|
||||||
|
|
||||||
|
query = query.toLowerCase();
|
||||||
|
|
||||||
|
let result: string[] = [...this._ordering];
|
||||||
|
let words: string[] = [];
|
||||||
|
|
||||||
|
let quoteSymbol = ``;
|
||||||
|
let queryStart = -1;
|
||||||
|
let lastQuote = -1;
|
||||||
|
for (let i = 0; i < query.length; i++) {
|
||||||
|
if (queryStart == -1 && query[i] != " " && query[i] != `"` && query[i] != `'`) {
|
||||||
|
queryStart = i;
|
||||||
|
}
|
||||||
|
if ((query[i] == `"` || query[i] == `'`) && (quoteSymbol == `` || query[i] == quoteSymbol)) {
|
||||||
|
if (lastQuote != -1) {
|
||||||
|
lastQuote = -1;
|
||||||
|
quoteSymbol = ``;
|
||||||
|
} else {
|
||||||
|
lastQuote = i;
|
||||||
|
quoteSymbol = query[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (query[i] == " " || i == query.length-1) {
|
||||||
|
if (lastQuote != -1) {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
let end = i+1;
|
||||||
|
if (query[i] == " ") {
|
||||||
|
end = i;
|
||||||
|
while (i+1 < query.length && query[i+1] == " ") {
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
words.push(query.substring(queryStart, end).replace(/['"]/g, ""));
|
||||||
|
console.log("pushed", words);
|
||||||
|
queryStart = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
query = "";
|
||||||
|
for (let word of words) {
|
||||||
|
if (!word.includes(":")) {
|
||||||
|
let cachedResult = [...result];
|
||||||
|
for (let id of cachedResult) {
|
||||||
|
const u = this._items[id];
|
||||||
|
if (!u.matchesSearch(word)) {
|
||||||
|
result.splice(result.indexOf(id), 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const split = [word.substring(0, word.indexOf(":")), word.substring(word.indexOf(":")+1)];
|
||||||
|
|
||||||
|
if (!(split[0] in this._c.queries)) continue;
|
||||||
|
|
||||||
|
const queryFormat = this._c.queries[split[0]];
|
||||||
|
|
||||||
|
if (queryFormat.bool) {
|
||||||
|
let isBool = false;
|
||||||
|
let boolState = false;
|
||||||
|
if (split[1] == "true" || split[1] == "yes" || split[1] == "t" || split[1] == "y") {
|
||||||
|
isBool = true;
|
||||||
|
boolState = true;
|
||||||
|
} else if (split[1] == "false" || split[1] == "no" || split[1] == "f" || split[1] == "n") {
|
||||||
|
isBool = true;
|
||||||
|
boolState = false;
|
||||||
|
}
|
||||||
|
if (isBool) {
|
||||||
|
const filterCard = document.createElement("span");
|
||||||
|
filterCard.ariaLabel = window.lang.strings("clickToRemoveFilter");
|
||||||
|
filterCard.classList.add("button", "~" + (boolState ? "positive" : "critical"), "@high", "center", "mx-2", "h-full");
|
||||||
|
filterCard.innerHTML = `
|
||||||
|
<span class="font-bold mr-2">${queryFormat.name}</span>
|
||||||
|
<i class="text-2xl ri-${boolState? "checkbox" : "close"}-circle-fill"></i>
|
||||||
|
`;
|
||||||
|
|
||||||
|
filterCard.addEventListener("click", () => {
|
||||||
|
for (let quote of [`"`, `'`, ``]) {
|
||||||
|
this._c.search.value = this._c.search.value.replace(split[0] + ":" + quote + split[1] + quote, "");
|
||||||
|
}
|
||||||
|
this._c.search.oninput((null as Event));
|
||||||
|
})
|
||||||
|
|
||||||
|
this._c.filterArea.appendChild(filterCard);
|
||||||
|
|
||||||
|
// console.log("is bool, state", boolState);
|
||||||
|
// So removing elements doesn't affect us
|
||||||
|
let cachedResult = [...result];
|
||||||
|
for (let id of cachedResult) {
|
||||||
|
const u = this._items[id];
|
||||||
|
const value = Object.getOwnPropertyDescriptor(Object.getPrototypeOf(u), queryFormat.getter).get.call(u);
|
||||||
|
// console.log("got", queryFormat.getter + ":", value);
|
||||||
|
// Remove from result if not matching query
|
||||||
|
if (!((value && boolState) || (!value && !boolState))) {
|
||||||
|
// console.log("not matching, result is", result);
|
||||||
|
result.splice(result.indexOf(id), 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (queryFormat.string) {
|
||||||
|
const filterCard = document.createElement("span");
|
||||||
|
filterCard.ariaLabel = window.lang.strings("clickToRemoveFilter");
|
||||||
|
filterCard.classList.add("button", "~neutral", "@low", "center", "mx-2", "h-full");
|
||||||
|
filterCard.innerHTML = `
|
||||||
|
<span class="font-bold mr-2">${queryFormat.name}:</span> "${split[1]}"
|
||||||
|
`;
|
||||||
|
|
||||||
|
filterCard.addEventListener("click", () => {
|
||||||
|
for (let quote of [`"`, `'`, ``]) {
|
||||||
|
let regex = new RegExp(split[0] + ":" + quote + split[1] + quote, "ig");
|
||||||
|
this._c.search.value = this._c.search.value.replace(regex, "");
|
||||||
|
}
|
||||||
|
this._c.search.oninput((null as Event));
|
||||||
|
})
|
||||||
|
|
||||||
|
this._c.filterArea.appendChild(filterCard);
|
||||||
|
|
||||||
|
let cachedResult = [...result];
|
||||||
|
for (let id of cachedResult) {
|
||||||
|
const u = this._items[id];
|
||||||
|
const value = Object.getOwnPropertyDescriptor(Object.getPrototypeOf(u), queryFormat.getter).get.call(u);
|
||||||
|
if (!(value.includes(split[1]))) {
|
||||||
|
result.splice(result.indexOf(id), 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (queryFormat.date) {
|
||||||
|
// -1 = Before, 0 = On, 1 = After, 2 = No symbol, assume 0
|
||||||
|
let compareType = (split[1][0] == ">") ? 1 : ((split[1][0] == "<") ? -1 : ((split[1][0] == "=") ? 0 : 2));
|
||||||
|
let unmodifiedValue = split[1];
|
||||||
|
if (compareType != 2) {
|
||||||
|
split[1] = split[1].substring(1);
|
||||||
|
}
|
||||||
|
if (compareType == 2) compareType = 0;
|
||||||
|
|
||||||
|
let attempt: { year?: number, month?: number, day?: number, hour?: number, minute?: number } = dateParser.attempt(split[1]);
|
||||||
|
// Month in Date objects is 0-based, so make our parsed date that way too
|
||||||
|
if ("month" in attempt) attempt.month -= 1;
|
||||||
|
|
||||||
|
let date: Date = (Date as any).fromString(split[1]) as Date;
|
||||||
|
console.log("Read", attempt, "and", date);
|
||||||
|
if ("invalid" in (date as any)) continue;
|
||||||
|
|
||||||
|
const filterCard = document.createElement("span");
|
||||||
|
filterCard.ariaLabel = window.lang.strings("clickToRemoveFilter");
|
||||||
|
filterCard.classList.add("button", "~neutral", "@low", "center", "m-2", "h-full");
|
||||||
|
filterCard.innerHTML = `
|
||||||
|
<span class="font-bold mr-2">${queryFormat.name}:</span> ${(compareType == 1) ? window.lang.strings("after")+" " : ((compareType == -1) ? window.lang.strings("before")+" " : "")}${split[1]}
|
||||||
|
`;
|
||||||
|
|
||||||
|
filterCard.addEventListener("click", () => {
|
||||||
|
for (let quote of [`"`, `'`, ``]) {
|
||||||
|
let regex = new RegExp(split[0] + ":" + quote + unmodifiedValue + quote, "ig");
|
||||||
|
this._c.search.value = this._c.search.value.replace(regex, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
this._c.search.oninput((null as Event));
|
||||||
|
})
|
||||||
|
|
||||||
|
this._c.filterArea.appendChild(filterCard);
|
||||||
|
|
||||||
|
let cachedResult = [...result];
|
||||||
|
for (let id of cachedResult) {
|
||||||
|
const u = this._items[id];
|
||||||
|
const unixValue = Object.getOwnPropertyDescriptor(Object.getPrototypeOf(u), queryFormat.getter).get.call(u);
|
||||||
|
if (unixValue == 0) {
|
||||||
|
result.splice(result.indexOf(id), 1);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let value = new Date(unixValue*1000);
|
||||||
|
|
||||||
|
const getterPairs: [string, () => number][] = [["year", Date.prototype.getFullYear], ["month", Date.prototype.getMonth], ["day", Date.prototype.getDate], ["hour", Date.prototype.getHours], ["minute", Date.prototype.getMinutes]];
|
||||||
|
|
||||||
|
// When doing > or < <time> with no date, we need to ignore the rest of the Date object
|
||||||
|
if (compareType != 0 && Object.keys(attempt).length == 2 && "hour" in attempt && "minute" in attempt) {
|
||||||
|
const temp = new Date(date.valueOf());
|
||||||
|
temp.setHours(value.getHours(), value.getMinutes());
|
||||||
|
value = temp;
|
||||||
|
console.log("just hours/minutes workaround, value set to", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
let match = true;
|
||||||
|
if (compareType == 0) {
|
||||||
|
for (let pair of getterPairs) {
|
||||||
|
if (pair[0] in attempt) {
|
||||||
|
if (compareType == 0 && attempt[pair[0]] != pair[1].call(value)) {
|
||||||
|
match = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (compareType == -1) {
|
||||||
|
match = (value < date);
|
||||||
|
} else if (compareType == 1) {
|
||||||
|
match = (value > date);
|
||||||
|
}
|
||||||
|
if (!match) {
|
||||||
|
result.splice(result.indexOf(id), 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
showHideSearchOptionsHeader = () => {
|
||||||
|
const sortingBy = !(this._c.sortingByButton.parentElement.classList.contains("hidden"));
|
||||||
|
const hasFilters = this._c.filterArea.textContent != "";
|
||||||
|
console.log("sortingBy", sortingBy, "hasFilters", hasFilters);
|
||||||
|
if (sortingBy || hasFilters) {
|
||||||
|
this._c.searchOptionsHeader.classList.remove("hidden");
|
||||||
|
} else {
|
||||||
|
this._c.searchOptionsHeader.classList.add("hidden");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
get items(): { [id: string]: SearchableItem } { return this._items; }
|
||||||
|
set items(v: { [id: string]: SearchableItem }) {
|
||||||
|
this._items = v;
|
||||||
|
}
|
||||||
|
|
||||||
|
get ordering(): string[] { return this._ordering; }
|
||||||
|
set ordering(v: string[]) { this._ordering = v; }
|
||||||
|
|
||||||
|
onSearchBoxChange = () => {
|
||||||
|
const query = this._c.search.value;
|
||||||
|
if (!query) {
|
||||||
|
this.inSearch = false;
|
||||||
|
} else {
|
||||||
|
this.inSearch = true;
|
||||||
|
}
|
||||||
|
const results = this.search(query);
|
||||||
|
this._c.setVisibility(results, true);
|
||||||
|
this._c.onSearchCallback();
|
||||||
|
this.showHideSearchOptionsHeader();
|
||||||
|
if (results.length == 0) {
|
||||||
|
this._c.notFoundPanel.classList.remove("unfocused");
|
||||||
|
} else {
|
||||||
|
this._c.notFoundPanel.classList.add("unfocused");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor(c: SearchConfiguration) {
|
||||||
|
this._c = c;
|
||||||
|
|
||||||
|
this._c.search.oninput = this.onSearchBoxChange;
|
||||||
|
|
||||||
|
const clearSearchButtons = Array.from(document.querySelectorAll(this._c.clearSearchButtonSelector)) as Array<HTMLSpanElement>;
|
||||||
|
for (let b of clearSearchButtons) {
|
||||||
|
b.addEventListener("click", () => {
|
||||||
|
this._c.search.value = "";
|
||||||
|
this.onSearchBoxChange();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user