diff options
| author | Jeremy Jackson <[email protected]> | 2019-10-17 15:01:44 +0000 |
|---|---|---|
| committer | XhmikosR <[email protected]> | 2019-10-17 18:01:44 +0300 |
| commit | c1ee395f80c05de8317588b07f34a65c5b95c42c (patch) | |
| tree | 56d88e6e3e4256ceeea5766af744679e763eb904 /js | |
| parent | 104385c508a4c77761b04a9842e978bab6f359f6 (diff) | |
| download | bootstrap-c1ee395f80c05de8317588b07f34a65c5b95c42c.tar.xz bootstrap-c1ee395f80c05de8317588b07f34a65c5b95c42c.zip | |
Skip hidden dropdowns while focusing (#29523)
Diffstat (limited to 'js')
| -rw-r--r-- | js/src/dropdown.js | 2 | ||||
| -rw-r--r-- | js/src/util/index.js | 9 | ||||
| -rw-r--r-- | js/tests/unit/dropdown.spec.js | 37 |
3 files changed, 45 insertions, 3 deletions
diff --git a/js/src/dropdown.js b/js/src/dropdown.js index 06a271ef8..4504d61e0 100644 --- a/js/src/dropdown.js +++ b/js/src/dropdown.js @@ -9,6 +9,7 @@ import { getjQuery, getElementFromSelector, isElement, + isVisible, makeArray, noop, typeCheckConfig @@ -478,6 +479,7 @@ class Dropdown { } const items = makeArray(SelectorEngine.find(Selector.VISIBLE_ITEMS, parent)) + .filter(isVisible) if (!items.length) { return diff --git a/js/src/util/index.js b/js/src/util/index.js index 150b4f877..8a5ae2156 100644 --- a/js/src/util/index.js +++ b/js/src/util/index.js @@ -138,9 +138,12 @@ const isVisible = element => { } if (element.style && element.parentNode && element.parentNode.style) { - return element.style.display !== 'none' && - element.parentNode.style.display !== 'none' && - element.style.visibility !== 'hidden' + const elementStyle = getComputedStyle(element) + const parentNodeStyle = getComputedStyle(element.parentNode) + + return elementStyle.display !== 'none' && + parentNodeStyle.display !== 'none' && + elementStyle.visibility !== 'hidden' } return false diff --git a/js/tests/unit/dropdown.spec.js b/js/tests/unit/dropdown.spec.js index 0046cf680..7491e38c0 100644 --- a/js/tests/unit/dropdown.spec.js +++ b/js/tests/unit/dropdown.spec.js @@ -1292,6 +1292,43 @@ describe('Dropdown', () => { triggerDropdown.click() }) + it('should skip hidden element when using keyboard navigation', done => { + fixtureEl.innerHTML = [ + '<style>', + ' .d-none {', + ' display: none;', + ' }', + '</style>', + '<div class="dropdown">', + ' <button href="#" class="btn dropdown-toggle" data-toggle="dropdown">Dropdown</button>', + ' <div class="dropdown-menu">', + ' <button class="dropdown-item d-none" type="button">Hidden button by class</button>', + ' <a class="dropdown-item" href="#sub1" style="display: none">Hidden link</a>', + ' <a class="dropdown-item" href="#sub1" style="visibility: hidden">Hidden link</a>', + ' <a id="item1" class="dropdown-item" href="#">Another link</a>', + ' </div>', + '</div>' + ].join('') + + const triggerDropdown = fixtureEl.querySelector('[data-toggle="dropdown"]') + const dropdown = fixtureEl.querySelector('.dropdown') + + dropdown.addEventListener('shown.bs.dropdown', () => { + const keyDown = createEvent('keydown') + keyDown.which = 40 + + triggerDropdown.dispatchEvent(keyDown) + + expect(document.activeElement.classList.contains('d-none')).toEqual(false, '.d-none not focused') + expect(document.activeElement.style.display === 'none').toEqual(false, '"display: none" not focused') + expect(document.activeElement.style.visibility === 'hidden').toEqual(false, '"visibility: hidden" not focused') + + done() + }) + + triggerDropdown.click() + }) + it('should focus next/previous element when using keyboard navigation', done => { fixtureEl.innerHTML = [ '<div class="dropdown">', |
