mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-25 06:44:17 -04:00 
			
		
		
		
	Fix #2616 This PR adds a new sort option for exclusive labels. For exclusive labels, a new property is exposed called "order", while in the UI options are populated automatically in the `Sort` column (see screenshot below) for each exclusive label scope. --------- Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
		
			
				
	
	
		
			92 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			92 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import {toggleElem} from '../../utils/dom.ts';
 | |
| import {fomanticQuery} from '../../modules/fomantic/base.ts';
 | |
| 
 | |
| function nameHasScope(name: string): boolean {
 | |
|   return /.*[^/]\/[^/].*/.test(name);
 | |
| }
 | |
| 
 | |
| export function initCompLabelEdit(pageSelector: string) {
 | |
|   const pageContent = document.querySelector<HTMLElement>(pageSelector);
 | |
|   if (!pageContent) return;
 | |
| 
 | |
|   // for guest view, the modal is not available, the "labels" are read-only
 | |
|   const elModal = pageContent.querySelector<HTMLElement>('#issue-label-edit-modal');
 | |
|   if (!elModal) return;
 | |
| 
 | |
|   const elLabelId = elModal.querySelector<HTMLInputElement>('input[name="id"]');
 | |
|   const elNameInput = elModal.querySelector<HTMLInputElement>('.label-name-input');
 | |
|   const elExclusiveField = elModal.querySelector('.label-exclusive-input-field');
 | |
|   const elExclusiveInput = elModal.querySelector<HTMLInputElement>('.label-exclusive-input');
 | |
|   const elExclusiveWarning = elModal.querySelector('.label-exclusive-warning');
 | |
|   const elExclusiveOrderField = elModal.querySelector<HTMLInputElement>('.label-exclusive-order-input-field');
 | |
|   const elExclusiveOrderInput = elModal.querySelector<HTMLInputElement>('.label-exclusive-order-input');
 | |
|   const elIsArchivedField = elModal.querySelector('.label-is-archived-input-field');
 | |
|   const elIsArchivedInput = elModal.querySelector<HTMLInputElement>('.label-is-archived-input');
 | |
|   const elDescInput = elModal.querySelector<HTMLInputElement>('.label-desc-input');
 | |
|   const elColorInput = elModal.querySelector<HTMLInputElement>('.js-color-picker-input input');
 | |
| 
 | |
|   const syncModalUi = () => {
 | |
|     const hasScope = nameHasScope(elNameInput.value);
 | |
|     elExclusiveField.classList.toggle('disabled', !hasScope);
 | |
|     const showExclusiveWarning = hasScope && elExclusiveInput.checked && elModal.hasAttribute('data-need-warn-exclusive');
 | |
|     toggleElem(elExclusiveWarning, showExclusiveWarning);
 | |
|     if (!hasScope) elExclusiveInput.checked = false;
 | |
|     toggleElem(elExclusiveOrderField, elExclusiveInput.checked);
 | |
| 
 | |
|     if (parseInt(elExclusiveOrderInput.value) <= 0) {
 | |
|       elExclusiveOrderInput.style.color = 'var(--color-placeholder-text) !important';
 | |
|     } else {
 | |
|       elExclusiveOrderInput.style.color = null;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   const showLabelEditModal = (btn:HTMLElement) => {
 | |
|     // the "btn" should contain the label's attributes by its `data-label-xxx` attributes
 | |
|     const form = elModal.querySelector<HTMLFormElement>('form');
 | |
|     elLabelId.value = btn.getAttribute('data-label-id') || '';
 | |
|     elNameInput.value = btn.getAttribute('data-label-name') || '';
 | |
|     elExclusiveOrderInput.value = btn.getAttribute('data-label-exclusive-order') || '0';
 | |
|     elIsArchivedInput.checked = btn.getAttribute('data-label-is-archived') === 'true';
 | |
|     elExclusiveInput.checked = btn.getAttribute('data-label-exclusive') === 'true';
 | |
|     elDescInput.value = btn.getAttribute('data-label-description') || '';
 | |
|     elColorInput.value = btn.getAttribute('data-label-color') || '';
 | |
|     elColorInput.dispatchEvent(new Event('input', {bubbles: true})); // trigger the color picker
 | |
| 
 | |
|     // if label id exists: "edit label" mode; otherwise: "new label" mode
 | |
|     const isEdit = Boolean(elLabelId.value);
 | |
| 
 | |
|     // if a label was not exclusive but has issues, then it should warn user if it will become exclusive
 | |
|     const numIssues = parseInt(btn.getAttribute('data-label-num-issues') || '0');
 | |
|     elModal.toggleAttribute('data-need-warn-exclusive', !elExclusiveInput.checked && numIssues > 0);
 | |
|     elModal.querySelector('.header').textContent = isEdit ? elModal.getAttribute('data-text-edit-label') : elModal.getAttribute('data-text-new-label');
 | |
| 
 | |
|     const curPageLink = elModal.getAttribute('data-current-page-link');
 | |
|     form.action = isEdit ? `${curPageLink}/edit` : `${curPageLink}/new`;
 | |
|     toggleElem(elIsArchivedField, isEdit);
 | |
|     syncModalUi();
 | |
|     fomanticQuery(elModal).modal({
 | |
|       onApprove() {
 | |
|         if (!form.checkValidity()) {
 | |
|           form.reportValidity();
 | |
|           return false;
 | |
|         }
 | |
|         form.submit();
 | |
|       },
 | |
|     }).modal('show');
 | |
|   };
 | |
| 
 | |
|   elModal.addEventListener('input', () => syncModalUi());
 | |
| 
 | |
|   // theoretically, if the modal exists, the "new label" button should also exist, just in case it doesn't, use "?."
 | |
|   const elNewLabel = pageContent.querySelector<HTMLElement>('.ui.button.new-label');
 | |
|   elNewLabel?.addEventListener('click', () => showLabelEditModal(elNewLabel));
 | |
| 
 | |
|   const elEditLabelButtons = pageContent.querySelectorAll<HTMLElement>('.edit-label-button');
 | |
|   for (const btn of elEditLabelButtons) {
 | |
|     btn.addEventListener('click', (e) => {
 | |
|       e.preventDefault();
 | |
|       showLabelEditModal(btn);
 | |
|     });
 | |
|   }
 | |
| }
 |