2020-06-28 23:15:05 +01:00
|
|
|
import {escapeHtml} from "../services/util";
|
|
|
|
import {onChildEvent} from "../services/dom";
|
|
|
|
|
|
|
|
const ajaxCache = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* AutoSuggest
|
|
|
|
* @extends {Component}
|
|
|
|
*/
|
|
|
|
class AutoSuggest {
|
|
|
|
setup() {
|
|
|
|
this.parent = this.$el.parentElement;
|
|
|
|
this.container = this.$el;
|
|
|
|
this.type = this.$opts.type;
|
|
|
|
this.url = this.$opts.url;
|
|
|
|
this.input = this.$refs.input;
|
|
|
|
this.list = this.$refs.list;
|
|
|
|
|
2020-06-29 22:11:03 +01:00
|
|
|
this.lastPopulated = 0;
|
2020-06-28 23:15:05 +01:00
|
|
|
this.setupListeners();
|
|
|
|
}
|
|
|
|
|
|
|
|
setupListeners() {
|
|
|
|
this.input.addEventListener('input', this.requestSuggestions.bind(this));
|
|
|
|
this.input.addEventListener('focus', this.requestSuggestions.bind(this));
|
|
|
|
this.input.addEventListener('keydown', event => {
|
|
|
|
if (event.key === 'Tab') {
|
|
|
|
this.hideSuggestions();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.input.addEventListener('blur', this.hideSuggestionsIfFocusedLost.bind(this));
|
|
|
|
this.container.addEventListener('keydown', this.containerKeyDown.bind(this));
|
|
|
|
|
|
|
|
onChildEvent(this.list, 'button', 'click', (event, el) => {
|
|
|
|
this.selectSuggestion(el.textContent);
|
|
|
|
});
|
|
|
|
onChildEvent(this.list, 'button', 'keydown', (event, el) => {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
this.selectSuggestion(el.textContent);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
selectSuggestion(value) {
|
|
|
|
this.input.value = value;
|
2020-06-29 22:11:03 +01:00
|
|
|
this.lastPopulated = Date.now();
|
2020-06-28 23:15:05 +01:00
|
|
|
this.input.focus();
|
2020-06-29 22:11:03 +01:00
|
|
|
this.input.dispatchEvent(new Event('input', {bubbles: true}));
|
|
|
|
this.input.dispatchEvent(new Event('change', {bubbles: true}));
|
2020-06-28 23:15:05 +01:00
|
|
|
this.hideSuggestions();
|
|
|
|
}
|
|
|
|
|
|
|
|
containerKeyDown(event) {
|
|
|
|
if (event.key === 'Enter') event.preventDefault();
|
|
|
|
if (this.list.classList.contains('hidden')) return;
|
|
|
|
|
|
|
|
// Down arrow
|
|
|
|
if (event.key === 'ArrowDown') {
|
|
|
|
this.moveFocus(true);
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
// Up Arrow
|
|
|
|
else if (event.key === 'ArrowUp') {
|
|
|
|
this.moveFocus(false);
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
// Escape key
|
|
|
|
else if (event.key === 'Escape') {
|
|
|
|
this.hideSuggestions();
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
moveFocus(forward = true) {
|
|
|
|
const focusables = Array.from(this.container.querySelectorAll('input,button'));
|
|
|
|
const index = focusables.indexOf(document.activeElement);
|
|
|
|
const newFocus = focusables[index + (forward ? 1 : -1)];
|
|
|
|
if (newFocus) {
|
|
|
|
newFocus.focus()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async requestSuggestions() {
|
2020-06-29 22:11:03 +01:00
|
|
|
if (Date.now() - this.lastPopulated < 50) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-28 23:15:05 +01:00
|
|
|
const nameFilter = this.getNameFilterIfNeeded();
|
2022-09-28 13:50:40 +01:00
|
|
|
const search = this.input.value.toLowerCase();
|
2020-06-28 23:15:05 +01:00
|
|
|
const suggestions = await this.loadSuggestions(search, nameFilter);
|
2022-09-28 13:50:40 +01:00
|
|
|
|
|
|
|
const toShow = suggestions.filter(val => {
|
|
|
|
return search === '' || val.toLowerCase().startsWith(search);
|
|
|
|
}).slice(0, 10);
|
2020-06-28 23:15:05 +01:00
|
|
|
|
|
|
|
this.displaySuggestions(toShow);
|
|
|
|
}
|
|
|
|
|
|
|
|
getNameFilterIfNeeded() {
|
|
|
|
if (this.type !== 'value') return null;
|
|
|
|
return this.parent.querySelector('input').value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {String} search
|
|
|
|
* @param {String|null} nameFilter
|
|
|
|
* @returns {Promise<Object|String|*>}
|
|
|
|
*/
|
|
|
|
async loadSuggestions(search, nameFilter = null) {
|
2022-09-28 13:50:40 +01:00
|
|
|
// Truncate search to prevent over numerous lookups
|
|
|
|
search = search.slice(0, 4);
|
|
|
|
|
2020-06-28 23:15:05 +01:00
|
|
|
const params = {search, name: nameFilter};
|
|
|
|
const cacheKey = `${this.url}:${JSON.stringify(params)}`;
|
|
|
|
|
|
|
|
if (ajaxCache[cacheKey]) {
|
|
|
|
return ajaxCache[cacheKey];
|
|
|
|
}
|
|
|
|
|
|
|
|
const resp = await window.$http.get(this.url, params);
|
|
|
|
ajaxCache[cacheKey] = resp.data;
|
|
|
|
return resp.data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {String[]} suggestions
|
|
|
|
*/
|
|
|
|
displaySuggestions(suggestions) {
|
|
|
|
if (suggestions.length === 0) {
|
|
|
|
return this.hideSuggestions();
|
|
|
|
}
|
|
|
|
|
2022-04-20 14:03:47 +01:00
|
|
|
this.list.innerHTML = suggestions.map(value => `<li><button type="button" class="text-item">${escapeHtml(value)}</button></li>`).join('');
|
2020-06-28 23:15:05 +01:00
|
|
|
this.list.style.display = 'block';
|
|
|
|
for (const button of this.list.querySelectorAll('button')) {
|
|
|
|
button.addEventListener('blur', this.hideSuggestionsIfFocusedLost.bind(this));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
hideSuggestions() {
|
|
|
|
this.list.style.display = 'none';
|
|
|
|
}
|
|
|
|
|
|
|
|
hideSuggestionsIfFocusedLost(event) {
|
|
|
|
if (!this.container.contains(event.relatedTarget)) {
|
|
|
|
this.hideSuggestions();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default AutoSuggest;
|