2020-08-25 20:10:31 -04:00
|
|
|
/* eslint-disable consistent-return */
|
2021-02-14 13:09:20 -05:00
|
|
|
import fuzzaldrinPlus from 'fuzzaldrin-plus';
|
2020-08-25 20:10:31 -04:00
|
|
|
import $ from 'jquery';
|
|
|
|
import { escape } from 'lodash';
|
|
|
|
import { isObject } from '~/lib/utils/type_utility';
|
2021-02-14 13:09:20 -05:00
|
|
|
import { visitUrl } from '~/lib/utils/url_utility';
|
2020-08-25 20:10:31 -04:00
|
|
|
import { GitLabDropdownFilter } from './gl_dropdown_filter';
|
2021-02-14 13:09:20 -05:00
|
|
|
import { GitLabDropdownInput } from './gl_dropdown_input';
|
|
|
|
import { GitLabDropdownRemote } from './gl_dropdown_remote';
|
|
|
|
import renderItem from './render';
|
2020-08-25 20:10:31 -04:00
|
|
|
|
|
|
|
const LOADING_CLASS = 'is-loading';
|
|
|
|
|
|
|
|
const PAGE_TWO_CLASS = 'is-page-two';
|
|
|
|
|
|
|
|
const ACTIVE_CLASS = 'is-active';
|
|
|
|
|
|
|
|
const INDETERMINATE_CLASS = 'is-indeterminate';
|
|
|
|
|
|
|
|
let currentIndex = -1;
|
|
|
|
|
|
|
|
const NON_SELECTABLE_CLASSES = '.divider, .separator, .dropdown-header, .dropdown-menu-empty-item';
|
|
|
|
|
|
|
|
const SELECTABLE_CLASSES = `.dropdown-content li:not(${NON_SELECTABLE_CLASSES}, .option-hidden)`;
|
|
|
|
|
|
|
|
const CURSOR_SELECT_SCROLL_PADDING = 5;
|
|
|
|
|
|
|
|
const FILTER_INPUT = '.dropdown-input .dropdown-input-field:not(.dropdown-no-filter)';
|
|
|
|
|
|
|
|
const NO_FILTER_INPUT = '.dropdown-input .dropdown-input-field.dropdown-no-filter';
|
|
|
|
|
2021-02-19 04:10:35 -05:00
|
|
|
let mouseEventListenersAdded = false;
|
|
|
|
let mousedownTarget = null;
|
|
|
|
let mouseupTarget = null;
|
|
|
|
|
|
|
|
function addGlobalMouseEventListeners() {
|
|
|
|
// Remember mousedown and mouseup locations.
|
|
|
|
// Required in the `hide.bs.dropdown` listener for
|
|
|
|
// dropdown close prevention in some cases.
|
|
|
|
document.addEventListener('mousedown', ({ target }) => {
|
|
|
|
mousedownTarget = target;
|
|
|
|
});
|
|
|
|
document.addEventListener('mouseup', ({ target }) => {
|
|
|
|
mouseupTarget = target;
|
|
|
|
});
|
|
|
|
document.addEventListener('click', () => {
|
|
|
|
mousedownTarget = null;
|
|
|
|
mouseupTarget = null;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-08-25 20:10:31 -04:00
|
|
|
export class GitLabDropdown {
|
|
|
|
constructor(el1, options) {
|
|
|
|
let selector;
|
|
|
|
let self;
|
|
|
|
this.el = el1;
|
|
|
|
this.options = options;
|
|
|
|
this.updateLabel = this.updateLabel.bind(this);
|
|
|
|
this.opened = this.opened.bind(this);
|
2021-02-19 04:10:35 -05:00
|
|
|
this.hide = this.hide.bind(this);
|
|
|
|
this.hidden = this.hidden.bind(this);
|
2020-08-25 20:10:31 -04:00
|
|
|
this.shouldPropagate = this.shouldPropagate.bind(this);
|
2021-02-19 04:10:35 -05:00
|
|
|
if (!mouseEventListenersAdded) {
|
|
|
|
addGlobalMouseEventListeners();
|
|
|
|
mouseEventListenersAdded = true;
|
|
|
|
}
|
2020-08-25 20:10:31 -04:00
|
|
|
self = this;
|
|
|
|
selector = $(this.el).data('target');
|
|
|
|
this.dropdown = selector != null ? $(selector) : $(this.el).parent();
|
|
|
|
// Set Defaults
|
|
|
|
this.filterInput = this.options.filterInput || this.getElement(FILTER_INPUT);
|
|
|
|
this.noFilterInput = this.options.noFilterInput || this.getElement(NO_FILTER_INPUT);
|
|
|
|
this.highlight = Boolean(this.options.highlight);
|
|
|
|
this.icon = Boolean(this.options.icon);
|
|
|
|
this.filterInputBlur =
|
|
|
|
this.options.filterInputBlur != null ? this.options.filterInputBlur : true;
|
|
|
|
// If no input is passed create a default one
|
|
|
|
self = this;
|
|
|
|
// If selector was passed
|
|
|
|
if (typeof this.filterInput === 'string') {
|
|
|
|
this.filterInput = this.getElement(this.filterInput);
|
|
|
|
}
|
|
|
|
const searchFields = this.options.search ? this.options.search.fields : [];
|
|
|
|
if (this.options.data) {
|
|
|
|
// If we provided data
|
|
|
|
// data could be an array of objects or a group of arrays
|
|
|
|
if (typeof this.options.data === 'object' && !(this.options.data instanceof Function)) {
|
|
|
|
this.fullData = this.options.data;
|
|
|
|
currentIndex = -1;
|
|
|
|
this.parseData(this.options.data);
|
|
|
|
this.focusTextInput();
|
|
|
|
} else {
|
|
|
|
this.remote = new GitLabDropdownRemote(this.options.data, {
|
|
|
|
dataType: this.options.dataType,
|
|
|
|
beforeSend: this.toggleLoading.bind(this),
|
2020-12-23 16:10:24 -05:00
|
|
|
success: (data) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
this.fullData = data;
|
|
|
|
this.parseData(this.fullData);
|
|
|
|
this.focusTextInput();
|
|
|
|
|
|
|
|
// Update dropdown position since remote data may have changed dropdown size
|
|
|
|
this.dropdown.find('.dropdown-menu-toggle').dropdown('update');
|
|
|
|
|
|
|
|
if (
|
|
|
|
this.options.filterable &&
|
|
|
|
this.filter &&
|
|
|
|
this.filter.input &&
|
|
|
|
this.filter.input.val() &&
|
|
|
|
this.filter.input.val().trim() !== ''
|
|
|
|
) {
|
|
|
|
return this.filter.input.trigger('input');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
instance: this,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (this.noFilterInput.length) {
|
|
|
|
this.plainInput = new GitLabDropdownInput(this.noFilterInput, this.options);
|
|
|
|
this.plainInput.onInput(this.addInput.bind(this));
|
|
|
|
}
|
|
|
|
// Init filterable
|
|
|
|
if (this.options.filterable) {
|
|
|
|
this.filter = new GitLabDropdownFilter(this.filterInput, {
|
|
|
|
elIsInput: $(this.el).is('input'),
|
|
|
|
filterInputBlur: this.filterInputBlur,
|
|
|
|
filterByText: this.options.filterByText,
|
|
|
|
onFilter: this.options.onFilter,
|
|
|
|
remote: this.options.filterRemote,
|
|
|
|
query: this.options.data,
|
|
|
|
keys: searchFields,
|
|
|
|
instance: this,
|
|
|
|
elements: () => {
|
|
|
|
selector = `.dropdown-content li:not(${NON_SELECTABLE_CLASSES})`;
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
selector = `.dropdown-page-one ${selector}`;
|
|
|
|
}
|
|
|
|
return $(selector, this.dropdown);
|
|
|
|
},
|
|
|
|
data: () => this.fullData,
|
2020-12-23 16:10:24 -05:00
|
|
|
callback: (data) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
this.parseData(data);
|
|
|
|
if (this.filterInput.val() !== '') {
|
|
|
|
selector = SELECTABLE_CLASSES;
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
selector = `.dropdown-page-one ${selector}`;
|
|
|
|
}
|
|
|
|
if ($(this.el).is('input')) {
|
|
|
|
currentIndex = -1;
|
|
|
|
} else {
|
2020-12-23 07:10:26 -05:00
|
|
|
$(selector, this.dropdown).first().find('a').addClass('is-focused');
|
2020-08-25 20:10:31 -04:00
|
|
|
currentIndex = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// Event listeners
|
|
|
|
this.dropdown.on('shown.bs.dropdown', this.opened);
|
2021-02-19 04:10:35 -05:00
|
|
|
this.dropdown.on('hide.bs.dropdown', this.hide);
|
2020-08-25 20:10:31 -04:00
|
|
|
this.dropdown.on('hidden.bs.dropdown', this.hidden);
|
|
|
|
$(this.el).on('update.label', this.updateLabel);
|
|
|
|
this.dropdown.on('click', '.dropdown-menu, .dropdown-menu-close', this.shouldPropagate);
|
2020-12-23 16:10:24 -05:00
|
|
|
this.dropdown.on('keyup', (e) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
// Escape key
|
|
|
|
if (e.which === 27) {
|
|
|
|
return $('.dropdown-menu-close', this.dropdown).trigger('click');
|
|
|
|
}
|
|
|
|
});
|
2020-12-23 16:10:24 -05:00
|
|
|
this.dropdown.on('blur', 'a', (e) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
let $dropdownMenu;
|
|
|
|
let $relatedTarget;
|
|
|
|
if (e.relatedTarget != null) {
|
|
|
|
$relatedTarget = $(e.relatedTarget);
|
|
|
|
$dropdownMenu = $relatedTarget.closest('.dropdown-menu');
|
|
|
|
if ($dropdownMenu.length === 0) {
|
|
|
|
return this.dropdown.removeClass('show');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
2020-12-23 16:10:24 -05:00
|
|
|
this.dropdown.find('.dropdown-toggle-page, .dropdown-menu-back').on('click', (e) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
return this.togglePage();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (this.options.selectable) {
|
|
|
|
selector = '.dropdown-content a';
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
selector = '.dropdown-page-one .dropdown-content a';
|
|
|
|
}
|
2020-12-23 16:10:24 -05:00
|
|
|
this.dropdown.on('click', selector, (e) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
const $el = $(e.currentTarget);
|
|
|
|
const selected = self.rowClicked($el);
|
|
|
|
const selectedObj = selected ? selected[0] : null;
|
|
|
|
const isMarking = selected ? selected[1] : null;
|
|
|
|
if (this.options.clicked) {
|
|
|
|
this.options.clicked.call(this, {
|
|
|
|
selectedObj,
|
|
|
|
$el,
|
|
|
|
e,
|
|
|
|
isMarking,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update label right after all modifications in dropdown has been done
|
|
|
|
if (this.options.toggleLabel) {
|
|
|
|
this.updateLabel(selectedObj, $el, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
$el.trigger('blur');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finds an element inside wrapper element
|
|
|
|
getElement(selector) {
|
|
|
|
return this.dropdown.find(selector);
|
|
|
|
}
|
|
|
|
|
|
|
|
toggleLoading() {
|
|
|
|
return $('.dropdown-menu', this.dropdown).toggleClass(LOADING_CLASS);
|
|
|
|
}
|
|
|
|
|
|
|
|
togglePage() {
|
|
|
|
const menu = $('.dropdown-menu', this.dropdown);
|
|
|
|
if (menu.hasClass(PAGE_TWO_CLASS)) {
|
|
|
|
if (this.remote) {
|
|
|
|
this.remote.execute();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
menu.toggleClass(PAGE_TWO_CLASS);
|
|
|
|
// Focus first visible input on active page
|
|
|
|
return this.dropdown.find('[class^="dropdown-page-"]:visible :text:visible:first').focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
parseData(data) {
|
|
|
|
let groupData;
|
|
|
|
let html;
|
|
|
|
this.renderedData = data;
|
|
|
|
if (this.options.filterable && data.length === 0) {
|
|
|
|
// render no matching results
|
|
|
|
html = [this.noResults()];
|
|
|
|
}
|
|
|
|
// Handle array groups
|
|
|
|
else if (isObject(data)) {
|
|
|
|
html = [];
|
|
|
|
|
2020-12-23 16:10:24 -05:00
|
|
|
Object.keys(data).forEach((name) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
groupData = data[name];
|
|
|
|
html.push(
|
|
|
|
this.renderItem(
|
|
|
|
{
|
|
|
|
content: name,
|
|
|
|
type: 'header',
|
|
|
|
},
|
|
|
|
name,
|
|
|
|
),
|
|
|
|
);
|
2020-12-23 16:10:24 -05:00
|
|
|
this.renderData(groupData, name).map((item) => html.push(item));
|
2020-08-25 20:10:31 -04:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// Render each row
|
|
|
|
html = this.renderData(data);
|
|
|
|
}
|
|
|
|
// Render the full menu
|
|
|
|
const fullHtml = this.renderMenu(html);
|
|
|
|
return this.appendMenu(fullHtml);
|
|
|
|
}
|
|
|
|
|
|
|
|
renderData(data, group) {
|
|
|
|
return data.map((obj, index) => this.renderItem(obj, group || false, index));
|
|
|
|
}
|
|
|
|
|
|
|
|
shouldPropagate(e) {
|
|
|
|
let $target;
|
|
|
|
if (this.options.multiSelect || this.options.shouldPropagate === false) {
|
|
|
|
$target = $(e.target);
|
|
|
|
if (
|
|
|
|
$target &&
|
|
|
|
!$target.hasClass('dropdown-menu-close') &&
|
|
|
|
!$target.hasClass('dropdown-menu-close-icon') &&
|
2021-07-20 23:09:46 -04:00
|
|
|
!$target.is('use') &&
|
2020-08-25 20:10:31 -04:00
|
|
|
!$target.data('isLink')
|
|
|
|
) {
|
|
|
|
e.stopPropagation();
|
|
|
|
|
|
|
|
// This prevents automatic scrolling to the top
|
|
|
|
if ($target.closest('a').length) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
filteredFullData() {
|
|
|
|
return this.fullData.filter(
|
2020-12-23 16:10:24 -05:00
|
|
|
(r) =>
|
2020-08-25 20:10:31 -04:00
|
|
|
typeof r === 'object' &&
|
|
|
|
!Object.prototype.hasOwnProperty.call(r, 'beforeDivider') &&
|
|
|
|
!Object.prototype.hasOwnProperty.call(r, 'header'),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
opened(e) {
|
|
|
|
this.resetRows();
|
|
|
|
this.addArrowKeyEvent();
|
|
|
|
|
|
|
|
const dropdownToggle = this.dropdown.find('.dropdown-menu-toggle');
|
|
|
|
const hasFilterBulkUpdate = dropdownToggle.hasClass('js-filter-bulk-update');
|
|
|
|
const shouldRefreshOnOpen = dropdownToggle.hasClass('js-gl-dropdown-refresh-on-open');
|
|
|
|
const hasMultiSelect = dropdownToggle.hasClass('js-multiselect');
|
|
|
|
|
|
|
|
// Makes indeterminate items effective
|
|
|
|
if (this.fullData && (shouldRefreshOnOpen || hasFilterBulkUpdate)) {
|
|
|
|
this.parseData(this.fullData);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process the data to make sure rendered data
|
|
|
|
// matches the correct layout
|
|
|
|
const inputValue = this.filterInput.val();
|
|
|
|
if (this.fullData && hasMultiSelect && this.options.processData && inputValue.length === 0) {
|
|
|
|
this.options.processData.call(
|
|
|
|
this.options,
|
|
|
|
inputValue,
|
|
|
|
this.filteredFullData(),
|
|
|
|
this.parseData.bind(this),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const contentHtml = $('.dropdown-content', this.dropdown).html();
|
|
|
|
if (this.remote && contentHtml === '') {
|
|
|
|
this.remote.execute();
|
|
|
|
} else {
|
|
|
|
this.focusTextInput();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.showMenuAbove) {
|
|
|
|
this.positionMenuAbove();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.opened) {
|
|
|
|
if (this.options.preserveContext) {
|
|
|
|
this.options.opened(e);
|
|
|
|
} else {
|
|
|
|
this.options.opened.call(this, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.dropdown.trigger('shown.gl.dropdown');
|
|
|
|
}
|
|
|
|
|
|
|
|
positionMenuAbove() {
|
|
|
|
const $menu = this.dropdown.find('.dropdown-menu');
|
|
|
|
|
|
|
|
$menu.addClass('dropdown-open-top');
|
|
|
|
$menu.css('top', 'initial');
|
|
|
|
$menu.css('bottom', '100%');
|
|
|
|
}
|
|
|
|
|
2021-02-19 04:10:35 -05:00
|
|
|
hide(e) {
|
|
|
|
// Prevent dropdowns with a search from being closed when the
|
|
|
|
// mousedown event happened inside the dropdown box and only
|
|
|
|
// the mouseup event did not.
|
|
|
|
if (this.options.search && mousedownTarget) {
|
|
|
|
const isIn = (element, $possibleContainer) => Boolean($possibleContainer.has(element).length);
|
|
|
|
const $menu = this.dropdown.find('.dropdown-menu');
|
|
|
|
const mousedownInsideDropdown = isIn(mousedownTarget, $menu);
|
|
|
|
const mouseupOutsideDropdown = !isIn(mouseupTarget, $menu);
|
|
|
|
if (mousedownInsideDropdown && mouseupOutsideDropdown) {
|
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-25 20:10:31 -04:00
|
|
|
hidden(e) {
|
|
|
|
this.resetRows();
|
|
|
|
this.removeArrowKeyEvent();
|
|
|
|
const $input = this.dropdown.find('.dropdown-input-field');
|
|
|
|
if (this.options.filterable) {
|
|
|
|
$input.blur();
|
|
|
|
}
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
$('.dropdown-menu', this.dropdown).removeClass(PAGE_TWO_CLASS);
|
|
|
|
}
|
|
|
|
if (this.options.hidden) {
|
|
|
|
this.options.hidden.call(this, e);
|
|
|
|
}
|
|
|
|
return this.dropdown.trigger('hidden.gl.dropdown');
|
|
|
|
}
|
|
|
|
|
|
|
|
// Render the full menu
|
|
|
|
renderMenu(html) {
|
|
|
|
if (this.options.renderMenu) {
|
|
|
|
return this.options.renderMenu(html);
|
|
|
|
}
|
|
|
|
return $('<ul>').append(html);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append the menu into the dropdown
|
|
|
|
appendMenu(html) {
|
|
|
|
return this.clearMenu().append(html);
|
|
|
|
}
|
|
|
|
|
|
|
|
clearMenu() {
|
|
|
|
let selector = '.dropdown-content';
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
if (this.options.containerSelector) {
|
|
|
|
selector = this.options.containerSelector;
|
|
|
|
} else {
|
|
|
|
selector = '.dropdown-page-one .dropdown-content';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $(selector, this.dropdown).empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
renderItem(data, group, index) {
|
|
|
|
let parent;
|
|
|
|
|
|
|
|
if (this.dropdown && this.dropdown[0]) {
|
|
|
|
parent = this.dropdown[0].parentNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
return renderItem({
|
|
|
|
instance: this,
|
|
|
|
options: {
|
|
|
|
...this.options,
|
|
|
|
icon: this.icon,
|
|
|
|
highlight: this.highlight,
|
2020-12-23 16:10:24 -05:00
|
|
|
highlightText: (text) => this.highlightTextMatches(text, this.filterInput.val()),
|
2020-08-25 20:10:31 -04:00
|
|
|
highlightTemplate: this.highlightTemplate.bind(this),
|
|
|
|
parent,
|
|
|
|
},
|
|
|
|
data,
|
|
|
|
group,
|
|
|
|
index,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// eslint-disable-next-line class-methods-use-this
|
|
|
|
highlightTemplate(text, template) {
|
|
|
|
return `"<b>${escape(text)}</b>" ${template}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
// eslint-disable-next-line class-methods-use-this
|
|
|
|
highlightTextMatches(text, term) {
|
|
|
|
const occurrences = fuzzaldrinPlus.match(text, term);
|
|
|
|
const { indexOf } = [];
|
|
|
|
|
|
|
|
return text
|
|
|
|
.split('')
|
|
|
|
.map((character, i) => {
|
|
|
|
if (indexOf.call(occurrences, i) !== -1) {
|
|
|
|
return `<b>${character}</b>`;
|
|
|
|
}
|
|
|
|
return character;
|
|
|
|
})
|
|
|
|
.join('');
|
|
|
|
}
|
|
|
|
|
|
|
|
// eslint-disable-next-line class-methods-use-this
|
|
|
|
noResults() {
|
|
|
|
return '<li class="dropdown-menu-empty-item"><a>No matching results</a></li>';
|
|
|
|
}
|
|
|
|
|
|
|
|
rowClicked(el) {
|
|
|
|
let field;
|
|
|
|
let groupName;
|
|
|
|
let selectedIndex;
|
|
|
|
let selectedObject;
|
|
|
|
let isMarking;
|
|
|
|
const { fieldName } = this.options;
|
|
|
|
const isInput = $(this.el).is('input');
|
|
|
|
if (this.renderedData) {
|
|
|
|
groupName = el.data('group');
|
|
|
|
if (groupName) {
|
|
|
|
selectedIndex = el.data('index');
|
2021-02-11 13:09:10 -05:00
|
|
|
this.selectedIndex = selectedIndex;
|
2020-08-25 20:10:31 -04:00
|
|
|
selectedObject = this.renderedData[groupName][selectedIndex];
|
|
|
|
} else {
|
|
|
|
selectedIndex = el.closest('li').index();
|
|
|
|
this.selectedIndex = selectedIndex;
|
|
|
|
selectedObject = this.renderedData[selectedIndex];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.vue) {
|
|
|
|
if (el.hasClass(ACTIVE_CLASS)) {
|
|
|
|
el.removeClass(ACTIVE_CLASS);
|
|
|
|
} else {
|
|
|
|
el.addClass(ACTIVE_CLASS);
|
|
|
|
}
|
|
|
|
|
|
|
|
return [selectedObject];
|
|
|
|
}
|
|
|
|
|
|
|
|
field = [];
|
|
|
|
const value = this.options.id ? this.options.id(selectedObject, el) : selectedObject.id;
|
|
|
|
if (isInput) {
|
|
|
|
field = $(this.el);
|
|
|
|
} else if (value != null) {
|
|
|
|
field = this.dropdown
|
|
|
|
.parent()
|
|
|
|
.find(`input[name='${fieldName}'][value='${value.toString().replace(/'/g, "\\'")}']`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.isSelectable && !this.options.isSelectable(selectedObject, el)) {
|
|
|
|
return [selectedObject];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (el.hasClass(ACTIVE_CLASS) && value !== 0) {
|
|
|
|
isMarking = false;
|
|
|
|
el.removeClass(ACTIVE_CLASS);
|
|
|
|
if (field && field.length) {
|
|
|
|
this.clearField(field, isInput);
|
|
|
|
}
|
|
|
|
} else if (el.hasClass(INDETERMINATE_CLASS)) {
|
|
|
|
isMarking = true;
|
|
|
|
el.addClass(ACTIVE_CLASS);
|
|
|
|
el.removeClass(INDETERMINATE_CLASS);
|
|
|
|
if (field && field.length && value == null) {
|
|
|
|
this.clearField(field, isInput);
|
|
|
|
}
|
|
|
|
if ((!field || !field.length) && fieldName) {
|
|
|
|
this.addInput(fieldName, value, selectedObject);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
isMarking = true;
|
|
|
|
if (!this.options.multiSelect || el.hasClass('dropdown-clear-active')) {
|
|
|
|
this.dropdown.find(`.${ACTIVE_CLASS}`).removeClass(ACTIVE_CLASS);
|
|
|
|
if (!isInput) {
|
2020-12-23 07:10:26 -05:00
|
|
|
this.dropdown.parent().find(`input[name='${fieldName}']`).remove();
|
2020-08-25 20:10:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (field && field.length && value == null) {
|
|
|
|
this.clearField(field, isInput);
|
|
|
|
}
|
|
|
|
// Toggle active class for the tick mark
|
|
|
|
el.addClass(ACTIVE_CLASS);
|
|
|
|
if (value != null) {
|
|
|
|
if ((!field || !field.length) && fieldName) {
|
|
|
|
this.addInput(fieldName, value, selectedObject);
|
|
|
|
} else if (field && field.length) {
|
|
|
|
field.val(value).trigger('change');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return [selectedObject, isMarking];
|
|
|
|
}
|
|
|
|
|
|
|
|
focusTextInput() {
|
|
|
|
if (this.options.filterable) {
|
|
|
|
const initialScrollTop = $(window).scrollTop();
|
|
|
|
|
|
|
|
if (this.dropdown.is('.show') && !this.filterInput.is(':focus')) {
|
|
|
|
this.filterInput.focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($(window).scrollTop() < initialScrollTop) {
|
|
|
|
$(window).scrollTop(initialScrollTop);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
addInput(fieldName, value, selectedObject, single) {
|
|
|
|
// Create hidden input for form
|
|
|
|
if (single) {
|
|
|
|
$(`input[name="${fieldName}"]`).remove();
|
|
|
|
}
|
|
|
|
|
2020-12-23 07:10:26 -05:00
|
|
|
const $input = $('<input>').attr('type', 'hidden').attr('name', fieldName).val(value);
|
2020-08-25 20:10:31 -04:00
|
|
|
if (this.options.inputId != null) {
|
|
|
|
$input.attr('id', this.options.inputId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.multiSelect) {
|
2020-12-23 16:10:24 -05:00
|
|
|
Object.keys(selectedObject).forEach((attribute) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
$input.attr(`data-${attribute}`, selectedObject[attribute]);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.inputMeta) {
|
|
|
|
$input.attr('data-meta', selectedObject[this.options.inputMeta]);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.dropdown.before($input).trigger('change');
|
|
|
|
}
|
|
|
|
|
|
|
|
selectRowAtIndex(index) {
|
|
|
|
// If we pass an option index
|
|
|
|
let selector;
|
|
|
|
if (typeof index !== 'undefined') {
|
|
|
|
selector = `${SELECTABLE_CLASSES}:eq(${index}) a`;
|
|
|
|
} else {
|
|
|
|
selector = '.dropdown-content .is-focused';
|
|
|
|
}
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
selector = `.dropdown-page-one ${selector}`;
|
|
|
|
}
|
|
|
|
// simulate a click on the first link
|
|
|
|
const $el = $(selector, this.dropdown);
|
|
|
|
if ($el.length) {
|
|
|
|
const href = $el.attr('href');
|
|
|
|
if (href && href !== '#') {
|
|
|
|
visitUrl(href);
|
|
|
|
} else {
|
|
|
|
$el.trigger('click');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
addArrowKeyEvent() {
|
|
|
|
const ARROW_KEY_CODES = [38, 40];
|
|
|
|
let selector = SELECTABLE_CLASSES;
|
|
|
|
if (this.dropdown.find('.dropdown-toggle-page').length) {
|
|
|
|
selector = `.dropdown-page-one ${selector}`;
|
|
|
|
}
|
2020-12-23 16:10:24 -05:00
|
|
|
return $('body').on('keydown', (e) => {
|
2020-08-25 20:10:31 -04:00
|
|
|
let $listItems;
|
|
|
|
let PREV_INDEX;
|
|
|
|
const currentKeyCode = e.which;
|
|
|
|
if (ARROW_KEY_CODES.indexOf(currentKeyCode) !== -1) {
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopImmediatePropagation();
|
|
|
|
PREV_INDEX = currentIndex;
|
|
|
|
$listItems = $(selector, this.dropdown);
|
|
|
|
// if @options.filterable
|
|
|
|
// $input.blur()
|
|
|
|
if (currentKeyCode === 40) {
|
|
|
|
// Move down
|
|
|
|
if (currentIndex < $listItems.length - 1) {
|
|
|
|
currentIndex += 1;
|
|
|
|
}
|
|
|
|
} else if (currentKeyCode === 38) {
|
|
|
|
// Move up
|
|
|
|
if (currentIndex > 0) {
|
|
|
|
currentIndex -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (currentIndex !== PREV_INDEX) {
|
|
|
|
this.highlightRowAtIndex($listItems, currentIndex);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (currentKeyCode === 13 && currentIndex !== -1) {
|
|
|
|
e.preventDefault();
|
|
|
|
this.selectRowAtIndex();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// eslint-disable-next-line class-methods-use-this
|
|
|
|
removeArrowKeyEvent() {
|
2020-12-08 16:10:06 -05:00
|
|
|
// eslint-disable-next-line @gitlab/no-global-event-off
|
2020-08-25 20:10:31 -04:00
|
|
|
return $('body').off('keydown');
|
|
|
|
}
|
|
|
|
|
|
|
|
resetRows() {
|
|
|
|
currentIndex = -1;
|
|
|
|
$('.is-focused', this.dropdown).removeClass('is-focused');
|
|
|
|
}
|
|
|
|
|
|
|
|
highlightRowAtIndex($listItems, index) {
|
|
|
|
if (!$listItems) {
|
|
|
|
// eslint-disable-next-line no-param-reassign
|
|
|
|
$listItems = $(SELECTABLE_CLASSES, this.dropdown);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove the class for the previously focused row
|
|
|
|
$('.is-focused', this.dropdown).removeClass('is-focused');
|
|
|
|
// Update the class for the row at the specific index
|
|
|
|
const $listItem = $listItems.eq(index);
|
|
|
|
$listItem.find('a:first-child').addClass('is-focused');
|
|
|
|
// Dropdown content scroll area
|
|
|
|
const $dropdownContent = $listItem.closest('.dropdown-content');
|
|
|
|
const dropdownScrollTop = $dropdownContent.scrollTop();
|
|
|
|
const dropdownContentHeight = $dropdownContent.outerHeight();
|
|
|
|
const dropdownContentTop = $dropdownContent.prop('offsetTop');
|
|
|
|
const dropdownContentBottom = dropdownContentTop + dropdownContentHeight;
|
|
|
|
// Get the offset bottom of the list item
|
|
|
|
const listItemHeight = $listItem.outerHeight();
|
|
|
|
const listItemTop = $listItem.prop('offsetTop');
|
|
|
|
const listItemBottom = listItemTop + listItemHeight;
|
|
|
|
if (!index) {
|
|
|
|
// Scroll the dropdown content to the top
|
|
|
|
$dropdownContent.scrollTop(0);
|
|
|
|
} else if (index === $listItems.length - 1) {
|
|
|
|
// Scroll the dropdown content to the bottom
|
|
|
|
$dropdownContent.scrollTop($dropdownContent.prop('scrollHeight'));
|
|
|
|
} else if (listItemBottom > dropdownContentBottom + dropdownScrollTop) {
|
|
|
|
// Scroll the dropdown content down
|
|
|
|
$dropdownContent.scrollTop(
|
|
|
|
listItemBottom - dropdownContentBottom + CURSOR_SELECT_SCROLL_PADDING,
|
|
|
|
);
|
|
|
|
} else if (listItemTop < dropdownContentTop + dropdownScrollTop) {
|
|
|
|
// Scroll the dropdown content up
|
|
|
|
return $dropdownContent.scrollTop(
|
|
|
|
listItemTop - dropdownContentTop - CURSOR_SELECT_SCROLL_PADDING,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
updateLabel(selected = null, el = null, instance = null) {
|
|
|
|
let toggleText = this.options.toggleLabel(selected, el, instance);
|
|
|
|
if (this.options.updateLabel) {
|
|
|
|
// Option to override the dropdown label text
|
|
|
|
toggleText = this.options.updateLabel;
|
|
|
|
}
|
|
|
|
|
2020-12-23 07:10:26 -05:00
|
|
|
return $(this.el).find('.dropdown-toggle-text').text(toggleText);
|
2020-08-25 20:10:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// eslint-disable-next-line class-methods-use-this
|
|
|
|
clearField(field, isInput) {
|
|
|
|
return isInput ? field.val('') : field.remove();
|
|
|
|
}
|
|
|
|
}
|