2023-04-18 23:20:02 +02:00
|
|
|
import {htmlToDom} from '../services/dom';
|
|
|
|
import {debounce} from '../services/util';
|
|
|
|
import {KeyboardNavigationHandler} from '../services/keyboard-navigation';
|
|
|
|
import {Component} from './component';
|
2022-11-20 23:20:31 +01:00
|
|
|
|
2022-11-14 11:24:14 +01:00
|
|
|
/**
|
2022-11-23 01:12:41 +01:00
|
|
|
* Global (header) search box handling.
|
|
|
|
* Mainly to show live results preview.
|
2022-11-14 11:24:14 +01:00
|
|
|
*/
|
2022-11-23 01:12:41 +01:00
|
|
|
export class GlobalSearch extends Component {
|
2022-11-14 11:24:14 +01:00
|
|
|
|
|
|
|
setup() {
|
2022-11-20 22:50:59 +01:00
|
|
|
this.container = this.$el;
|
2022-11-14 11:24:14 +01:00
|
|
|
this.input = this.$refs.input;
|
|
|
|
this.suggestions = this.$refs.suggestions;
|
|
|
|
this.suggestionResultsWrap = this.$refs.suggestionResults;
|
2022-11-20 23:20:31 +01:00
|
|
|
this.loadingWrap = this.$refs.loading;
|
2022-11-21 11:29:12 +01:00
|
|
|
this.button = this.$refs.button;
|
2022-11-14 11:24:14 +01:00
|
|
|
|
|
|
|
this.setupListeners();
|
|
|
|
}
|
|
|
|
|
|
|
|
setupListeners() {
|
2022-11-20 23:20:31 +01:00
|
|
|
const updateSuggestionsDebounced = debounce(this.updateSuggestions.bind(this), 200, false);
|
|
|
|
|
|
|
|
// Handle search input changes
|
2022-11-14 11:24:14 +01:00
|
|
|
this.input.addEventListener('input', () => {
|
2023-04-18 23:20:02 +02:00
|
|
|
const {value} = this.input;
|
2022-11-14 11:24:14 +01:00
|
|
|
if (value.length > 0) {
|
2022-11-20 23:20:31 +01:00
|
|
|
this.loadingWrap.style.display = 'block';
|
|
|
|
this.suggestionResultsWrap.style.opacity = '0.5';
|
|
|
|
updateSuggestionsDebounced(value);
|
2023-04-18 23:20:02 +02:00
|
|
|
} else {
|
2022-11-14 11:24:14 +01:00
|
|
|
this.hideSuggestions();
|
|
|
|
}
|
|
|
|
});
|
2022-11-20 23:20:31 +01:00
|
|
|
|
|
|
|
// Allow double click to show auto-click suggestions
|
|
|
|
this.input.addEventListener('dblclick', () => {
|
|
|
|
this.input.setAttribute('autocomplete', 'on');
|
2022-11-21 11:29:12 +01:00
|
|
|
this.button.focus();
|
2022-11-20 23:20:31 +01:00
|
|
|
this.input.focus();
|
2022-11-21 18:35:19 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
new KeyboardNavigationHandler(this.container, () => {
|
|
|
|
this.hideSuggestions();
|
|
|
|
});
|
2022-11-14 11:24:14 +01:00
|
|
|
}
|
|
|
|
|
2022-11-20 23:20:31 +01:00
|
|
|
/**
|
|
|
|
* @param {String} search
|
|
|
|
*/
|
2022-11-14 11:24:14 +01:00
|
|
|
async updateSuggestions(search) {
|
2022-11-21 11:29:12 +01:00
|
|
|
const {data: results} = await window.$http.get('/search/suggest', {term: search});
|
2022-11-20 23:20:31 +01:00
|
|
|
if (!this.input.value) {
|
|
|
|
return;
|
|
|
|
}
|
2023-04-18 23:20:02 +02:00
|
|
|
|
2022-11-14 11:24:14 +01:00
|
|
|
const resultDom = htmlToDom(results);
|
|
|
|
|
|
|
|
this.suggestionResultsWrap.innerHTML = '';
|
2022-11-20 23:20:31 +01:00
|
|
|
this.suggestionResultsWrap.style.opacity = '1';
|
|
|
|
this.loadingWrap.style.display = 'none';
|
2022-11-14 11:24:14 +01:00
|
|
|
this.suggestionResultsWrap.append(resultDom);
|
2022-11-20 22:50:59 +01:00
|
|
|
if (!this.container.classList.contains('search-active')) {
|
|
|
|
this.showSuggestions();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
showSuggestions() {
|
|
|
|
this.container.classList.add('search-active');
|
|
|
|
window.requestAnimationFrame(() => {
|
|
|
|
this.suggestions.classList.add('search-suggestions-animation');
|
2023-04-18 23:20:02 +02:00
|
|
|
});
|
2022-11-14 11:24:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
hideSuggestions() {
|
2022-11-20 22:50:59 +01:00
|
|
|
this.container.classList.remove('search-active');
|
|
|
|
this.suggestions.classList.remove('search-suggestions-animation');
|
2022-11-14 11:24:14 +01:00
|
|
|
this.suggestionResultsWrap.innerHTML = '';
|
|
|
|
}
|
2023-04-18 23:20:02 +02:00
|
|
|
|
|
|
|
}
|