2024-11-03 12:00:12 +01:00
|
|
|
import $ from 'jquery';
|
|
|
|
import {POST} from '../modules/fetch.ts';
|
|
|
|
import {updateIssuesMeta} from './repo-common.ts';
|
|
|
|
import {svg} from '../svg.ts';
|
|
|
|
import {htmlEscape} from 'escape-goat';
|
2024-11-10 09:26:42 +01:00
|
|
|
import {queryElems, toggleElem} from '../utils/dom.ts';
|
2024-11-09 05:48:31 +01:00
|
|
|
import {initIssueSidebarComboList, issueSidebarReloadConfirmDraftComment} from './repo-issue-sidebar-combolist.ts';
|
2024-11-03 12:00:12 +01:00
|
|
|
|
|
|
|
function initBranchSelector() {
|
|
|
|
const elSelectBranch = document.querySelector('.ui.dropdown.select-branch');
|
|
|
|
if (!elSelectBranch) return;
|
|
|
|
|
|
|
|
const urlUpdateIssueRef = elSelectBranch.getAttribute('data-url-update-issueref');
|
|
|
|
const $selectBranch = $(elSelectBranch);
|
|
|
|
const $branchMenu = $selectBranch.find('.reference-list-menu');
|
|
|
|
$branchMenu.find('.item:not(.no-select)').on('click', async function (e) {
|
|
|
|
e.preventDefault();
|
|
|
|
const selectedValue = this.getAttribute('data-id'); // eg: "refs/heads/my-branch"
|
|
|
|
const selectedText = this.getAttribute('data-name'); // eg: "my-branch"
|
|
|
|
if (urlUpdateIssueRef) {
|
|
|
|
// for existing issue, send request to update issue ref, and reload page
|
|
|
|
try {
|
|
|
|
await POST(urlUpdateIssueRef, {data: new URLSearchParams({ref: selectedValue})});
|
|
|
|
window.location.reload();
|
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// for new issue, only update UI&form, do not send request/reload
|
|
|
|
const selectedHiddenSelector = this.getAttribute('data-id-selector');
|
2024-11-10 09:26:42 +01:00
|
|
|
document.querySelector<HTMLInputElement>(selectedHiddenSelector).value = selectedValue;
|
2024-11-03 12:00:12 +01:00
|
|
|
elSelectBranch.querySelector('.text-branch-name').textContent = selectedText;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// List submits
|
|
|
|
function initListSubmits(selector, outerSelector) {
|
|
|
|
const $list = $(`.ui.${outerSelector}.list`);
|
|
|
|
const $noSelect = $list.find('.no-select');
|
|
|
|
const $listMenu = $(`.${selector} .menu`);
|
|
|
|
let hasUpdateAction = $listMenu.data('action') === 'update';
|
|
|
|
const items = {};
|
|
|
|
|
|
|
|
$(`.${selector}`).dropdown({
|
|
|
|
'action': 'nothing', // do not hide the menu if user presses Enter
|
|
|
|
fullTextSearch: 'exact',
|
|
|
|
async onHide() {
|
|
|
|
hasUpdateAction = $listMenu.data('action') === 'update'; // Update the var
|
|
|
|
if (hasUpdateAction) {
|
|
|
|
// TODO: Add batch functionality and make this 1 network request.
|
|
|
|
const itemEntries = Object.entries(items);
|
|
|
|
for (const [elementId, item] of itemEntries) {
|
|
|
|
await updateIssuesMeta(
|
|
|
|
item['update-url'],
|
2024-11-10 09:26:42 +01:00
|
|
|
item['action'],
|
2024-11-03 12:00:12 +01:00
|
|
|
item['issue-id'],
|
|
|
|
elementId,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (itemEntries.length) {
|
2024-11-09 05:48:31 +01:00
|
|
|
issueSidebarReloadConfirmDraftComment();
|
2024-11-03 12:00:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
$listMenu.find('.item:not(.no-select)').on('click', function (e) {
|
|
|
|
e.preventDefault();
|
|
|
|
if (this.classList.contains('ban-change')) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
hasUpdateAction = $listMenu.data('action') === 'update'; // Update the var
|
|
|
|
|
|
|
|
const clickedItem = this; // eslint-disable-line unicorn/no-this-assignment
|
|
|
|
const scope = this.getAttribute('data-scope');
|
|
|
|
|
|
|
|
$(this).parent().find('.item').each(function () {
|
|
|
|
if (scope) {
|
|
|
|
// Enable only clicked item for scoped labels
|
|
|
|
if (this.getAttribute('data-scope') !== scope) {
|
2024-11-10 09:26:42 +01:00
|
|
|
return;
|
2024-11-03 12:00:12 +01:00
|
|
|
}
|
|
|
|
if (this !== clickedItem && !this.classList.contains('checked')) {
|
2024-11-10 09:26:42 +01:00
|
|
|
return;
|
2024-11-03 12:00:12 +01:00
|
|
|
}
|
|
|
|
} else if (this !== clickedItem) {
|
|
|
|
// Toggle for other labels
|
2024-11-10 09:26:42 +01:00
|
|
|
return;
|
2024-11-03 12:00:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.classList.contains('checked')) {
|
|
|
|
$(this).removeClass('checked');
|
|
|
|
$(this).find('.octicon-check').addClass('tw-invisible');
|
|
|
|
if (hasUpdateAction) {
|
|
|
|
if (!($(this).data('id') in items)) {
|
|
|
|
items[$(this).data('id')] = {
|
|
|
|
'update-url': $listMenu.data('update-url'),
|
|
|
|
action: 'detach',
|
|
|
|
'issue-id': $listMenu.data('issue-id'),
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
delete items[$(this).data('id')];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$(this).addClass('checked');
|
|
|
|
$(this).find('.octicon-check').removeClass('tw-invisible');
|
|
|
|
if (hasUpdateAction) {
|
|
|
|
if (!($(this).data('id') in items)) {
|
|
|
|
items[$(this).data('id')] = {
|
|
|
|
'update-url': $listMenu.data('update-url'),
|
|
|
|
action: 'attach',
|
|
|
|
'issue-id': $listMenu.data('issue-id'),
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
delete items[$(this).data('id')];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO: Which thing should be done for choosing review requests
|
|
|
|
// to make chosen items be shown on time here?
|
2024-11-09 05:48:31 +01:00
|
|
|
if (selector === 'select-assignees-modify') {
|
2024-11-03 12:00:12 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const listIds = [];
|
|
|
|
$(this).parent().find('.item').each(function () {
|
|
|
|
if (this.classList.contains('checked')) {
|
|
|
|
listIds.push($(this).data('id'));
|
|
|
|
$($(this).data('id-selector')).removeClass('tw-hidden');
|
|
|
|
} else {
|
|
|
|
$($(this).data('id-selector')).addClass('tw-hidden');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (!listIds.length) {
|
|
|
|
$noSelect.removeClass('tw-hidden');
|
|
|
|
} else {
|
|
|
|
$noSelect.addClass('tw-hidden');
|
|
|
|
}
|
|
|
|
$($(this).parent().data('id')).val(listIds.join(','));
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
$listMenu.find('.no-select.item').on('click', function (e) {
|
|
|
|
e.preventDefault();
|
|
|
|
if (hasUpdateAction) {
|
|
|
|
(async () => {
|
|
|
|
await updateIssuesMeta(
|
|
|
|
$listMenu.data('update-url'),
|
|
|
|
'clear',
|
|
|
|
$listMenu.data('issue-id'),
|
|
|
|
'',
|
|
|
|
);
|
2024-11-09 05:48:31 +01:00
|
|
|
issueSidebarReloadConfirmDraftComment();
|
2024-11-03 12:00:12 +01:00
|
|
|
})();
|
|
|
|
}
|
|
|
|
|
|
|
|
$(this).parent().find('.item').each(function () {
|
|
|
|
$(this).removeClass('checked');
|
|
|
|
$(this).find('.octicon-check').addClass('tw-invisible');
|
|
|
|
});
|
|
|
|
|
2024-11-09 05:48:31 +01:00
|
|
|
if (selector === 'select-assignees-modify') {
|
2024-11-03 12:00:12 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$list.find('.item').each(function () {
|
|
|
|
$(this).addClass('tw-hidden');
|
|
|
|
});
|
|
|
|
$noSelect.removeClass('tw-hidden');
|
|
|
|
$($(this).parent().data('id')).val('');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function selectItem(select_id, input_id) {
|
|
|
|
const $menu = $(`${select_id} .menu`);
|
|
|
|
const $list = $(`.ui${select_id}.list`);
|
|
|
|
const hasUpdateAction = $menu.data('action') === 'update';
|
|
|
|
|
|
|
|
$menu.find('.item:not(.no-select)').on('click', function () {
|
|
|
|
$(this).parent().find('.item').each(function () {
|
|
|
|
$(this).removeClass('selected active');
|
|
|
|
});
|
|
|
|
|
|
|
|
$(this).addClass('selected active');
|
|
|
|
if (hasUpdateAction) {
|
|
|
|
(async () => {
|
|
|
|
await updateIssuesMeta(
|
|
|
|
$menu.data('update-url'),
|
|
|
|
'',
|
|
|
|
$menu.data('issue-id'),
|
|
|
|
$(this).data('id'),
|
|
|
|
);
|
2024-11-09 05:48:31 +01:00
|
|
|
issueSidebarReloadConfirmDraftComment();
|
2024-11-03 12:00:12 +01:00
|
|
|
})();
|
|
|
|
}
|
|
|
|
|
|
|
|
let icon = '';
|
|
|
|
if (input_id === '#milestone_id') {
|
|
|
|
icon = svg('octicon-milestone', 18, 'tw-mr-2');
|
|
|
|
} else if (input_id === '#project_id') {
|
|
|
|
icon = svg('octicon-project', 18, 'tw-mr-2');
|
|
|
|
} else if (input_id === '#assignee_id') {
|
|
|
|
icon = `<img class="ui avatar image tw-mr-2" alt="avatar" src=${$(this).data('avatar')}>`;
|
|
|
|
}
|
|
|
|
|
|
|
|
$list.find('.selected').html(`
|
|
|
|
<a class="item muted sidebar-item-link" href="${htmlEscape(this.getAttribute('data-href'))}">
|
|
|
|
${icon}
|
|
|
|
${htmlEscape(this.textContent)}
|
|
|
|
</a>
|
|
|
|
`);
|
|
|
|
|
|
|
|
$(`.ui${select_id}.list .no-select`).addClass('tw-hidden');
|
|
|
|
$(input_id).val($(this).data('id'));
|
|
|
|
});
|
|
|
|
$menu.find('.no-select.item').on('click', function () {
|
|
|
|
$(this).parent().find('.item:not(.no-select)').each(function () {
|
|
|
|
$(this).removeClass('selected active');
|
|
|
|
});
|
|
|
|
|
|
|
|
if (hasUpdateAction) {
|
|
|
|
(async () => {
|
|
|
|
await updateIssuesMeta(
|
|
|
|
$menu.data('update-url'),
|
|
|
|
'',
|
|
|
|
$menu.data('issue-id'),
|
|
|
|
$(this).data('id'),
|
|
|
|
);
|
2024-11-09 05:48:31 +01:00
|
|
|
issueSidebarReloadConfirmDraftComment();
|
2024-11-03 12:00:12 +01:00
|
|
|
})();
|
|
|
|
}
|
|
|
|
|
|
|
|
$list.find('.selected').html('');
|
|
|
|
$list.find('.no-select').removeClass('tw-hidden');
|
|
|
|
$(input_id).val('');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-11-05 08:46:40 +01:00
|
|
|
function initRepoIssueDue() {
|
|
|
|
const form = document.querySelector<HTMLFormElement>('.issue-due-form');
|
|
|
|
if (!form) return;
|
|
|
|
const deadline = form.querySelector<HTMLInputElement>('input[name=deadline]');
|
|
|
|
document.querySelector('.issue-due-edit')?.addEventListener('click', () => {
|
|
|
|
toggleElem(form);
|
|
|
|
});
|
|
|
|
document.querySelector('.issue-due-remove')?.addEventListener('click', () => {
|
|
|
|
deadline.value = '';
|
|
|
|
form.dispatchEvent(new Event('submit', {cancelable: true, bubbles: true}));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-11-03 12:00:12 +01:00
|
|
|
export function initRepoIssueSidebar() {
|
|
|
|
initBranchSelector();
|
2024-11-05 08:46:40 +01:00
|
|
|
initRepoIssueDue();
|
2024-11-03 12:00:12 +01:00
|
|
|
|
2024-11-09 05:48:31 +01:00
|
|
|
// TODO: refactor the legacy initListSubmits&selectItem to initIssueSidebarComboList
|
2024-11-03 12:00:12 +01:00
|
|
|
initListSubmits('select-assignees', 'assignees');
|
|
|
|
initListSubmits('select-assignees-modify', 'assignees');
|
2024-11-10 09:26:42 +01:00
|
|
|
selectItem('.select-assignee', '#assignee_id');
|
|
|
|
|
2024-11-03 12:00:12 +01:00
|
|
|
selectItem('.select-project', '#project_id');
|
|
|
|
selectItem('.select-milestone', '#milestone_id');
|
2024-11-09 05:48:31 +01:00
|
|
|
|
2024-11-10 09:26:42 +01:00
|
|
|
// init the combo list: a dropdown for selecting items, and a list for showing selected items and related actions
|
|
|
|
queryElems<HTMLElement>(document, '.issue-sidebar-combo', (el) => initIssueSidebarComboList(el));
|
2024-11-03 12:00:12 +01:00
|
|
|
}
|