gitlab-org--gitlab-foss/app/assets/javascripts/milestone_select.js

255 lines
9.4 KiB
JavaScript
Raw Normal View History

2017-12-28 20:07:05 +00:00
/* eslint-disable func-names, space-before-function-paren, wrap-iife, no-underscore-dangle, prefer-arrow-callback, max-len, one-var, one-var-declaration-per-line, no-unused-vars, object-shorthand, comma-dangle, no-else-return, no-self-compare, consistent-return, no-param-reassign, no-shadow */
/* global Issuable */
/* global ListMilestone */
import $ from 'jquery';
import _ from 'underscore';
import { __ } from '~/locale';
2018-01-31 09:34:25 +00:00
import axios from './lib/utils/axios_utils';
import { timeFor } from './lib/utils/datetime_utility';
import ModalStore from './boards/stores/modal_store';
2017-12-28 20:07:05 +00:00
export default class MilestoneSelect {
constructor(currentProject, els, options = {}) {
if (currentProject !== null) {
this.currentProject =
typeof currentProject === 'string' ? JSON.parse(currentProject) : currentProject;
2017-12-28 20:07:05 +00:00
}
2017-02-01 17:02:17 +00:00
2017-12-28 20:07:05 +00:00
this.init(els, options);
}
2017-02-01 17:02:17 +00:00
2017-12-28 20:07:05 +00:00
init(els, options) {
let $els = $(els);
2017-02-01 17:02:17 +00:00
2017-12-28 20:07:05 +00:00
if (!els) {
$els = $('.js-milestone-select');
}
2017-12-28 20:07:05 +00:00
$els.each((i, dropdown) => {
let milestoneLinkNoneTemplate,
milestoneLinkTemplate,
selectedMilestone,
selectedMilestoneDefault;
2017-12-28 20:07:05 +00:00
const $dropdown = $(dropdown);
2018-02-20 22:20:48 +00:00
const projectId = $dropdown.data('projectId');
2017-12-28 20:07:05 +00:00
const milestonesUrl = $dropdown.data('milestones');
const issueUpdateURL = $dropdown.data('issueUpdate');
2018-02-20 22:20:48 +00:00
const showNo = $dropdown.data('showNo');
const showAny = $dropdown.data('showAny');
2017-12-28 20:07:05 +00:00
const showMenuAbove = $dropdown.data('showMenuAbove');
2018-02-20 22:20:48 +00:00
const showUpcoming = $dropdown.data('showUpcoming');
const showStarted = $dropdown.data('showStarted');
const useId = $dropdown.data('useId');
const defaultLabel = $dropdown.data('defaultLabel');
const defaultNo = $dropdown.data('defaultNo');
const issuableId = $dropdown.data('issuableId');
const abilityName = $dropdown.data('abilityName');
2017-12-28 20:07:05 +00:00
const $selectBox = $dropdown.closest('.selectbox');
const $block = $selectBox.closest('.block');
const $sidebarCollapsedValue = $block.find('.sidebar-collapsed-icon');
const $value = $block.find('.value');
const $loading = $block.find('.block-loading').fadeOut();
selectedMilestoneDefault = showAny ? '' : null;
selectedMilestoneDefault = showNo && defaultNo ? 'No Milestone' : selectedMilestoneDefault;
2017-12-28 20:07:05 +00:00
selectedMilestone = $dropdown.data('selected') || selectedMilestoneDefault;
2017-12-28 20:07:05 +00:00
if (issueUpdateURL) {
milestoneLinkTemplate = _.template(
'<a href="<%- web_url %>" class="bold has-tooltip" data-container="body" title="<%- remaining %>"><%- title %></a>',
);
2017-12-28 20:07:05 +00:00
milestoneLinkNoneTemplate = '<span class="no-value">None</span>';
}
return $dropdown.glDropdown({
showMenuAbove: showMenuAbove,
data: (term, callback) =>
axios.get(milestonesUrl).then(({ data }) => {
2018-01-31 09:34:25 +00:00
const extraOptions = [];
if (showAny) {
extraOptions.push({
id: null,
name: null,
title: 'Any Milestone',
2018-01-31 09:34:25 +00:00
});
}
if (showNo) {
extraOptions.push({
id: -1,
name: 'No Milestone',
title: 'No Milestone',
2018-01-31 09:34:25 +00:00
});
}
if (showUpcoming) {
extraOptions.push({
id: -2,
name: '#upcoming',
title: 'Upcoming',
2018-01-31 09:34:25 +00:00
});
}
if (showStarted) {
extraOptions.push({
id: -3,
name: '#started',
title: 'Started',
2018-01-31 09:34:25 +00:00
});
}
if (extraOptions.length) {
extraOptions.push('divider');
}
2018-01-31 09:34:25 +00:00
callback(extraOptions.concat(data));
if (showMenuAbove) {
$dropdown.data('glDropdown').positionMenuAbove();
}
$(`[data-milestone-id="${_.escape(selectedMilestone)}"] > a`).addClass('is-active');
2018-01-31 09:34:25 +00:00
}),
2017-12-28 20:07:05 +00:00
renderRow: milestone => `
<li data-milestone-id="${_.escape(milestone.name)}">
2017-12-28 20:07:05 +00:00
<a href='#' class='dropdown-menu-milestone-link'>
${_.escape(milestone.title)}
</a>
</li>
`,
filterable: true,
search: {
fields: ['title'],
2017-12-28 20:07:05 +00:00
},
selectable: true,
toggleLabel: (selected, el, e) => {
if (selected && 'id' in selected && $(el).hasClass('is-active')) {
return selected.title;
} else {
return defaultLabel;
}
},
defaultLabel: defaultLabel,
2018-02-20 22:20:48 +00:00
fieldName: $dropdown.data('fieldName'),
2017-12-28 20:07:05 +00:00
text: milestone => _.escape(milestone.title),
id: milestone => {
2017-12-28 20:07:05 +00:00
if (!useId && !$dropdown.is('.js-issuable-form-dropdown')) {
return milestone.name;
} else {
return milestone.id;
}
},
hidden: () => {
$selectBox.hide();
// display:block overrides the hide-collapse rule
return $value.css('display', '');
},
opened: e => {
2017-12-28 20:07:05 +00:00
const $el = $(e.currentTarget);
if ($dropdown.hasClass('js-issue-board-sidebar') || options.handleClick) {
selectedMilestone = $dropdown[0].dataset.selected || selectedMilestoneDefault;
}
$('a.is-active', $el).removeClass('is-active');
$(`[data-milestone-id="${_.escape(selectedMilestone)}"] > a`, $el).addClass('is-active');
2017-12-28 20:07:05 +00:00
},
vue: $dropdown.hasClass('js-issue-board-sidebar'),
clicked: clickEvent => {
2017-12-28 20:07:05 +00:00
const { $el, e } = clickEvent;
let selected = clickEvent.selectedObj;
2017-12-28 20:07:05 +00:00
let data, boardsStore;
if (!selected) return;
2017-02-01 17:02:17 +00:00
2017-12-28 20:07:05 +00:00
if (options.handleClick) {
e.preventDefault();
options.handleClick(selected);
return;
}
2017-02-01 17:02:17 +00:00
2017-12-28 20:07:05 +00:00
const page = $('body').attr('data-page');
const isIssueIndex = page === 'projects:issues:index';
const isMRIndex = page === page && page === 'projects:merge_requests:index';
const isSelecting = selected.name !== selectedMilestone;
2017-12-28 20:07:05 +00:00
selectedMilestone = isSelecting ? selected.name : selectedMilestoneDefault;
if (
$dropdown.hasClass('js-filter-bulk-update') ||
$dropdown.hasClass('js-issuable-form-dropdown')
) {
2017-12-28 20:07:05 +00:00
e.preventDefault();
return;
}
2017-12-28 20:07:05 +00:00
if ($dropdown.closest('.add-issues-modal').length) {
boardsStore = ModalStore.store.filter;
2017-12-28 20:07:05 +00:00
}
2017-12-28 20:07:05 +00:00
if (boardsStore) {
2018-02-20 22:20:48 +00:00
boardsStore[$dropdown.data('fieldName')] = selected.name;
2017-12-28 20:07:05 +00:00
e.preventDefault();
} 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-issue-board-sidebar')) {
if (selected.id !== -1 && isSelecting) {
gl.issueBoards.boardStoreIssueSet(
'milestone',
new ListMilestone({
id: selected.id,
title: selected.name,
}),
);
2016-07-24 20:45:11 +00:00
} else {
2017-12-28 20:07:05 +00:00
gl.issueBoards.boardStoreIssueDelete('milestone');
}
$dropdown.trigger('loading.gl.dropdown');
$loading.removeClass('hidden').fadeIn();
gl.issueBoards.BoardsStore.detail.issue
.update($dropdown.attr('data-issue-update'))
2017-12-28 20:07:05 +00:00
.then(() => {
2016-07-24 20:45:11 +00:00
$dropdown.trigger('loaded.gl.dropdown');
$loading.fadeOut();
2017-12-28 20:07:05 +00:00
})
.catch(() => {
$loading.fadeOut();
2016-07-24 20:45:11 +00:00
});
2017-12-28 20:07:05 +00:00
} else {
selected = $selectBox.find('input[type="hidden"]').val();
data = {};
data[abilityName] = {};
data[abilityName].milestone_id = selected != null ? selected : null;
$loading.removeClass('hidden').fadeIn();
$dropdown.trigger('loading.gl.dropdown');
return axios
.put(issueUpdateURL, data)
2018-01-31 09:34:25 +00:00
.then(({ data }) => {
$dropdown.trigger('loaded.gl.dropdown');
$loading.fadeOut();
$selectBox.hide();
$value.css('display', '');
if (data.milestone != null) {
data.milestone.full_path = this.currentProject.full_path;
data.milestone.remaining = timeFor(data.milestone.due_date);
data.milestone.name = data.milestone.title;
$value.html(milestoneLinkTemplate(data.milestone));
return $sidebarCollapsedValue
.attr(
'data-original-title',
`${data.milestone.name}<br />${data.milestone.remaining}`,
)
.find('span')
.text(data.milestone.title);
2018-01-31 09:34:25 +00:00
} else {
$value.html(milestoneLinkNoneTemplate);
return $sidebarCollapsedValue
.attr('data-original-title', __('Milestone'))
.find('span')
.text(__('None'));
2018-01-31 09:34:25 +00:00
}
})
.catch(() => {
$loading.fadeOut();
2018-01-31 09:34:25 +00:00
});
2016-07-24 20:45:11 +00:00
}
},
2016-07-24 20:45:11 +00:00
});
2017-12-28 20:07:05 +00:00
});
}
}