/* eslint-disable func-names, one-var, no-var, prefer-rest-params, vars-on-top, prefer-arrow-callback, consistent-return, object-shorthand, no-shadow, no-unused-vars, no-else-return, no-self-compare, prefer-template, no-unused-expressions, yoda, prefer-spread, camelcase, no-param-reassign */ /* global Issuable */ /* global emitSidebarEvent */ import $ from 'jquery'; import _ from 'underscore'; import axios from './lib/utils/axios_utils'; import { s__, __, sprintf } from './locale'; import ModalStore from './boards/stores/modal_store'; // TODO: remove eventHub hack after code splitting refactor window.emitSidebarEvent = window.emitSidebarEvent || $.noop; function UsersSelect(currentUser, els, options = {}) { var $els; this.users = this.users.bind(this); this.user = this.user.bind(this); this.usersPath = '/autocomplete/users.json'; this.userPath = '/autocomplete/users/:id.json'; if (currentUser != null) { if (typeof currentUser === 'object') { this.currentUser = currentUser; } else { this.currentUser = JSON.parse(currentUser); } } const { handleClick } = options; $els = $(els); if (!els) { $els = $('.js-user-search'); } $els.each( (function(_this) { return function(i, dropdown) { var options = {}; var $block, $collapsedSidebar, $dropdown, $loading, $selectbox, $value, abilityName, assignTo, assigneeTemplate, collapsedAssigneeTemplate, defaultLabel, defaultNullUser, firstUser, issueURL, selectedId, selectedIdDefault, showAnyUser, showNullUser, showMenuAbove; $dropdown = $(dropdown); options.projectId = $dropdown.data('projectId'); options.groupId = $dropdown.data('groupId'); options.showCurrentUser = $dropdown.data('currentUser'); options.todoFilter = $dropdown.data('todoFilter'); options.todoStateFilter = $dropdown.data('todoStateFilter'); options.iid = $dropdown.data('iid'); options.issuableType = $dropdown.data('issuableType'); showNullUser = $dropdown.data('nullUser'); defaultNullUser = $dropdown.data('nullUserDefault'); showMenuAbove = $dropdown.data('showMenuAbove'); showAnyUser = $dropdown.data('anyUser'); firstUser = $dropdown.data('firstUser'); options.authorId = $dropdown.data('authorId'); defaultLabel = $dropdown.data('defaultLabel'); issueURL = $dropdown.data('issueUpdate'); $selectbox = $dropdown.closest('.selectbox'); $block = $selectbox.closest('.block'); abilityName = $dropdown.data('abilityName'); $value = $block.find('.value'); $collapsedSidebar = $block.find('.sidebar-collapsed-user'); $loading = $block.find('.block-loading').fadeOut(); selectedIdDefault = defaultNullUser && showNullUser ? 0 : null; selectedId = $dropdown.data('selected'); if (selectedId === undefined) { selectedId = selectedIdDefault; } const assignYourself = function() { const unassignedSelected = $dropdown .closest('.selectbox') .find(`input[name='${$dropdown.data('fieldName')}'][value=0]`); if (unassignedSelected) { unassignedSelected.remove(); } // Save current selected user to the DOM const currentUserInfo = $dropdown.data('currentUserInfo') || {}; const currentUser = _this.currentUser || {}; const fieldName = $dropdown.data('fieldName'); const userName = currentUserInfo.name; const userId = currentUserInfo.id || currentUser.id; const inputHtmlString = _.template(` `)({ fieldName, userName, userId }); if ($selectbox) { $dropdown.parent().before(inputHtmlString); } else { $dropdown.after(inputHtmlString); } }; if ($block[0]) { $block[0].addEventListener('assignYourself', assignYourself); } const getSelectedUserInputs = function() { return $selectbox.find(`input[name="${$dropdown.data('fieldName')}"]`); }; const getSelected = function() { return getSelectedUserInputs() .map((index, input) => parseInt(input.value, 10)) .get(); }; const checkMaxSelect = function() { const maxSelect = $dropdown.data('maxSelect'); if (maxSelect) { const selected = getSelected(); if (selected.length > maxSelect) { const firstSelectedId = selected[0]; const firstSelected = $dropdown .closest('.selectbox') .find(`input[name='${$dropdown.data('fieldName')}'][value=${firstSelectedId}]`); firstSelected.remove(); emitSidebarEvent('sidebar.removeAssignee', { id: firstSelectedId, }); } } }; const getMultiSelectDropdownTitle = function(selectedUser, isSelected) { const selectedUsers = getSelected().filter(u => u !== 0); const firstUser = getSelectedUserInputs() .map((index, input) => ({ name: input.dataset.meta, value: parseInt(input.value, 10), })) .filter(u => u.id !== 0) .get(0); if (selectedUsers.length === 0) { return s__('UsersSelect|Unassigned'); } else if (selectedUsers.length === 1) { return firstUser.name; } else if (isSelected) { const otherSelected = selectedUsers.filter(s => s !== selectedUser.id); return sprintf(s__('UsersSelect|%{name} + %{length} more'), { name: selectedUser.name, length: otherSelected.length, }); } else { return sprintf(s__('UsersSelect|%{name} + %{length} more'), { name: firstUser.name, length: selectedUsers.length - 1, }); } }; $('.assign-to-me-link').on('click', e => { e.preventDefault(); $(e.currentTarget).hide(); if ($dropdown.data('multiSelect')) { assignYourself(); checkMaxSelect(); const currentUserInfo = $dropdown.data('currentUserInfo'); $dropdown .find('.dropdown-toggle-text') .text(getMultiSelectDropdownTitle(currentUserInfo)) .removeClass('is-default'); } else { const $input = $(`input[name="${$dropdown.data('fieldName')}"]`); $input.val(gon.current_user_id); selectedId = $input.val(); $dropdown .find('.dropdown-toggle-text') .text(gon.current_user_fullname) .removeClass('is-default'); } }); $block.on('click', '.js-assign-yourself', e => { e.preventDefault(); return assignTo(_this.currentUser.id); }); assignTo = function(selected) { var data; data = {}; data[abilityName] = {}; data[abilityName].assignee_id = selected != null ? selected : null; $loading.removeClass('hidden').fadeIn(); $dropdown.trigger('loading.gl.dropdown'); return axios.put(issueURL, data).then(({ data }) => { var user, tooltipTitle; $dropdown.trigger('loaded.gl.dropdown'); $loading.fadeOut(); if (data.assignee) { user = { name: data.assignee.name, username: data.assignee.username, avatar: data.assignee.avatar_url, }; tooltipTitle = _.escape(user.name); } else { user = { name: s__('UsersSelect|Unassigned'), username: '', avatar: '', }; tooltipTitle = s__('UsersSelect|Assignee'); } $value.html(assigneeTemplate(user)); $collapsedSidebar.attr('title', tooltipTitle).tooltip('_fixTitle'); return $collapsedSidebar.html(collapsedAssigneeTemplate(user)); }); }; collapsedAssigneeTemplate = _.template( '<% if( avatar ) { %> <% } else { %> <% } %>', ); assigneeTemplate = _.template( `<% if (username) { %> <% if( avatar ) { %> <% } %> <%- name %> @<%- username %> <% } else { %> ${sprintf(s__('UsersSelect|No assignee - %{openingTag} assign yourself %{closingTag}'), { openingTag: '', closingTag: '', })} <% } %>`, ); return $dropdown.glDropdown({ showMenuAbove: showMenuAbove, data: function(term, callback) { return _this.users( term, options, function(users) { // GitLabDropdownFilter returns this.instance // GitLabDropdownRemote returns this.options.instance const glDropdown = this.instance || this.options.instance; glDropdown.options.processData(term, users, callback); }.bind(this), ); }, processData: function(term, data, callback) { let users = data; // Only show assigned user list when there is no search term if ($dropdown.hasClass('js-multiselect') && term.length === 0) { const selectedInputs = getSelectedUserInputs(); // Potential duplicate entries when dealing with issue board // because issue board is also managed by vue const selectedUsers = _.uniq(selectedInputs, false, a => a.value) .filter(input => { const userId = parseInt(input.value, 10); const inUsersArray = users.find(u => u.id === userId); return !inUsersArray && userId !== 0; }) .map(input => { const userId = parseInt(input.value, 10); const { avatarUrl, avatar_url, name, username } = input.dataset; return { avatar_url: avatarUrl || avatar_url, id: userId, name, username, }; }); users = data.concat(selectedUsers); } let anyUser; let index; let len; let name; let obj; let showDivider; if (term.length === 0) { showDivider = 0; if (firstUser) { // Move current user to the front of the list for (index = 0, len = users.length; index < len; index += 1) { obj = users[index]; if (obj.username === firstUser) { users.splice(index, 1); users.unshift(obj); break; } } } if (showNullUser) { showDivider += 1; users.unshift({ beforeDivider: true, name: s__('UsersSelect|Unassigned'), id: 0, }); } if (showAnyUser) { showDivider += 1; name = showAnyUser; if (name === true) { name = s__('UsersSelect|Any User'); } anyUser = { beforeDivider: true, name: name, id: null, }; users.unshift(anyUser); } if (showDivider) { users.splice(showDivider, 0, 'divider'); } if ($dropdown.hasClass('js-multiselect')) { const selected = getSelected().filter(i => i !== 0); if (selected.length > 0) { if ($dropdown.data('dropdownHeader')) { showDivider += 1; users.splice(showDivider, 0, { header: $dropdown.data('dropdownHeader'), }); } const selectedUsers = users .filter(u => selected.indexOf(u.id) !== -1) .sort((a, b) => a.name > b.name); users = users.filter(u => selected.indexOf(u.id) === -1); selectedUsers.forEach(selectedUser => { showDivider += 1; users.splice(showDivider, 0, selectedUser); }); users.splice(showDivider + 1, 0, 'divider'); } } } callback(users); if (showMenuAbove) { $dropdown.data('glDropdown').positionMenuAbove(); } }, filterable: true, filterRemote: true, search: { fields: ['name', 'username'], }, selectable: true, fieldName: $dropdown.data('fieldName'), toggleLabel: function(selected, el, glDropdown) { const inputValue = glDropdown.filterInput.val(); if (this.multiSelect && inputValue === '') { // Remove non-users from the fullData array const users = glDropdown.filteredFullData(); const callback = glDropdown.parseData.bind(glDropdown); // Update the data model this.processData(inputValue, users, callback); } if (this.multiSelect) { return getMultiSelectDropdownTitle(selected, $(el).hasClass('is-active')); } if (selected && 'id' in selected && $(el).hasClass('is-active')) { $dropdown.find('.dropdown-toggle-text').removeClass('is-default'); if (selected.text) { return selected.text; } else { return selected.name; } } else { $dropdown.find('.dropdown-toggle-text').addClass('is-default'); return defaultLabel; } }, defaultLabel: defaultLabel, hidden: function(e) { if ($dropdown.hasClass('js-multiselect')) { emitSidebarEvent('sidebar.saveAssignees'); } if (!$dropdown.data('alwaysShowSelectbox')) { $selectbox.hide(); // Recalculate where .value is because vue might have changed it $block = $selectbox.closest('.block'); $value = $block.find('.value'); // display:block overrides the hide-collapse rule $value.css('display', ''); } }, multiSelect: $dropdown.hasClass('js-multiselect'), inputMeta: $dropdown.data('inputMeta'), clicked: function(options) { const { $el, e, isMarking } = options; const user = options.selectedObj; $el.tooltip('dispose'); if ($dropdown.hasClass('js-multiselect')) { const isActive = $el.hasClass('is-active'); const previouslySelected = $dropdown .closest('.selectbox') /* eslint-disable-next-line @gitlab/i18n/no-non-i18n-strings */ .find("input[name='" + $dropdown.data('fieldName') + "'][value!=0]"); // Enables support for limiting the number of users selected // Automatically removes the first on the list if more users are selected checkMaxSelect(); if (user.beforeDivider && user.name.toLowerCase() === 'unassigned') { // Unassigned selected previouslySelected.each((index, element) => { const id = parseInt(element.value, 10); element.remove(); }); emitSidebarEvent('sidebar.removeAllAssignees'); } else if (isActive) { // user selected emitSidebarEvent('sidebar.addAssignee', user); // Remove unassigned selection (if it was previously selected) const unassignedSelected = $dropdown .closest('.selectbox') .find("input[name='" + $dropdown.data('fieldName') + "'][value=0]"); if (unassignedSelected) { unassignedSelected.remove(); } } else { if (previouslySelected.length === 0) { // Select unassigned because there is no more selected users this.addInput($dropdown.data('fieldName'), 0, {}); } // User unselected emitSidebarEvent('sidebar.removeAssignee', user); } if (getSelected().find(u => u === gon.current_user_id)) { $('.assign-to-me-link').hide(); } else { $('.assign-to-me-link').show(); } } var isIssueIndex, isMRIndex, page, selected; page = $('body').attr('data-page'); isIssueIndex = page === 'projects:issues:index'; isMRIndex = page === page && page === 'projects:merge_requests:index'; if ( $dropdown.hasClass('js-filter-bulk-update') || $dropdown.hasClass('js-issuable-form-dropdown') ) { e.preventDefault(); const isSelecting = user.id !== selectedId; selectedId = isSelecting ? user.id : selectedIdDefault; if (selectedId === gon.current_user_id) { $('.assign-to-me-link').hide(); } else { $('.assign-to-me-link').show(); } return; } if ($el.closest('.add-issues-modal').length) { ModalStore.store.filter[$dropdown.data('fieldName')] = user.id; } else if (handleClick) { e.preventDefault(); handleClick(user, isMarking); } else if ($dropdown.hasClass('js-filter-submit') && (isIssueIndex || isMRIndex)) { return Issuable.filterResults($dropdown.closest('form')); } else if ($dropdown.hasClass('js-filter-submit')) { return $dropdown.closest('form').submit(); } else if (!$dropdown.hasClass('js-multiselect')) { selected = $dropdown .closest('.selectbox') .find("input[name='" + $dropdown.data('fieldName') + "']") .val(); return assignTo(selected); } // Automatically close dropdown after assignee is selected // since CE has no multiple assignees // EE does not have a max-select if ( $dropdown.data('maxSelect') && getSelected().length === $dropdown.data('maxSelect') ) { // Close the dropdown $dropdown.dropdown('toggle'); } }, id: function(user) { return user.id; }, opened: function(e) { const $el = $(e.currentTarget); const selected = getSelected(); if ($dropdown.hasClass('js-issue-board-sidebar') && selected.length === 0) { this.addInput($dropdown.data('fieldName'), 0, {}); } $el.find('.is-active').removeClass('is-active'); function highlightSelected(id) { $el.find(`li[data-user-id="${id}"] .dropdown-menu-user-link`).addClass('is-active'); } if (selected.length > 0) { getSelected().forEach(selectedId => highlightSelected(selectedId)); } else if ($dropdown.hasClass('js-issue-board-sidebar')) { highlightSelected(0); } else { highlightSelected(selectedId); } }, updateLabel: $dropdown.data('dropdownTitle'), renderRow: function(user) { var avatar, img, listClosingTags, listWithName, listWithUserName, username; username = user.username ? '@' + user.username : ''; avatar = user.avatar_url ? user.avatar_url : gon.default_avatar_url; let selected = false; if (this.multiSelect) { selected = getSelected().find(u => user.id === u); const { fieldName } = this; const field = $dropdown .closest('.selectbox') .find("input[name='" + fieldName + "'][value='" + user.id + "']"); if (field.length) { selected = true; } } else { selected = user.id === selectedId; } img = ''; if (user.beforeDivider != null) { `
  • ${_.escape( user.name, )}
  • `; } else { // 0 margin, because it's now handled by a wrapper img = ""; } return _this.renderRow(options.issuableType, user, selected, username, img); }, }); }; })(this), ); import(/* webpackChunkName: 'select2' */ 'select2/select2') .then(() => { $('.ajax-users-select').each( (function(_this) { return function(i, select) { var firstUser, showAnyUser, showEmailUser, showNullUser; var options = {}; options.skipLdap = $(select).hasClass('skip_ldap'); options.projectId = $(select).data('projectId'); options.groupId = $(select).data('groupId'); options.showCurrentUser = $(select).data('currentUser'); options.authorId = $(select).data('authorId'); options.skipUsers = $(select).data('skipUsers'); showNullUser = $(select).data('nullUser'); showAnyUser = $(select).data('anyUser'); showEmailUser = $(select).data('emailUser'); firstUser = $(select).data('firstUser'); return $(select).select2({ placeholder: __('Search for a user'), multiple: $(select).hasClass('multiselect'), minimumInputLength: 0, query: function(query) { return _this.users(query.term, options, function(users) { var anyUser, data, emailUser, index, len, name, nullUser, obj, ref; data = { results: users, }; if (query.term.length === 0) { if (firstUser) { // Move current user to the front of the list ref = data.results; for (index = 0, len = ref.length; index < len; index += 1) { obj = ref[index]; if (obj.username === firstUser) { data.results.splice(index, 1); data.results.unshift(obj); break; } } } if (showNullUser) { nullUser = { name: s__('UsersSelect|Unassigned'), id: 0, }; data.results.unshift(nullUser); } if (showAnyUser) { name = showAnyUser; if (name === true) { name = s__('UsersSelect|Any User'); } anyUser = { name: name, id: null, }; data.results.unshift(anyUser); } } if ( showEmailUser && data.results.length === 0 && query.term.match(/^[^@]+@[^@]+$/) ) { var trimmed = query.term.trim(); emailUser = { name: sprintf(__('Invite "%{trimmed}" by email'), { trimmed }), username: trimmed, id: trimmed, invite: true, }; data.results.unshift(emailUser); } return query.callback(data); }); }, initSelection: function() { var args; args = 1 <= arguments.length ? [].slice.call(arguments, 0) : []; return _this.initSelection.apply(_this, args); }, formatResult: function() { var args; args = 1 <= arguments.length ? [].slice.call(arguments, 0) : []; return _this.formatResult.apply(_this, args); }, formatSelection: function() { var args; args = 1 <= arguments.length ? [].slice.call(arguments, 0) : []; return _this.formatSelection.apply(_this, args); }, dropdownCssClass: 'ajax-users-dropdown', // we do not want to escape markup since we are displaying html in results escapeMarkup: function(m) { return m; }, }); }; })(this), ); }) .catch(() => {}); } UsersSelect.prototype.initSelection = function(element, callback) { var id, nullUser; id = $(element).val(); if (id === '0') { nullUser = { name: s__('UsersSelect|Unassigned'), }; return callback(nullUser); } else if (id !== '') { return this.user(id, callback); } }; UsersSelect.prototype.formatResult = function(user) { var avatar; if (user.avatar_url) { avatar = user.avatar_url; } else { avatar = gon.default_avatar_url; } return `
    `; }; UsersSelect.prototype.formatSelection = function(user) { return _.escape(user.name); }; UsersSelect.prototype.user = function(user_id, callback) { if (!/^\d+$/.test(user_id)) { return false; } var url; url = this.buildUrl(this.userPath); url = url.replace(':id', user_id); return axios.get(url).then(({ data }) => { callback(data); }); }; // Return users list. Filtered by query // Only active users retrieved UsersSelect.prototype.users = function(query, options, callback) { const url = this.buildUrl(this.usersPath); const params = { search: query, active: true, project_id: options.projectId || null, group_id: options.groupId || null, skip_ldap: options.skipLdap || null, todo_filter: options.todoFilter || null, todo_state_filter: options.todoStateFilter || null, current_user: options.showCurrentUser || null, author_id: options.authorId || null, skip_users: options.skipUsers || null, }; if (options.issuableType === 'merge_request') { params.merge_request_iid = options.iid || null; } return axios.get(url, { params }).then(({ data }) => { callback(data); }); }; UsersSelect.prototype.buildUrl = function(url) { if (gon.relative_url_root != null) { url = gon.relative_url_root.replace(/\/$/, '') + url; } return url; }; UsersSelect.prototype.renderRow = function(issuableType, user, selected, username, img) { const tooltip = issuableType === 'merge_request' && !user.can_merge ? __('Cannot merge') : ''; const tooltipClass = tooltip ? `has-tooltip` : ''; const selectedClass = selected === true ? 'is-active' : ''; const linkClasses = `${selectedClass} ${tooltipClass}`; const tooltipAttributes = tooltip ? `data-container="body" data-placement="left" data-title="${tooltip}"` : ''; return `
  • ${this.renderRowAvatar(issuableType, user, img)} ${_.escape(user.name)} ${username ? `${username}` : ''}
  • `; }; UsersSelect.prototype.renderRowAvatar = function(issuableType, user, img) { if (user.beforeDivider) { return img; } const mergeIcon = issuableType === 'merge_request' && !user.can_merge ? '' : ''; return ` ${img} ${mergeIcon} `; }; export default UsersSelect;