Skip to content

Commit

Permalink
fix(keyboard): dispatch change event on blur (#703)
Browse files Browse the repository at this point in the history
  • Loading branch information
ph-fritsche authored Jul 25, 2021
1 parent e5e78af commit 9600abb
Show file tree
Hide file tree
Showing 2 changed files with 63 additions and 0 deletions.
23 changes: 23 additions & 0 deletions src/__tests__/keyboard/shared/fireInputEvent.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import {setup} from '__tests__/helpers/utils'
import userEvent from '../../../'

it('dispatch change event on blur', () => {
const {element, getEvents} = setup('<input/>')

;(element as HTMLInputElement).focus()
userEvent.keyboard('foo')
;(element as HTMLInputElement).blur()

expect(getEvents('change')).toHaveLength(1)
})

it('do not dispatch change event if value did not change', () => {
const {element, getEvents} = setup('<input/>')

;(element as HTMLInputElement).focus()
userEvent.keyboard('foo')
userEvent.keyboard('{backspace}{backspace}{backspace}')
;(element as HTMLInputElement).blur()

expect(getEvents('change')).toHaveLength(0)
})
40 changes: 40 additions & 0 deletions src/keyboard/shared/fireInputEvent.ts
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,15 @@ function setSelectionRangeAfterInputHandler(
}
}

const initial = Symbol('initial input value/textContent')
const onBlur = Symbol('onBlur')
declare global {
interface Element {
[initial]?: string
[onBlur]?: EventListener
}
}

/**
* React tracks the changes on element properties.
* This workaround tries to alter the DOM element without React noticing,
Expand All @@ -92,8 +101,39 @@ function applyNative<T extends Element, P extends keyof T>(
Object.defineProperty(element, propName, nativeDescriptor)
}

// Keep track of the initial value to determine if a change event should be dispatched.
// CONSTRAINT: We can not determine what happened between focus event and our first API call.
if (element[initial] === undefined) {
element[initial] = String(element[propName])
}

element[propName] = propValue

// Add an event listener for the blur event to the capture phase on the window.
// CONSTRAINT: Currently there is no cross-platform solution to unshift the event handler stack.
// Our change event might occur after other event handlers on the blur event have been processed.
if (!element[onBlur]) {
element.ownerDocument.defaultView?.addEventListener(
'blur',
(element[onBlur] = () => {
const initV = element[initial]

// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
delete element[onBlur]
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
delete element[initial]

if (String(element[propName]) !== initV) {
fireEvent.change(element)
}
}),
{
capture: true,
once: true,
},
)
}

if (descriptor) {
Object.defineProperty(element, propName, descriptor)
}
Expand Down

0 comments on commit 9600abb

Please sign in to comment.