82 lines
2.4 KiB
JavaScript
82 lines
2.4 KiB
JavaScript
import {debounce} from "../services/util";
|
|
import {transitionHeight} from "../services/animations";
|
|
import {Component} from "./component";
|
|
|
|
export class DropdownSearch extends Component {
|
|
|
|
setup() {
|
|
this.elem = this.$el;
|
|
this.searchInput = this.$refs.searchInput;
|
|
this.loadingElem = this.$refs.loading;
|
|
this.listContainerElem = this.$refs.listContainer;
|
|
|
|
this.localSearchSelector = this.$opts.localSearchSelector;
|
|
this.url = this.$opts.url;
|
|
|
|
this.elem.addEventListener('show', this.onShow.bind(this));
|
|
this.searchInput.addEventListener('input', this.onSearch.bind(this));
|
|
|
|
this.runAjaxSearch = debounce(this.runAjaxSearch, 300, false);
|
|
}
|
|
|
|
onShow() {
|
|
this.loadList();
|
|
}
|
|
|
|
onSearch() {
|
|
const input = this.searchInput.value.toLowerCase().trim();
|
|
if (this.localSearchSelector) {
|
|
this.runLocalSearch(input);
|
|
} else {
|
|
this.toggleLoading(true);
|
|
this.listContainerElem.innerHTML = '';
|
|
this.runAjaxSearch(input);
|
|
}
|
|
}
|
|
|
|
runAjaxSearch(searchTerm) {
|
|
this.loadList(searchTerm);
|
|
}
|
|
|
|
runLocalSearch(searchTerm) {
|
|
const listItems = this.listContainerElem.querySelectorAll(this.localSearchSelector);
|
|
for (let listItem of listItems) {
|
|
const match = !searchTerm || listItem.textContent.toLowerCase().includes(searchTerm);
|
|
listItem.style.display = match ? 'flex' : 'none';
|
|
listItem.classList.toggle('hidden', !match);
|
|
}
|
|
}
|
|
|
|
async loadList(searchTerm = '') {
|
|
this.listContainerElem.innerHTML = '';
|
|
this.toggleLoading(true);
|
|
|
|
try {
|
|
const resp = await window.$http.get(this.getAjaxUrl(searchTerm));
|
|
const animate = transitionHeight(this.listContainerElem, 80);
|
|
this.listContainerElem.innerHTML = resp.data;
|
|
animate();
|
|
} catch (err) {
|
|
console.error(err);
|
|
}
|
|
|
|
this.toggleLoading(false);
|
|
if (this.localSearchSelector) {
|
|
this.onSearch();
|
|
}
|
|
}
|
|
|
|
getAjaxUrl(searchTerm = null) {
|
|
if (!searchTerm) {
|
|
return this.url;
|
|
}
|
|
|
|
const joiner = this.url.includes('?') ? '&' : '?';
|
|
return `${this.url}${joiner}search=${encodeURIComponent(searchTerm)}`;
|
|
}
|
|
|
|
toggleLoading(show = false) {
|
|
this.loadingElem.style.display = show ? 'block' : 'none';
|
|
}
|
|
|
|
} |