mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 00:48:29 +00:00 
			
		
		
		
	Fix #27466 The problem is that any item in the menu could be hidden, pure CSS won't work, and dropdown's builtin "hideDividers" doesn't work with our "scope dividers". The newly introduced "archived" label makes the dividers regression more.
		
			
				
	
	
		
			226 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			226 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import {throttle} from 'throttle-debounce';
 | |
| import {createTippy} from '../modules/tippy.ts';
 | |
| import {isDocumentFragmentOrElementNode} from '../utils/dom.ts';
 | |
| import octiconKebabHorizontal from '../../../public/assets/img/svg/octicon-kebab-horizontal.svg';
 | |
| 
 | |
| window.customElements.define('overflow-menu', class extends HTMLElement {
 | |
|   tippyContent: HTMLDivElement;
 | |
|   tippyItems: Array<HTMLElement>;
 | |
|   button: HTMLButtonElement;
 | |
|   menuItemsEl: HTMLElement;
 | |
|   resizeObserver: ResizeObserver;
 | |
|   mutationObserver: MutationObserver;
 | |
|   lastWidth: number;
 | |
| 
 | |
|   updateItems = throttle(100, () => {
 | |
|     if (!this.tippyContent) {
 | |
|       const div = document.createElement('div');
 | |
|       div.classList.add('tippy-target');
 | |
|       div.tabIndex = -1; // for initial focus, programmatic focus only
 | |
|       div.addEventListener('keydown', (e) => {
 | |
|         if (e.key === 'Tab') {
 | |
|           const items = this.tippyContent.querySelectorAll<HTMLElement>('[role="menuitem"]');
 | |
|           if (e.shiftKey) {
 | |
|             if (document.activeElement === items[0]) {
 | |
|               e.preventDefault();
 | |
|               items[items.length - 1].focus();
 | |
|             }
 | |
|           } else {
 | |
|             if (document.activeElement === items[items.length - 1]) {
 | |
|               e.preventDefault();
 | |
|               items[0].focus();
 | |
|             }
 | |
|           }
 | |
|         } else if (e.key === 'Escape') {
 | |
|           e.preventDefault();
 | |
|           e.stopPropagation();
 | |
|           this.button._tippy.hide();
 | |
|           this.button.focus();
 | |
|         } else if (e.key === ' ' || e.code === 'Enter') {
 | |
|           if (document.activeElement?.matches('[role="menuitem"]')) {
 | |
|             e.preventDefault();
 | |
|             e.stopPropagation();
 | |
|             (document.activeElement as HTMLElement).click();
 | |
|           }
 | |
|         } else if (e.key === 'ArrowDown') {
 | |
|           if (document.activeElement?.matches('.tippy-target')) {
 | |
|             e.preventDefault();
 | |
|             e.stopPropagation();
 | |
|             document.activeElement.querySelector<HTMLElement>('[role="menuitem"]:first-of-type').focus();
 | |
|           } else if (document.activeElement?.matches('[role="menuitem"]')) {
 | |
|             e.preventDefault();
 | |
|             e.stopPropagation();
 | |
|             (document.activeElement.nextElementSibling as HTMLElement)?.focus();
 | |
|           }
 | |
|         } else if (e.key === 'ArrowUp') {
 | |
|           if (document.activeElement?.matches('.tippy-target')) {
 | |
|             e.preventDefault();
 | |
|             e.stopPropagation();
 | |
|             document.activeElement.querySelector<HTMLElement>('[role="menuitem"]:last-of-type').focus();
 | |
|           } else if (document.activeElement?.matches('[role="menuitem"]')) {
 | |
|             e.preventDefault();
 | |
|             e.stopPropagation();
 | |
|             (document.activeElement.previousElementSibling as HTMLElement)?.focus();
 | |
|           }
 | |
|         }
 | |
|       });
 | |
|       this.append(div);
 | |
|       this.tippyContent = div;
 | |
|     }
 | |
| 
 | |
|     const itemFlexSpace = this.menuItemsEl.querySelector<HTMLSpanElement>('.item-flex-space');
 | |
|     const itemOverFlowMenuButton = this.querySelector<HTMLButtonElement>('.overflow-menu-button');
 | |
| 
 | |
|     // move items in tippy back into the menu items for subsequent measurement
 | |
|     for (const item of this.tippyItems || []) {
 | |
|       if (!itemFlexSpace || item.getAttribute('data-after-flex-space')) {
 | |
|         this.menuItemsEl.append(item);
 | |
|       } else {
 | |
|         itemFlexSpace.insertAdjacentElement('beforebegin', item);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // measure which items are partially outside the element and move them into the button menu
 | |
|     // flex space and overflow menu are excluded from measurement
 | |
|     itemFlexSpace?.style.setProperty('display', 'none', 'important');
 | |
|     itemOverFlowMenuButton?.style.setProperty('display', 'none', 'important');
 | |
|     this.tippyItems = [];
 | |
|     const menuRight = this.offsetLeft + this.offsetWidth;
 | |
|     const menuItems = this.menuItemsEl.querySelectorAll<HTMLElement>('.item, .item-flex-space');
 | |
|     let afterFlexSpace = false;
 | |
|     for (const item of menuItems) {
 | |
|       if (item.classList.contains('item-flex-space')) {
 | |
|         afterFlexSpace = true;
 | |
|         continue;
 | |
|       }
 | |
|       if (afterFlexSpace) item.setAttribute('data-after-flex-space', 'true');
 | |
|       const itemRight = item.offsetLeft + item.offsetWidth;
 | |
|       if (menuRight - itemRight < 38) { // roughly the width of .overflow-menu-button with some extra space
 | |
|         this.tippyItems.push(item);
 | |
|       }
 | |
|     }
 | |
|     itemFlexSpace?.style.removeProperty('display');
 | |
|     itemOverFlowMenuButton?.style.removeProperty('display');
 | |
| 
 | |
|     // if there are no overflown items, remove any previously created button
 | |
|     if (!this.tippyItems?.length) {
 | |
|       const btn = this.querySelector('.overflow-menu-button');
 | |
|       btn?._tippy?.destroy();
 | |
|       btn?.remove();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // remove aria role from items that moved from tippy to menu
 | |
|     for (const item of menuItems) {
 | |
|       if (!this.tippyItems.includes(item)) {
 | |
|         item.removeAttribute('role');
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // move all items that overflow into tippy
 | |
|     for (const item of this.tippyItems) {
 | |
|       item.setAttribute('role', 'menuitem');
 | |
|       this.tippyContent.append(item);
 | |
|     }
 | |
| 
 | |
|     // update existing tippy
 | |
|     if (this.button?._tippy) {
 | |
|       this.button._tippy.setContent(this.tippyContent);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // create button initially
 | |
|     const btn = document.createElement('button');
 | |
|     btn.classList.add('overflow-menu-button');
 | |
|     btn.setAttribute('aria-label', window.config.i18n.more_items);
 | |
|     btn.innerHTML = octiconKebabHorizontal;
 | |
|     this.append(btn);
 | |
|     this.button = btn;
 | |
| 
 | |
|     createTippy(btn, {
 | |
|       trigger: 'click',
 | |
|       hideOnClick: true,
 | |
|       interactive: true,
 | |
|       placement: 'bottom-end',
 | |
|       role: 'menu',
 | |
|       theme: 'menu',
 | |
|       content: this.tippyContent,
 | |
|       onShow: () => { // FIXME: onShown doesn't work (never be called)
 | |
|         setTimeout(() => {
 | |
|           this.tippyContent.focus();
 | |
|         }, 0);
 | |
|       },
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   init() {
 | |
|     // for horizontal menus where fomantic boldens active items, prevent this bold text from
 | |
|     // enlarging the menu's active item replacing the text node with a div that renders a
 | |
|     // invisible pseudo-element that enlarges the box.
 | |
|     if (this.matches('.ui.secondary.pointing.menu, .ui.tabular.menu')) {
 | |
|       for (const item of this.querySelectorAll('.item')) {
 | |
|         for (const child of item.childNodes) {
 | |
|           if (child.nodeType === Node.TEXT_NODE) {
 | |
|             const text = child.textContent.trim(); // whitespace is insignificant inside flexbox
 | |
|             if (!text) continue;
 | |
|             const span = document.createElement('span');
 | |
|             span.classList.add('resize-for-semibold');
 | |
|             span.setAttribute('data-text', text);
 | |
|             span.textContent = text;
 | |
|             child.replaceWith(span);
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // ResizeObserver triggers on initial render, so we don't manually call `updateItems` here which
 | |
|     // also avoids a full-page FOUC in Firefox that happens when `updateItems` is called too soon.
 | |
|     this.resizeObserver = new ResizeObserver((entries) => {
 | |
|       for (const entry of entries) {
 | |
|         const newWidth = entry.contentBoxSize[0].inlineSize;
 | |
|         if (newWidth !== this.lastWidth) {
 | |
|           requestAnimationFrame(() => {
 | |
|             this.updateItems();
 | |
|           });
 | |
|           this.lastWidth = newWidth;
 | |
|         }
 | |
|       }
 | |
|     });
 | |
|     this.resizeObserver.observe(this);
 | |
|   }
 | |
| 
 | |
|   connectedCallback() {
 | |
|     this.setAttribute('role', 'navigation');
 | |
| 
 | |
|     // check whether the mandatory `.overflow-menu-items` element is present initially which happens
 | |
|     // with Vue which renders differently than browsers. If it's not there, like in the case of browser
 | |
|     // template rendering, wait for its addition.
 | |
|     // The eslint rule is not sophisticated enough or aware of this problem, see
 | |
|     // https://github.com/43081j/eslint-plugin-wc/pull/130
 | |
|     const menuItemsEl = this.querySelector<HTMLElement>('.overflow-menu-items'); // eslint-disable-line wc/no-child-traversal-in-connectedcallback
 | |
|     if (menuItemsEl) {
 | |
|       this.menuItemsEl = menuItemsEl;
 | |
|       this.init();
 | |
|     } else {
 | |
|       this.mutationObserver = new MutationObserver((mutations) => {
 | |
|         for (const mutation of mutations) {
 | |
|           for (const node of mutation.addedNodes as NodeListOf<HTMLElement>) {
 | |
|             if (!isDocumentFragmentOrElementNode(node)) continue;
 | |
|             if (node.classList.contains('overflow-menu-items')) {
 | |
|               this.menuItemsEl = node;
 | |
|               this.mutationObserver?.disconnect();
 | |
|               this.init();
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       });
 | |
|       this.mutationObserver.observe(this, {childList: true});
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   disconnectedCallback() {
 | |
|     this.mutationObserver?.disconnect();
 | |
|     this.resizeObserver?.disconnect();
 | |
|   }
 | |
| });
 |