@@ -9,26 +9,33 @@ const fomanticDropdownFn = $.fn.dropdown;
9
9
// use our own `$().dropdown` function to patch Fomantic's dropdown module
10
10
export function initAriaDropdownPatch ( ) {
11
11
if ( $ . fn . dropdown === ariaDropdownFn ) throw new Error ( 'initAriaDropdownPatch could only be called once' ) ;
12
+ $ . fn . dropdown . settings . onAfterFiltered = onAfterFiltered ;
12
13
$ . fn . dropdown = ariaDropdownFn ;
13
14
$ . fn . fomanticExt . onResponseKeepSelectedItem = onResponseKeepSelectedItem ;
14
15
( ariaDropdownFn as FomanticInitFunction ) . settings = fomanticDropdownFn . settings ;
15
16
}
16
17
17
18
// the patched `$.fn.dropdown` function, it passes the arguments to Fomantic's `$.fn.dropdown` function, and:
18
- // * it does the one-time attaching on the first call
19
- // * it delegates the `onLabelCreate` to the patched `onLabelCreate` to add necessary aria attributes
19
+ // * it does the one-time element event attaching on the first call
20
+ // * it delegates the module internal functions like `onLabelCreate` to the patched functions to add more features.
20
21
function ariaDropdownFn ( this : any , ...args : Parameters < FomanticInitFunction > ) {
21
22
const ret = fomanticDropdownFn . apply ( this , args ) ;
22
23
23
- // if the `$().dropdown()` call is without arguments, or it has non-string (object) argument,
24
- // it means that this call will reset the dropdown internal settings, then we need to re-delegate the callbacks.
25
- const needDelegate = ( ! args . length || typeof args [ 0 ] !== 'string' ) ;
26
24
for ( const el of this ) {
27
25
if ( ! el [ ariaPatchKey ] ) {
28
- attachInit ( el ) ;
26
+ // the elements don't belong to the dropdown "module" and won't be reset
27
+ // so we only need to initialize them once.
28
+ attachInitElements ( el ) ;
29
29
}
30
- if ( needDelegate ) {
31
- delegateOne ( $ ( el ) ) ;
30
+
31
+ // if the `$().dropdown()` call is without arguments, or it has non-string (object) argument,
32
+ // it means that such call will reset the dropdown "module" including internal settings,
33
+ // then we need to re-delegate the callbacks.
34
+ const $dropdown = $ ( el ) ;
35
+ const dropdownModule = $dropdown . data ( 'module-dropdown' ) ;
36
+ if ( ! dropdownModule . giteaDelegated ) {
37
+ dropdownModule . giteaDelegated = true ;
38
+ delegateDropdownModule ( $dropdown ) ;
32
39
}
33
40
}
34
41
return ret ;
@@ -61,37 +68,17 @@ function updateSelectionLabel(label: HTMLElement) {
61
68
}
62
69
}
63
70
64
- function processMenuItems ( $dropdown : any , dropdownCall : any ) {
65
- const hideEmptyDividers = dropdownCall ( 'setting' , 'hideDividers' ) === 'empty' ;
71
+ function onAfterFiltered ( this : any ) {
72
+ const $dropdown = $ ( this ) ;
73
+ const hideEmptyDividers = $dropdown . dropdown ( 'setting' , 'hideDividers' ) === 'empty' ;
66
74
const itemsMenu = $dropdown [ 0 ] . querySelector ( '.scrolling.menu' ) || $dropdown [ 0 ] . querySelector ( '.menu' ) ;
67
75
if ( hideEmptyDividers ) hideScopedEmptyDividers ( itemsMenu ) ;
68
76
}
69
77
70
78
// delegate the dropdown's template functions and callback functions to add aria attributes.
71
- function delegateOne ( $dropdown : any ) {
79
+ function delegateDropdownModule ( $dropdown : any ) {
72
80
const dropdownCall = fomanticDropdownFn . bind ( $dropdown ) ;
73
81
74
- // If there is a "search input" in the "menu", Fomantic will only "focus the input" but not "toggle the menu" when the "dropdown icon" is clicked.
75
- // Actually, Fomantic UI doesn't support such layout/usage. It needs to patch the "focusSearch" / "blurSearch" functions to make sure it toggles the menu.
76
- const oldFocusSearch = dropdownCall ( 'internal' , 'focusSearch' ) ;
77
- const oldBlurSearch = dropdownCall ( 'internal' , 'blurSearch' ) ;
78
- // * If the "dropdown icon" is clicked, Fomantic calls "focusSearch", so show the menu
79
- dropdownCall ( 'internal' , 'focusSearch' , function ( this : any ) { dropdownCall ( 'show' ) ; oldFocusSearch . call ( this ) } ) ;
80
- // * If the "dropdown icon" is clicked again when the menu is visible, Fomantic calls "blurSearch", so hide the menu
81
- dropdownCall ( 'internal' , 'blurSearch' , function ( this : any ) { oldBlurSearch . call ( this ) ; dropdownCall ( 'hide' ) } ) ;
82
-
83
- const oldFilterItems = dropdownCall ( 'internal' , 'filterItems' ) ;
84
- dropdownCall ( 'internal' , 'filterItems' , function ( this : any , ...args : any [ ] ) {
85
- oldFilterItems . call ( this , ...args ) ;
86
- processMenuItems ( $dropdown , dropdownCall ) ;
87
- } ) ;
88
-
89
- const oldShow = dropdownCall ( 'internal' , 'show' ) ;
90
- dropdownCall ( 'internal' , 'show' , function ( this : any , ...args : any [ ] ) {
91
- oldShow . call ( this , ...args ) ;
92
- processMenuItems ( $dropdown , dropdownCall ) ;
93
- } ) ;
94
-
95
82
// the "template" functions are used for dynamic creation (eg: AJAX)
96
83
const dropdownTemplates = { ...dropdownCall ( 'setting' , 'templates' ) , t : performance . now ( ) } ;
97
84
const dropdownTemplatesMenuOld = dropdownTemplates . menu ;
@@ -163,9 +150,8 @@ function attachStaticElements(dropdown: HTMLElement, focusable: HTMLElement, men
163
150
}
164
151
}
165
152
166
- function attachInit ( dropdown : HTMLElement ) {
153
+ function attachInitElements ( dropdown : HTMLElement ) {
167
154
( dropdown as any ) [ ariaPatchKey ] = { } ;
168
- if ( dropdown . classList . contains ( 'custom' ) ) return ;
169
155
170
156
// Dropdown has 2 different focusing behaviors
171
157
// * with search input: the input is focused, and it works with aria-activedescendant pointing another sibling element.
@@ -305,9 +291,11 @@ export function hideScopedEmptyDividers(container: Element) {
305
291
const visibleItems : Element [ ] = [ ] ;
306
292
const curScopeVisibleItems : Element [ ] = [ ] ;
307
293
let curScope : string = '' , lastVisibleScope : string = '' ;
308
- const isScopedDivider = ( item : Element ) => item . matches ( '.divider' ) && item . hasAttribute ( 'data-scope' ) ;
294
+ const isDivider = ( item : Element ) => item . classList . contains ( 'divider' ) ;
295
+ const isScopedDivider = ( item : Element ) => isDivider ( item ) && item . hasAttribute ( 'data-scope' ) ;
309
296
const hideDivider = ( item : Element ) => item . classList . add ( 'hidden' , 'transition' ) ; // dropdown has its own classes to hide items
310
-
297
+ const showDivider = ( item : Element ) => item . classList . remove ( 'hidden' , 'transition' ) ;
298
+ const isHidden = ( item : Element ) => item . classList . contains ( 'hidden' ) || item . classList . contains ( 'filtered' ) || item . classList . contains ( 'tw-hidden' ) ;
311
299
const handleScopeSwitch = ( itemScope : string ) => {
312
300
if ( curScopeVisibleItems . length === 1 && isScopedDivider ( curScopeVisibleItems [ 0 ] ) ) {
313
301
hideDivider ( curScopeVisibleItems [ 0 ] ) ;
@@ -323,34 +311,37 @@ export function hideScopedEmptyDividers(container: Element) {
323
311
curScopeVisibleItems . length = 0 ;
324
312
} ;
325
313
314
+ // reset hidden dividers
315
+ queryElems ( container , '.divider' , showDivider ) ;
316
+
326
317
// hide the scope dividers if the scope items are empty
327
318
for ( const item of container . children ) {
328
319
const itemScope = item . getAttribute ( 'data-scope' ) || '' ;
329
320
if ( itemScope !== curScope ) {
330
321
handleScopeSwitch ( itemScope ) ;
331
322
}
332
- if ( ! item . classList . contains ( 'filtered' ) && ! item . classList . contains ( 'tw-hidden' ) ) {
323
+ if ( ! isHidden ( item ) ) {
333
324
curScopeVisibleItems . push ( item as HTMLElement ) ;
334
325
}
335
326
}
336
327
handleScopeSwitch ( '' ) ;
337
328
338
329
// hide all leading and trailing dividers
339
330
while ( visibleItems . length ) {
340
- if ( ! visibleItems [ 0 ] . matches ( '.divider' ) ) break ;
331
+ if ( ! isDivider ( visibleItems [ 0 ] ) ) break ;
341
332
hideDivider ( visibleItems [ 0 ] ) ;
342
333
visibleItems . shift ( ) ;
343
334
}
344
335
while ( visibleItems . length ) {
345
- if ( ! visibleItems [ visibleItems . length - 1 ] . matches ( '.divider' ) ) break ;
336
+ if ( ! isDivider ( visibleItems [ visibleItems . length - 1 ] ) ) break ;
346
337
hideDivider ( visibleItems [ visibleItems . length - 1 ] ) ;
347
338
visibleItems . pop ( ) ;
348
339
}
349
340
// hide all duplicate dividers, hide current divider if next sibling is still divider
350
341
// no need to update "visibleItems" array since this is the last loop
351
- for ( const item of visibleItems ) {
352
- if ( ! item . matches ( '.divider' ) ) continue ;
353
- if ( item . nextElementSibling ?. matches ( '.divider' ) ) hideDivider ( item ) ;
342
+ for ( let i = 0 ; i < visibleItems . length - 1 ; i ++ ) {
343
+ if ( ! visibleItems [ i ] . matches ( '.divider' ) ) continue ;
344
+ if ( visibleItems [ i + 1 ] . matches ( '.divider' ) ) hideDivider ( visibleItems [ i ] ) ;
354
345
}
355
346
}
356
347
0 commit comments