diff options
| author | Brandon Rainey <83629154+brandonrainey@users.noreply.github.com> | 2024-05-17 00:06:33 -0400 | 
|---|---|---|
| committer | GitHub <noreply@github.com> | 2024-05-17 04:06:33 +0000 | 
| commit | a3ed56c3deb1a286a7e84c86064fd33fc33e8f61 (patch) | |
| tree | d14b47e509adde3430c7f4b1fb57b951d8299662 | |
| parent | 6ade1eb2e0d8037472631a5e6718c33621303e31 (diff) | |
refactored onKeyDown to be easier to read, and added test (#943)
* refactored onKeyDown to be easier to read, and added test to ensure preserved behavior
* added new keypress events to test
* refactored test to call method directly from SearchDisplayController, removed second test, removed old copyright
* added test for invalid keys, split keyboard events into valid and invalid
* added crtl + backspace as valid keypress, added 2 new invaid keypresses to test
| -rw-r--r-- | ext/js/display/search-display-controller.js | 19 | ||||
| -rw-r--r-- | test/search-display-controller.test.js | 104 | 
2 files changed, 113 insertions, 10 deletions
| diff --git a/ext/js/display/search-display-controller.js b/ext/js/display/search-display-controller.js index 9b2311d1..e63b96e8 100644 --- a/ext/js/display/search-display-controller.js +++ b/ext/js/display/search-display-controller.js @@ -154,16 +154,15 @@ export class SearchDisplayController {       * @param {KeyboardEvent} e       */      _onKeyDown(e) { -        const {activeElement} = document; -        if ( -            activeElement !== this._queryInput && -            !this._isElementInput(activeElement) && -            (!e.ctrlKey || e.key === 'Backspace') && -            !e.metaKey && -            !e.altKey && -            (e.key.length === 1 || e.key === 'Backspace') && -            e.key !== ' ' -        ) { +        const activeElement = document.activeElement; + +        const isInputField = this._isElementInput(activeElement); +        const isAllowedKey = e.key.length === 1 || e.key === 'Backspace'; +        const isModifierKey = e.ctrlKey || e.metaKey || e.altKey; +        const isSpaceKey = e.key === ' '; +        const isCtrlBackspace = e.ctrlKey && e.key === 'Backspace'; + +        if (!isInputField && (!isModifierKey || isCtrlBackspace) && isAllowedKey && !isSpaceKey) {              this._queryInput.focus({preventScroll: true});          }      } diff --git a/test/search-display-controller.test.js b/test/search-display-controller.test.js new file mode 100644 index 00000000..277a51c3 --- /dev/null +++ b/test/search-display-controller.test.js @@ -0,0 +1,104 @@ +/* + * Copyright (C) 2023-2024  Yomitan Authors + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program.  If not, see <https://www.gnu.org/licenses/>. + */ + +import {describe, expect, afterAll, test, vi} from 'vitest'; +import {setupDomTest} from './fixtures/dom-test.js'; +import {querySelectorNotNull} from '../ext/js/dom/query-selector.js'; +import {SearchDisplayController} from '../ext/js/display/search-display-controller.js'; +import {Display} from '../ext/js/display/display.js'; +import {DisplayAudio} from '../ext/js/display/display-audio.js'; +import {SearchPersistentStateController} from '../ext/js/display/search-persistent-state-controller.js'; +import {Application} from '../ext/js/application.js'; +import {CrossFrameAPI} from '../ext/js/comm/cross-frame-api.js'; +import {API} from '../ext/js/comm/api.js'; +import {DocumentFocusController} from '../ext/js/dom/document-focus-controller.js'; +import {HotkeyHandler} from '../ext/js/input/hotkey-handler.js'; +import {WebExtension} from '../ext/js/extension/web-extension.js'; + +const documentSearchDisplayControllerEnv = await setupDomTest('ext/search.html'); + +const {window, teardown} = documentSearchDisplayControllerEnv; + +const {document} = window; + +const frameId = 1; +const tabId = 1; +const webExtension = new WebExtension(); +const hotkeyHandler = new HotkeyHandler(); +const documentFocusController = new DocumentFocusController(); +const displayPageType = 'search'; +const api = new API(webExtension); +const crossFrameAPI = new CrossFrameAPI(api, tabId, frameId); +const application = new Application(api, crossFrameAPI); +const display = new Display(application, displayPageType, documentFocusController, hotkeyHandler); +const displayAudio = new DisplayAudio(display); +const searchPersistentStateController = new SearchPersistentStateController(); + +const searchDisplayController = new SearchDisplayController(display, displayAudio, searchPersistentStateController); + +// eslint-disable-next-line no-underscore-dangle +const onKeyDownMethod = searchDisplayController._onKeyDown.bind(searchDisplayController); + +/** + * @type {HTMLInputElement} + */ +const queryInput = querySelectorNotNull(document, '#search-textbox'); + +const focusSpy = vi.spyOn(queryInput, 'focus'); + +describe('Keyboard Event Handling', () => { +    afterAll(() => teardown(global)); + +    const validKeypressEvents = [ +        new KeyboardEvent('keydown', {key: 'a', ctrlKey: false, metaKey: false, altKey: false}), +        new KeyboardEvent('keydown', {key: 'Backspace'}), +        new KeyboardEvent('keydown', {key: 'Backspace', ctrlKey: true, metaKey: false, altKey: false}) +    ]; + +    const invalidKeypressEvents = [ +        new KeyboardEvent('keydown', {key: '', ctrlKey: true, metaKey: false, altKey: false}), +        new KeyboardEvent('keydown', {key: '', ctrlKey: false, metaKey: true, altKey: false}), +        new KeyboardEvent('keydown', {key: '', ctrlKey: false, metaKey: false, altKey: true}), +        new KeyboardEvent('keydown', {key: ' ', ctrlKey: false, metaKey: false, altKey: false}), +        new KeyboardEvent('keydown', {key: 'a', ctrlKey: true, metaKey: false, altKey: false}), +        new KeyboardEvent('keydown', {key: 'a', ctrlKey: false, metaKey: true, altKey: false}), +        new KeyboardEvent('keydown', {key: 'a', ctrlKey: false, metaKey: false, altKey: true}), +        new KeyboardEvent('keydown', {key: 'Backspace', ctrlKey: false, metaKey: true, altKey: false}), +        new KeyboardEvent('keydown', {key: 'Backspace', ctrlKey: false, metaKey: false, altKey: true}), +        new KeyboardEvent('keydown', {key: 'ArrowDown'}) +    ]; + +    test('should test that onKeyDown function focuses input for valid keys', () => { +        for (const event of validKeypressEvents) { +            queryInput.blur(); +            onKeyDownMethod(event); +        } + +        expect(focusSpy.mock.calls.length).toBe(validKeypressEvents.length); +        focusSpy.mockReset(); +    }); + + +    test('should test that onKeyDown function does not focus input for invalid keys', () => { +        for (const event of invalidKeypressEvents) { +            queryInput.blur(); +            onKeyDownMethod(event); +        } + +        expect(focusSpy.mock.calls.length).toBe(0); +    }); +}); |