2024-11-09 12:48:31 +08:00
|
|
|
import {fomanticQuery} from '../modules/fomantic/base.ts';
|
|
|
|
import {POST} from '../modules/fetch.ts';
|
2024-11-10 16:26:42 +08:00
|
|
|
import {queryElemChildren, queryElems, toggleElem} from '../utils/dom.ts';
|
2024-11-09 12:48:31 +08:00
|
|
|
|
|
|
|
// if there are draft comments, confirm before reloading, to avoid losing comments
|
|
|
|
export function issueSidebarReloadConfirmDraftComment() {
|
|
|
|
const commentTextareas = [
|
|
|
|
document.querySelector<HTMLTextAreaElement>('.edit-content-zone:not(.tw-hidden) textarea'),
|
|
|
|
document.querySelector<HTMLTextAreaElement>('#comment-form textarea'),
|
|
|
|
];
|
|
|
|
for (const textarea of commentTextareas) {
|
|
|
|
// Most users won't feel too sad if they lose a comment with 10 chars, they can re-type these in seconds.
|
|
|
|
// But if they have typed more (like 50) chars and the comment is lost, they will be very unhappy.
|
|
|
|
if (textarea && textarea.value.trim().length > 10) {
|
|
|
|
textarea.parentElement.scrollIntoView();
|
|
|
|
if (!window.confirm('Page will be reloaded, but there are draft comments. Continuing to reload will discard the comments. Continue?')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
window.location.reload();
|
|
|
|
}
|
|
|
|
|
|
|
|
function collectCheckedValues(elDropdown: HTMLElement) {
|
|
|
|
return Array.from(elDropdown.querySelectorAll('.menu > .item.checked'), (el) => el.getAttribute('data-value'));
|
|
|
|
}
|
|
|
|
|
|
|
|
export function initIssueSidebarComboList(container: HTMLElement) {
|
|
|
|
const updateUrl = container.getAttribute('data-update-url');
|
|
|
|
const elDropdown = container.querySelector<HTMLElement>(':scope > .ui.dropdown');
|
|
|
|
const elList = container.querySelector<HTMLElement>(':scope > .ui.list');
|
|
|
|
const elComboValue = container.querySelector<HTMLInputElement>(':scope > .combo-value');
|
2024-11-10 16:26:42 +08:00
|
|
|
let initialValues = collectCheckedValues(elDropdown);
|
2024-11-09 12:48:31 +08:00
|
|
|
|
|
|
|
elDropdown.addEventListener('click', (e) => {
|
|
|
|
const elItem = (e.target as HTMLElement).closest('.item');
|
|
|
|
if (!elItem) return;
|
|
|
|
e.preventDefault();
|
2024-11-10 16:26:42 +08:00
|
|
|
if (elItem.hasAttribute('data-can-change') && elItem.getAttribute('data-can-change') !== 'true') return;
|
|
|
|
|
|
|
|
if (elItem.matches('.clear-selection')) {
|
|
|
|
queryElems(elDropdown, '.menu > .item', (el) => el.classList.remove('checked'));
|
|
|
|
elComboValue.value = '';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const scope = elItem.getAttribute('data-scope');
|
|
|
|
if (scope) {
|
|
|
|
// scoped items could only be checked one at a time
|
|
|
|
const elSelected = elDropdown.querySelector<HTMLElement>(`.menu > .item.checked[data-scope="${CSS.escape(scope)}"]`);
|
|
|
|
if (elSelected === elItem) {
|
|
|
|
elItem.classList.toggle('checked');
|
|
|
|
} else {
|
|
|
|
queryElems(elDropdown, `.menu > .item[data-scope="${CSS.escape(scope)}"]`, (el) => el.classList.remove('checked'));
|
|
|
|
elItem.classList.toggle('checked', true);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
elItem.classList.toggle('checked');
|
|
|
|
}
|
2024-11-09 12:48:31 +08:00
|
|
|
elComboValue.value = collectCheckedValues(elDropdown).join(',');
|
|
|
|
});
|
|
|
|
|
|
|
|
const updateToBackend = async (changedValues) => {
|
|
|
|
let changed = false;
|
|
|
|
for (const value of initialValues) {
|
|
|
|
if (!changedValues.includes(value)) {
|
|
|
|
await POST(updateUrl, {data: new URLSearchParams({action: 'detach', id: value})});
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (const value of changedValues) {
|
|
|
|
if (!initialValues.includes(value)) {
|
|
|
|
await POST(updateUrl, {data: new URLSearchParams({action: 'attach', id: value})});
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (changed) issueSidebarReloadConfirmDraftComment();
|
|
|
|
};
|
|
|
|
|
2024-11-10 16:26:42 +08:00
|
|
|
const syncUiList = (changedValues) => {
|
2024-11-09 12:48:31 +08:00
|
|
|
const elEmptyTip = elList.querySelector('.item.empty-list');
|
|
|
|
queryElemChildren(elList, '.item:not(.empty-list)', (el) => el.remove());
|
|
|
|
for (const value of changedValues) {
|
2024-11-10 16:26:42 +08:00
|
|
|
const el = elDropdown.querySelector<HTMLElement>(`.menu > .item[data-value="${CSS.escape(value)}"]`);
|
2024-11-09 12:48:31 +08:00
|
|
|
const listItem = el.cloneNode(true) as HTMLElement;
|
2024-11-10 16:26:42 +08:00
|
|
|
queryElems(listItem, '.item-check-mark, .item-secondary-info', (el) => el.remove());
|
2024-11-09 12:48:31 +08:00
|
|
|
elList.append(listItem);
|
|
|
|
}
|
|
|
|
const hasItems = Boolean(elList.querySelector('.item:not(.empty-list)'));
|
|
|
|
toggleElem(elEmptyTip, !hasItems);
|
|
|
|
};
|
|
|
|
|
2024-11-10 16:26:42 +08:00
|
|
|
fomanticQuery(elDropdown).dropdown('setting', {
|
2024-11-09 12:48:31 +08:00
|
|
|
action: 'nothing', // do not hide the menu if user presses Enter
|
|
|
|
fullTextSearch: 'exact',
|
|
|
|
async onHide() {
|
2024-11-10 16:26:42 +08:00
|
|
|
// TODO: support "Esc" to cancel the selection. Use partial page loading to avoid losing inputs.
|
2024-11-09 12:48:31 +08:00
|
|
|
const changedValues = collectCheckedValues(elDropdown);
|
2024-11-10 16:26:42 +08:00
|
|
|
syncUiList(changedValues);
|
|
|
|
if (updateUrl) await updateToBackend(changedValues);
|
|
|
|
initialValues = changedValues;
|
2024-11-09 12:48:31 +08:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|