170 lines
5.2 KiB
JavaScript
170 lines
5.2 KiB
JavaScript
/* eslint-disable class-methods-use-this, no-unneeded-ternary */
|
|
|
|
import $ from 'jquery';
|
|
import { visitUrl } from '~/lib/utils/url_utility';
|
|
import UsersSelect from '~/users_select';
|
|
import { isMetaClick } from '~/lib/utils/common_utils';
|
|
import { __ } from '~/locale';
|
|
import flash from '~/flash';
|
|
import axios from '~/lib/utils/axios_utils';
|
|
|
|
export default class Todos {
|
|
constructor() {
|
|
this.initFilters();
|
|
this.bindEvents();
|
|
this.todo_ids = [];
|
|
|
|
this.cleanupWrapper = this.cleanup.bind(this);
|
|
document.addEventListener('beforeunload', this.cleanupWrapper);
|
|
}
|
|
|
|
cleanup() {
|
|
this.unbindEvents();
|
|
document.removeEventListener('beforeunload', this.cleanupWrapper);
|
|
}
|
|
|
|
unbindEvents() {
|
|
$('.js-done-todo, .js-undo-todo, .js-add-todo').off('click', this.updateRowStateClickedWrapper);
|
|
$('.js-todos-mark-all', '.js-todos-undo-all').off('click', this.updateallStateClickedWrapper);
|
|
$('.todo').off('click', this.goToTodoUrl);
|
|
}
|
|
|
|
bindEvents() {
|
|
this.updateRowStateClickedWrapper = this.updateRowStateClicked.bind(this);
|
|
this.updateAllStateClickedWrapper = this.updateAllStateClicked.bind(this);
|
|
|
|
$('.js-done-todo, .js-undo-todo, .js-add-todo').on('click', this.updateRowStateClickedWrapper);
|
|
$('.js-todos-mark-all, .js-todos-undo-all').on('click', this.updateAllStateClickedWrapper);
|
|
$('.todo').on('click', this.goToTodoUrl);
|
|
}
|
|
|
|
initFilters() {
|
|
this.initFilterDropdown($('.js-group-search'), 'group_id', ['text']);
|
|
this.initFilterDropdown($('.js-project-search'), 'project_id', ['text']);
|
|
this.initFilterDropdown($('.js-type-search'), 'type');
|
|
this.initFilterDropdown($('.js-action-search'), 'action_id');
|
|
|
|
return new UsersSelect();
|
|
}
|
|
|
|
initFilterDropdown($dropdown, fieldName, searchFields) {
|
|
$dropdown.glDropdown({
|
|
fieldName,
|
|
selectable: true,
|
|
filterable: searchFields ? true : false,
|
|
search: { fields: searchFields },
|
|
data: $dropdown.data('data'),
|
|
clicked: () => {
|
|
const $formEl = $dropdown.closest('form.filter-form');
|
|
const mutexDropdowns = {
|
|
group_id: 'project_id',
|
|
project_id: 'group_id',
|
|
};
|
|
|
|
$formEl.find(`input[name="${mutexDropdowns[fieldName]}"]`).remove();
|
|
$formEl.submit();
|
|
},
|
|
});
|
|
}
|
|
|
|
updateRowStateClicked(e) {
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
|
|
const { target } = e;
|
|
target.setAttribute('disabled', true);
|
|
target.classList.add('disabled');
|
|
|
|
axios[target.dataset.method](target.dataset.href)
|
|
.then(({ data }) => {
|
|
this.updateRowState(target);
|
|
this.updateBadges(data);
|
|
})
|
|
.catch(() => {
|
|
this.updateRowState(target, true);
|
|
return flash(__('Error updating status of to-do item.'));
|
|
});
|
|
}
|
|
|
|
updateRowState(target, isInactive = false) {
|
|
const row = target.closest('li');
|
|
const restoreBtn = row.querySelector('.js-undo-todo');
|
|
const doneBtn = row.querySelector('.js-done-todo');
|
|
|
|
target.classList.add('hidden');
|
|
target.removeAttribute('disabled');
|
|
target.classList.remove('disabled');
|
|
|
|
if (isInactive === true) {
|
|
restoreBtn.classList.add('hidden');
|
|
doneBtn.classList.remove('hidden');
|
|
} else if (target === doneBtn) {
|
|
row.classList.add('done-reversible');
|
|
restoreBtn.classList.remove('hidden');
|
|
} else if (target === restoreBtn) {
|
|
row.classList.remove('done-reversible');
|
|
doneBtn.classList.remove('hidden');
|
|
} else {
|
|
row.parentNode.removeChild(row);
|
|
}
|
|
}
|
|
|
|
updateAllStateClicked(e) {
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
|
|
const target = e.currentTarget;
|
|
target.setAttribute('disabled', true);
|
|
target.classList.add('disabled');
|
|
|
|
axios[target.dataset.method](target.dataset.href, {
|
|
ids: this.todo_ids,
|
|
})
|
|
.then(({ data }) => {
|
|
this.updateAllState(target, data);
|
|
this.updateBadges(data);
|
|
})
|
|
.catch(() => flash(__('Error updating status for all to-do items.')));
|
|
}
|
|
|
|
updateAllState(target, data) {
|
|
const markAllDoneBtn = document.querySelector('.js-todos-mark-all');
|
|
const undoAllBtn = document.querySelector('.js-todos-undo-all');
|
|
const todoListContainer = document.querySelector('.js-todos-list-container');
|
|
const nothingHereContainer = document.querySelector('.js-nothing-here-container');
|
|
|
|
target.removeAttribute('disabled');
|
|
target.classList.remove('disabled');
|
|
|
|
this.todo_ids = target === markAllDoneBtn ? data.updated_ids : [];
|
|
undoAllBtn.classList.toggle('hidden');
|
|
markAllDoneBtn.classList.toggle('hidden');
|
|
todoListContainer.classList.toggle('hidden');
|
|
nothingHereContainer.classList.toggle('hidden');
|
|
}
|
|
|
|
updateBadges(data) {
|
|
$(document).trigger('todo:toggle', data.count);
|
|
document.querySelector('.todos-pending .badge').innerHTML = data.count;
|
|
document.querySelector('.todos-done .badge').innerHTML = data.done_count;
|
|
}
|
|
|
|
goToTodoUrl(e) {
|
|
const todoLink = this.dataset.url;
|
|
|
|
if (!todoLink || e.target.tagName === 'A' || e.target.tagName === 'IMG') {
|
|
return;
|
|
}
|
|
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
|
|
if (isMetaClick(e)) {
|
|
const windowTarget = '_blank';
|
|
|
|
window.open(todoLink, windowTarget);
|
|
} else {
|
|
visitUrl(todoLink);
|
|
}
|
|
}
|
|
}
|