mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-26 12:27:06 +00:00 
			
		
		
		
	Remove jQuery .attr from the Fomantic dropdowns (#30114)
				
					
				
			- Switched from jQuery `attr` to plain javascript `getAttribute` and `setAttribute` - Tested the dropdowns and they work as before Signed-off-by: Yarden Shoham <git@yardenshoham.com> Co-authored-by: Giteabot <teabot@gitea.io>
This commit is contained in:
		| @@ -21,12 +21,11 @@ function ariaDropdownFn(...args) { | |||||||
|   // it means that this call will reset the dropdown internal settings, then we need to re-delegate the callbacks. |   // it means that this call will reset the dropdown internal settings, then we need to re-delegate the callbacks. | ||||||
|   const needDelegate = (!args.length || typeof args[0] !== 'string'); |   const needDelegate = (!args.length || typeof args[0] !== 'string'); | ||||||
|   for (const el of this) { |   for (const el of this) { | ||||||
|     const $dropdown = $(el); |  | ||||||
|     if (!el[ariaPatchKey]) { |     if (!el[ariaPatchKey]) { | ||||||
|       attachInit($dropdown); |       attachInit(el); | ||||||
|     } |     } | ||||||
|     if (needDelegate) { |     if (needDelegate) { | ||||||
|       delegateOne($dropdown); |       delegateOne($(el)); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   return ret; |   return ret; | ||||||
| @@ -40,17 +39,23 @@ function updateMenuItem(dropdown, item) { | |||||||
|   item.setAttribute('tabindex', '-1'); |   item.setAttribute('tabindex', '-1'); | ||||||
|   for (const el of item.querySelectorAll('a, input, button')) el.setAttribute('tabindex', '-1'); |   for (const el of item.querySelectorAll('a, input, button')) el.setAttribute('tabindex', '-1'); | ||||||
| } | } | ||||||
|  | /** | ||||||
| // make the label item and its "delete icon" has correct aria attributes |  * make the label item and its "delete icon" have correct aria attributes | ||||||
| function updateSelectionLabel($label) { |  * @param {HTMLElement} label | ||||||
|  |  */ | ||||||
|  | function updateSelectionLabel(label) { | ||||||
|   // the "label" is like this: "<a|div class="ui label" data-value="1">the-label-name <i|svg class="delete icon"/></a>" |   // the "label" is like this: "<a|div class="ui label" data-value="1">the-label-name <i|svg class="delete icon"/></a>" | ||||||
|   if (!$label.attr('id')) $label.attr('id', generateAriaId()); |   if (!label.id) { | ||||||
|   $label.attr('tabindex', '-1'); |     label.id = generateAriaId(); | ||||||
|   $label.find('.delete.icon').attr({ |   } | ||||||
|     'aria-hidden': 'false', |   label.tabIndex = -1; | ||||||
|     'aria-label': window.config.i18n.remove_label_str.replace('%s', $label.attr('data-value')), |  | ||||||
|     'role': 'button', |   const deleteIcon = label.querySelector('.delete.icon'); | ||||||
|   }); |   if (deleteIcon) { | ||||||
|  |     deleteIcon.setAttribute('aria-hidden', 'false'); | ||||||
|  |     deleteIcon.setAttribute('aria-label', window.config.i18n.remove_label_str.replace('%s', label.getAttribute('data-value'))); | ||||||
|  |     deleteIcon.setAttribute('role', 'button'); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| // delegate the dropdown's template functions and callback functions to add aria attributes. | // delegate the dropdown's template functions and callback functions to add aria attributes. | ||||||
| @@ -86,43 +91,44 @@ function delegateOne($dropdown) { | |||||||
|   const dropdownOnLabelCreateOld = dropdownCall('setting', 'onLabelCreate'); |   const dropdownOnLabelCreateOld = dropdownCall('setting', 'onLabelCreate'); | ||||||
|   dropdownCall('setting', 'onLabelCreate', function(value, text) { |   dropdownCall('setting', 'onLabelCreate', function(value, text) { | ||||||
|     const $label = dropdownOnLabelCreateOld.call(this, value, text); |     const $label = dropdownOnLabelCreateOld.call(this, value, text); | ||||||
|     updateSelectionLabel($label); |     updateSelectionLabel($label[0]); | ||||||
|     return $label; |     return $label; | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
|  |  | ||||||
| // for static dropdown elements (generated by server-side template), prepare them with necessary aria attributes | // for static dropdown elements (generated by server-side template), prepare them with necessary aria attributes | ||||||
| function attachStaticElements($dropdown, $focusable, $menu) { | function attachStaticElements(dropdown, focusable, menu) { | ||||||
|   const dropdown = $dropdown[0]; |  | ||||||
|  |  | ||||||
|   // prepare static dropdown menu list popup |   // prepare static dropdown menu list popup | ||||||
|   if (!$menu.attr('id')) $menu.attr('id', generateAriaId()); |   if (!menu.id) { | ||||||
|   $menu.find('> .item').each((_, item) => updateMenuItem(dropdown, item)); |     menu.id = generateAriaId(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   $(menu).find('> .item').each((_, item) => updateMenuItem(dropdown, item)); | ||||||
|  |  | ||||||
|   // this role could only be changed after its content is ready, otherwise some browsers+readers (like Chrome+AppleVoice) crash |   // this role could only be changed after its content is ready, otherwise some browsers+readers (like Chrome+AppleVoice) crash | ||||||
|   $menu.attr('role', dropdown[ariaPatchKey].listPopupRole); |   menu.setAttribute('role', dropdown[ariaPatchKey].listPopupRole); | ||||||
|  |  | ||||||
|   // prepare selection label items |   // prepare selection label items | ||||||
|   $dropdown.find('.ui.label').each((_, label) => updateSelectionLabel($(label))); |   for (const label of dropdown.querySelectorAll('.ui.label')) { | ||||||
|  |     updateSelectionLabel(label); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // make the primary element (focusable) aria-friendly |   // make the primary element (focusable) aria-friendly | ||||||
|   $focusable.attr({ |   focusable.setAttribute('role', focusable.getAttribute('role') ?? dropdown[ariaPatchKey].focusableRole); | ||||||
|     'role': $focusable.attr('role') ?? dropdown[ariaPatchKey].focusableRole, |   focusable.setAttribute('aria-haspopup', dropdown[ariaPatchKey].listPopupRole); | ||||||
|     'aria-haspopup': dropdown[ariaPatchKey].listPopupRole, |   focusable.setAttribute('aria-controls', menu.id); | ||||||
|     'aria-controls': $menu.attr('id'), |   focusable.setAttribute('aria-expanded', 'false'); | ||||||
|     'aria-expanded': 'false', |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   // use tooltip's content as aria-label if there is no aria-label |   // use tooltip's content as aria-label if there is no aria-label | ||||||
|   const tooltipContent = $dropdown.attr('data-tooltip-content'); |   const tooltipContent = dropdown.getAttribute('data-tooltip-content'); | ||||||
|   if (tooltipContent && !$dropdown.attr('aria-label')) { |   if (tooltipContent && !dropdown.getAttribute('aria-label')) { | ||||||
|     $dropdown.attr('aria-label', tooltipContent); |     dropdown.setAttribute('aria-label', tooltipContent); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| function attachInit($dropdown) { | function attachInit(dropdown) { | ||||||
|   const dropdown = $dropdown[0]; |  | ||||||
|   dropdown[ariaPatchKey] = {}; |   dropdown[ariaPatchKey] = {}; | ||||||
|   if ($dropdown.hasClass('custom')) return; |   if (dropdown.classList.contains('custom')) return; | ||||||
|  |  | ||||||
|   // Dropdown has 2 different focusing behaviors |   // Dropdown has 2 different focusing behaviors | ||||||
|   // * with search input: the input is focused, and it works with aria-activedescendant pointing another sibling element. |   // * with search input: the input is focused, and it works with aria-activedescendant pointing another sibling element. | ||||||
| @@ -139,64 +145,66 @@ function attachInit($dropdown) { | |||||||
|  |  | ||||||
|   // TODO: multiple selection is only partially supported. Check and test them one by one in the future. |   // TODO: multiple selection is only partially supported. Check and test them one by one in the future. | ||||||
|  |  | ||||||
|   const $textSearch = $dropdown.find('input.search').eq(0); |   const textSearch = dropdown.querySelector('input.search'); | ||||||
|   const $focusable = $textSearch.length ? $textSearch : $dropdown; // the primary element for focus, see comment above |   const focusable = textSearch || dropdown; // the primary element for focus, see comment above | ||||||
|   if (!$focusable.length) return; |   if (!focusable) return; | ||||||
|  |  | ||||||
|   // as a combobox, the input should not have autocomplete by default |   // as a combobox, the input should not have autocomplete by default | ||||||
|   if ($textSearch.length && !$textSearch.attr('autocomplete')) { |   if (textSearch && !textSearch.getAttribute('autocomplete')) { | ||||||
|     $textSearch.attr('autocomplete', 'off'); |     textSearch.setAttribute('autocomplete', 'off'); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   let $menu = $dropdown.find('> .menu'); |   let menu = $(dropdown).find('> .menu')[0]; | ||||||
|   if (!$menu.length) { |   if (!menu) { | ||||||
|     // some "multiple selection" dropdowns don't have a static menu element in HTML, we need to pre-create it to make it have correct aria attributes |     // some "multiple selection" dropdowns don't have a static menu element in HTML, we need to pre-create it to make it have correct aria attributes | ||||||
|     $menu = $('<div class="menu"></div>').appendTo($dropdown); |     menu = document.createElement('div'); | ||||||
|  |     menu.classList.add('menu'); | ||||||
|  |     dropdown.append(menu); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // There are 2 possible solutions about the role: combobox or menu. |   // There are 2 possible solutions about the role: combobox or menu. | ||||||
|   // The idea is that if there is an input, then it's a combobox, otherwise it's a menu. |   // The idea is that if there is an input, then it's a combobox, otherwise it's a menu. | ||||||
|   // Since #19861 we have prepared the "combobox" solution, but didn't get enough time to put it into practice and test before. |   // Since #19861 we have prepared the "combobox" solution, but didn't get enough time to put it into practice and test before. | ||||||
|   const isComboBox = $dropdown.find('input').length > 0; |   const isComboBox = dropdown.querySelectorAll('input').length > 0; | ||||||
|  |  | ||||||
|   dropdown[ariaPatchKey].focusableRole = isComboBox ? 'combobox' : 'menu'; |   dropdown[ariaPatchKey].focusableRole = isComboBox ? 'combobox' : 'menu'; | ||||||
|   dropdown[ariaPatchKey].listPopupRole = isComboBox ? 'listbox' : ''; |   dropdown[ariaPatchKey].listPopupRole = isComboBox ? 'listbox' : ''; | ||||||
|   dropdown[ariaPatchKey].listItemRole = isComboBox ? 'option' : 'menuitem'; |   dropdown[ariaPatchKey].listItemRole = isComboBox ? 'option' : 'menuitem'; | ||||||
|  |  | ||||||
|   attachDomEvents($dropdown, $focusable, $menu); |   attachDomEvents(dropdown, focusable, menu); | ||||||
|   attachStaticElements($dropdown, $focusable, $menu); |   attachStaticElements(dropdown, focusable, menu); | ||||||
| } | } | ||||||
|  |  | ||||||
| function attachDomEvents($dropdown, $focusable, $menu) { | function attachDomEvents(dropdown, focusable, menu) { | ||||||
|   const dropdown = $dropdown[0]; |  | ||||||
|   // when showing, it has class: ".animating.in" |   // when showing, it has class: ".animating.in" | ||||||
|   // when hiding, it has class: ".visible.animating.out" |   // when hiding, it has class: ".visible.animating.out" | ||||||
|   const isMenuVisible = () => ($menu.hasClass('visible') && !$menu.hasClass('out')) || $menu.hasClass('in'); |   const isMenuVisible = () => (menu.classList.contains('visible') && !menu.classList.contains('out')) || menu.classList.contains('in'); | ||||||
|  |  | ||||||
|   // update aria attributes according to current active/selected item |   // update aria attributes according to current active/selected item | ||||||
|   const refreshAriaActiveItem = () => { |   const refreshAriaActiveItem = () => { | ||||||
|     const menuVisible = isMenuVisible(); |     const menuVisible = isMenuVisible(); | ||||||
|     $focusable.attr('aria-expanded', menuVisible ? 'true' : 'false'); |     focusable.setAttribute('aria-expanded', menuVisible ? 'true' : 'false'); | ||||||
|  |  | ||||||
|     // if there is an active item, use it (the user is navigating between items) |     // if there is an active item, use it (the user is navigating between items) | ||||||
|     // otherwise use the "selected" for combobox (for the last selected item) |     // otherwise use the "selected" for combobox (for the last selected item) | ||||||
|     const $active = $menu.find('> .item.active, > .item.selected'); |     const active = $(menu).find('> .item.active, > .item.selected')[0]; | ||||||
|  |     if (!active) return; | ||||||
|     // if the popup is visible and has an active/selected item, use its id as aria-activedescendant |     // if the popup is visible and has an active/selected item, use its id as aria-activedescendant | ||||||
|     if (menuVisible) { |     if (menuVisible) { | ||||||
|       $focusable.attr('aria-activedescendant', $active.attr('id')); |       focusable.setAttribute('aria-activedescendant', active.id); | ||||||
|     } else if (dropdown[ariaPatchKey].listPopupRole === 'menu') { |     } else if (dropdown[ariaPatchKey].listPopupRole === 'menu') { | ||||||
|       // for menu, when the popup is hidden, no need to keep the aria-activedescendant, and clear the active/selected item |       // for menu, when the popup is hidden, no need to keep the aria-activedescendant, and clear the active/selected item | ||||||
|       $focusable.removeAttr('aria-activedescendant'); |       focusable.removeAttribute('aria-activedescendant'); | ||||||
|       $active.removeClass('active').removeClass('selected'); |       active.classList.remove('active', 'selected'); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   $dropdown.on('keydown', (e) => { |   dropdown.addEventListener('keydown', (e) => { | ||||||
|     // here it must use keydown event before dropdown's keyup handler, otherwise there is no Enter event in our keyup handler |     // here it must use keydown event before dropdown's keyup handler, otherwise there is no Enter event in our keyup handler | ||||||
|     if (e.key === 'Enter') { |     if (e.key === 'Enter') { | ||||||
|       const dropdownCall = fomanticDropdownFn.bind($dropdown); |       const dropdownCall = fomanticDropdownFn.bind($(dropdown)); | ||||||
|       let $item = dropdownCall('get item', dropdownCall('get value')); |       let $item = dropdownCall('get item', dropdownCall('get value')); | ||||||
|       if (!$item) $item = $menu.find('> .item.selected'); // when dropdown filters items by input, there is no "value", so query the "selected" item |       if (!$item) $item = $(menu).find('> .item.selected'); // when dropdown filters items by input, there is no "value", so query the "selected" item | ||||||
|       // if the selected item is clickable, then trigger the click event. |       // if the selected item is clickable, then trigger the click event. | ||||||
|       // we can not click any item without check, because Fomantic code might also handle the Enter event. that would result in double click. |       // we can not click any item without check, because Fomantic code might also handle the Enter event. that would result in double click. | ||||||
|       if ($item && ($item[0].matches('a') || $item.hasClass('js-aria-clickable'))) $item[0].click(); |       if ($item && ($item[0].matches('a') || $item.hasClass('js-aria-clickable'))) $item[0].click(); | ||||||
| @@ -209,7 +217,7 @@ function attachDomEvents($dropdown, $focusable, $menu) { | |||||||
|   // without the delay for hiding, the UI will be somewhat laggy and sometimes may get stuck in the animation. |   // without the delay for hiding, the UI will be somewhat laggy and sometimes may get stuck in the animation. | ||||||
|   const deferredRefreshAriaActiveItem = (delay = 0) => { setTimeout(refreshAriaActiveItem, delay) }; |   const deferredRefreshAriaActiveItem = (delay = 0) => { setTimeout(refreshAriaActiveItem, delay) }; | ||||||
|   dropdown[ariaPatchKey].deferredRefreshAriaActiveItem = deferredRefreshAriaActiveItem; |   dropdown[ariaPatchKey].deferredRefreshAriaActiveItem = deferredRefreshAriaActiveItem; | ||||||
|   $dropdown.on('keyup', (e) => { if (e.key.startsWith('Arrow')) deferredRefreshAriaActiveItem(); }); |   dropdown.addEventListener('keyup', (e) => { if (e.key.startsWith('Arrow')) deferredRefreshAriaActiveItem(); }); | ||||||
|  |  | ||||||
|   // if the dropdown has been opened by focus, do not trigger the next click event again. |   // if the dropdown has been opened by focus, do not trigger the next click event again. | ||||||
|   // otherwise the dropdown will be closed immediately, especially on Android with TalkBack |   // otherwise the dropdown will be closed immediately, especially on Android with TalkBack | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Yarden Shoham
					Yarden Shoham