Skip to content

Commit e33eb86

Browse files
committedNov 28, 2021
feat: apply modifier keys in pointer events (#751)
1 parent c12ee44 commit e33eb86

File tree

8 files changed

+101
-54
lines changed

8 files changed

+101
-54
lines changed
 

‎src/__tests__/pointer/index.ts

+25
Original file line numberDiff line numberDiff line change
@@ -387,3 +387,28 @@ test('asynchronous pointer', async () => {
387387
mousemove - button=0; buttons=0; detail=0
388388
`)
389389
})
390+
391+
test('apply modifiers from keyboardstate', async () => {
392+
const {element, getEvents} = setup(`<input/>`)
393+
394+
element.focus()
395+
let keyboardState = userEvent.keyboard('[ShiftLeft>]')
396+
userEvent.pointer({keys: '[MouseLeft]', target: element}, {keyboardState})
397+
keyboardState = userEvent.keyboard('[/ShiftLeft][ControlRight>]', {
398+
keyboardState,
399+
})
400+
userEvent.pointer({keys: '[MouseLeft]', target: element}, {keyboardState})
401+
keyboardState = userEvent.keyboard('[/ControlRight][AltLeft>]', {
402+
keyboardState,
403+
})
404+
userEvent.pointer({keys: '[MouseLeft]', target: element}, {keyboardState})
405+
keyboardState = userEvent.keyboard('[/AltLeft][MetaLeft>]', {keyboardState})
406+
userEvent.pointer({keys: '[MouseLeft]', target: element}, {keyboardState})
407+
408+
expect(getEvents('click')).toEqual([
409+
expect.objectContaining({shiftKey: true}),
410+
expect.objectContaining({ctrlKey: true}),
411+
expect.objectContaining({altKey: true}),
412+
expect.objectContaining({metaKey: true}),
413+
])
414+
})

‎src/pointer/index.ts

+13-13
Original file line numberDiff line numberDiff line change
@@ -1,38 +1,37 @@
11
import {getConfig as getDOMTestingLibraryConfig} from '@testing-library/dom'
2+
import {createKeyboardState} from '../keyboard'
23
import {parseKeyDef} from './parseKeyDef'
34
import {defaultKeyMap} from './keyMap'
45
import {
56
pointerAction,
67
PointerAction,
78
PointerActionTarget,
89
} from './pointerAction'
9-
import {pointerOptions, pointerState} from './types'
10+
import type {inputDeviceState, pointerOptions, pointerState} from './types'
1011

1112
export function pointer(
1213
input: PointerInput,
13-
options?: Partial<pointerOptions & {pointerState: pointerState; delay: 0}>,
14+
options?: Partial<pointerOptions & {delay: 0} & inputDeviceState>,
1415
): pointerState
1516
export function pointer(
1617
input: PointerInput,
17-
options: Partial<
18-
pointerOptions & {pointerState: pointerState; delay: number}
19-
>,
18+
options: Partial<pointerOptions & {delay: number} & inputDeviceState>,
2019
): Promise<pointerState>
2120
export function pointer(
2221
input: PointerInput,
23-
options: Partial<pointerOptions & {pointerState: pointerState}> = {},
22+
options: Partial<pointerOptions & inputDeviceState> = {},
2423
) {
25-
const {promise, state} = pointerImplementationWrapper(input, options)
24+
const {promise, pointerState} = pointerImplementationWrapper(input, options)
2625

2726
if ((options.delay ?? 0) > 0) {
2827
return getDOMTestingLibraryConfig().asyncWrapper(() =>
29-
promise.then(() => state),
28+
promise.then(() => pointerState),
3029
)
3130
} else {
3231
// prevent users from dealing with UnhandledPromiseRejectionWarning in sync call
3332
promise.catch(console.error)
3433

35-
return state
34+
return pointerState
3635
}
3736
}
3837

@@ -44,10 +43,11 @@ type PointerInput = PointerActionInput | Array<PointerActionInput>
4443

4544
export function pointerImplementationWrapper(
4645
input: PointerInput,
47-
config: Partial<pointerOptions & {pointerState: pointerState}>,
46+
config: Partial<pointerOptions & inputDeviceState>,
4847
) {
4948
const {
50-
pointerState: state = createPointerState(),
49+
pointerState = createPointerState(),
50+
keyboardState = createKeyboardState(),
5151
delay = 0,
5252
pointerMap = defaultKeyMap,
5353
} = config
@@ -73,8 +73,8 @@ export function pointerImplementationWrapper(
7373
})
7474

7575
return {
76-
promise: pointerAction(actions, options, state),
77-
state,
76+
promise: pointerAction(actions, options, {pointerState, keyboardState}),
77+
pointerState,
7878
}
7979
}
8080

‎src/pointer/pointerAction.ts

+7-6
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import {Coords, wait} from '../utils'
22
import {pointerMove, PointerMoveAction} from './pointerMove'
33
import {pointerPress, PointerPressAction} from './pointerPress'
4-
import {pointerOptions, pointerState} from './types'
4+
import {inputDeviceState, pointerOptions, pointerState} from './types'
55

66
export type PointerActionTarget = {
77
target?: Element
@@ -17,7 +17,7 @@ export type PointerAction = PointerActionTarget &
1717
export async function pointerAction(
1818
actions: PointerAction[],
1919
options: pointerOptions,
20-
state: pointerState,
20+
state: inputDeviceState,
2121
): Promise<unknown[]> {
2222
const ret: Array<Promise<void>> = []
2323

@@ -32,10 +32,11 @@ export async function pointerAction(
3232
: action.keyDef.pointerType
3333
: 'mouse'
3434

35-
const target = action.target ?? getPrevTarget(pointerName, state)
35+
const target =
36+
action.target ?? getPrevTarget(pointerName, state.pointerState)
3637
const coords = completeCoords({
37-
...(pointerName in state.position
38-
? state.position[pointerName].coords
38+
...(pointerName in state.pointerState.position
39+
? state.pointerState.position[pointerName].coords
3940
: undefined),
4041
...action.coords,
4142
})
@@ -55,7 +56,7 @@ export async function pointerAction(
5556
}
5657
}
5758

58-
delete state.activeClickCount
59+
delete state.pointerState.activeClickCount
5960

6061
return Promise.all(ret)
6162
}

‎src/pointer/pointerMove.ts

+7-8
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,15 @@
11
import {Coords, firePointerEvent, isDescendantOrSelf} from '../utils'
2-
import {pointerState, PointerTarget} from './types'
2+
import {inputDeviceState, PointerTarget} from './types'
33

44
export interface PointerMoveAction extends PointerTarget {
55
pointerName?: string
66
}
77

88
export async function pointerMove(
99
{pointerName = 'mouse', target, coords}: PointerMoveAction,
10-
state: pointerState,
10+
{pointerState, keyboardState}: inputDeviceState,
1111
): Promise<void> {
12-
if (!(pointerName in state.position)) {
12+
if (!(pointerName in pointerState.position)) {
1313
throw new Error(
1414
`Trying to move pointer "${pointerName}" which does not exist.`,
1515
)
@@ -20,7 +20,7 @@ export async function pointerMove(
2020
pointerType,
2121
target: prevTarget,
2222
coords: prevCoords,
23-
} = state.position[pointerName]
23+
} = pointerState.position[pointerName]
2424

2525
if (prevTarget && prevTarget !== target) {
2626
// Here we could probably calculate a few coords to a fake boundary(?)
@@ -42,7 +42,7 @@ export async function pointerMove(
4242
// Here we could probably calculate a few coords leading up to the final position
4343
fireMove(target, coords)
4444

45-
state.position[pointerName] = {pointerId, pointerType, target, coords}
45+
pointerState.position[pointerName] = {pointerId, pointerType, target, coords}
4646

4747
function fireMove(eventTarget: Element, eventCoords: Coords) {
4848
fire(eventTarget, 'pointermove', eventCoords)
@@ -71,9 +71,8 @@ export async function pointerMove(
7171

7272
function fire(eventTarget: Element, type: string, eventCoords: Coords) {
7373
return firePointerEvent(eventTarget, type, {
74-
buttons: state.pressed
75-
.filter(p => p.keyDef.pointerType === pointerType)
76-
.map(p => p.keyDef.button ?? 0),
74+
pointerState,
75+
keyboardState,
7776
coords: eventCoords,
7877
pointerId,
7978
pointerType,

‎src/pointer/pointerPress.ts

+27-23
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,10 @@
11
import {Coords, firePointerEvent} from '../utils'
2-
import type {pointerKey, pointerState, PointerTarget} from './types'
2+
import type {
3+
inputDeviceState,
4+
pointerKey,
5+
pointerState,
6+
PointerTarget,
7+
} from './types'
38

49
export interface PointerPressAction extends PointerTarget {
510
keyDef: pointerKey
@@ -9,9 +14,9 @@ export interface PointerPressAction extends PointerTarget {
914

1015
export async function pointerPress(
1116
{keyDef, releasePrevious, releaseSelf, target, coords}: PointerPressAction,
12-
state: pointerState,
17+
state: inputDeviceState,
1318
): Promise<void> {
14-
const previous = state.pressed.find(p => p.keyDef === keyDef)
19+
const previous = state.pointerState.pressed.find(p => p.keyDef === keyDef)
1520

1621
const pointerName =
1722
keyDef.pointerType === 'touch' ? keyDef.name : keyDef.pointerType
@@ -39,12 +44,12 @@ function down(
3944
keyDef: pointerKey,
4045
target: Element,
4146
coords: Coords,
42-
state: pointerState,
47+
{pointerState, keyboardState}: inputDeviceState,
4348
) {
4449
const {name, pointerType, button} = keyDef
45-
const pointerId = pointerType === 'mouse' ? 1 : getNextPointerId(state)
50+
const pointerId = pointerType === 'mouse' ? 1 : getNextPointerId(pointerState)
4651

47-
state.position[pointerName] = {
52+
pointerState.position[pointerName] = {
4853
pointerId,
4954
pointerType,
5055
target,
@@ -54,7 +59,7 @@ function down(
5459
let isMultiTouch = false
5560
let isPrimary = true
5661
if (pointerType !== 'mouse') {
57-
for (const obj of state.pressed) {
62+
for (const obj of pointerState.pressed) {
5863
// TODO: test multi device input across browsers
5964
// istanbul ignore else
6065
if (obj.keyDef.pointerType === pointerType) {
@@ -65,11 +70,11 @@ function down(
6570
}
6671
}
6772

68-
if (state.activeClickCount?.[0] !== name) {
69-
delete state.activeClickCount
73+
if (pointerState.activeClickCount?.[0] !== name) {
74+
delete pointerState.activeClickCount
7075
}
71-
const clickCount = Number(state.activeClickCount?.[1] ?? 0) + 1
72-
state.activeClickCount = [name, clickCount]
76+
const clickCount = Number(pointerState.activeClickCount?.[1] ?? 0) + 1
77+
pointerState.activeClickCount = [name, clickCount]
7378

7479
const pressObj = {
7580
keyDef,
@@ -80,15 +85,15 @@ function down(
8085
isPrimary,
8186
clickCount,
8287
}
83-
state.pressed.push(pressObj)
88+
pointerState.pressed.push(pressObj)
8489

8590
if (pointerType !== 'mouse') {
8691
fire('pointerover')
8792
fire('pointerenter')
8893
}
8994
if (
9095
pointerType !== 'mouse' ||
91-
!state.pressed.some(
96+
!pointerState.pressed.some(
9297
p => p.keyDef !== keyDef && p.keyDef.pointerType === pointerType,
9398
)
9499
) {
@@ -104,10 +109,9 @@ function down(
104109

105110
function fire(type: string) {
106111
return firePointerEvent(target, type, {
112+
pointerState,
113+
keyboardState,
107114
button,
108-
buttons: state.pressed
109-
.filter(p => p.keyDef.pointerType === pointerType)
110-
.map(p => p.keyDef.button ?? 0),
111115
clickCount,
112116
coords,
113117
isPrimary,
@@ -122,15 +126,15 @@ function up(
122126
{pointerType, button}: pointerKey,
123127
target: Element,
124128
coords: Coords,
125-
state: pointerState,
129+
{pointerState, keyboardState}: inputDeviceState,
126130
pressed: pointerState['pressed'][number],
127131
) {
128-
state.pressed = state.pressed.filter(p => p !== pressed)
132+
pointerState.pressed = pointerState.pressed.filter(p => p !== pressed)
129133

130134
const {isMultiTouch, isPrimary, pointerId, clickCount} = pressed
131135
let {unpreventedDefault} = pressed
132136

133-
state.position[pointerName] = {
137+
pointerState.position[pointerName] = {
134138
pointerId,
135139
pointerType,
136140
target,
@@ -141,7 +145,8 @@ function up(
141145

142146
if (
143147
pointerType !== 'mouse' ||
144-
!state.pressed.filter(p => p.keyDef.pointerType === pointerType).length
148+
!pointerState.pressed.filter(p => p.keyDef.pointerType === pointerType)
149+
.length
145150
) {
146151
fire('pointerup')
147152
}
@@ -169,10 +174,9 @@ function up(
169174

170175
function fire(type: string) {
171176
return firePointerEvent(target, type, {
177+
pointerState,
178+
keyboardState,
172179
button,
173-
buttons: state.pressed
174-
.filter(p => p.keyDef.pointerType === pointerType)
175-
.map(p => p.keyDef.button ?? 0),
176180
clickCount,
177181
coords,
178182
isPrimary,

‎src/pointer/types.ts

+6
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import {keyboardState} from 'keyboard/types'
12
import {Coords, MouseButton} from '../utils'
23

34
/**
@@ -61,3 +62,8 @@ export interface PointerTarget {
6162
target: Element
6263
coords: Coords
6364
}
65+
66+
export interface inputDeviceState {
67+
pointerState: pointerState
68+
keyboardState: keyboardState
69+
}

‎src/setup.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -160,7 +160,7 @@ function _setup(
160160

161161
// pointer needs typecasting because of the overloading
162162
pointer: ((...args: Parameters<typeof pointer>) => {
163-
args[1] = {...pointerApiDefaults, ...args[1], pointerState}
163+
args[1] = {...pointerApiDefaults, ...args[1], pointerState, keyboardState}
164164
const ret = pointer(...args) as pointerState | Promise<pointerState>
165165
if (ret instanceof Promise) {
166166
return ret.then(() => undefined)

‎src/utils/pointer/firePointerEvents.ts

+15-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
import {fireEvent} from '@testing-library/dom'
2+
import type {pointerState} from 'pointer/types'
3+
import type {keyboardState} from 'keyboard/types'
24
import {FakeEventInit, FakeMouseEvent, FakePointerEvent} from './fakeEvent'
35
import {getMouseButton, getMouseButtons, MouseButton} from './mouseButtons'
46

@@ -17,17 +19,19 @@ export function firePointerEvent(
1719
target: Element,
1820
type: string,
1921
{
22+
pointerState,
23+
keyboardState,
2024
pointerType,
2125
button,
22-
buttons,
2326
coords,
2427
pointerId,
2528
isPrimary,
2629
clickCount,
2730
}: {
31+
pointerState: pointerState
32+
keyboardState: keyboardState
2833
pointerType?: 'mouse' | 'pen' | 'touch'
2934
button?: MouseButton
30-
buttons: MouseButton[]
3135
coords: Coords
3236
pointerId?: number
3337
isPrimary?: boolean
@@ -41,6 +45,10 @@ export function firePointerEvent(
4145

4246
let init: FakeEventInit = {
4347
...coords,
48+
altKey: keyboardState.modifiers.alt,
49+
ctrlKey: keyboardState.modifiers.ctrl,
50+
metaKey: keyboardState.modifiers.meta,
51+
shiftKey: keyboardState.modifiers.shift,
4452
}
4553
if (Event === FakePointerEvent) {
4654
init = {...init, pointerId, pointerType}
@@ -52,7 +60,11 @@ export function firePointerEvent(
5260
['mousedown', 'mouseup', 'pointerdown', 'pointerup', 'click'].includes(type)
5361
) {
5462
init.button = getMouseButton(button ?? 0)
55-
init.buttons = getMouseButtons(...buttons)
63+
init.buttons = getMouseButtons(
64+
...pointerState.pressed
65+
.filter(p => p.keyDef.pointerType === pointerType)
66+
.map(p => p.keyDef.button ?? 0),
67+
)
5668
}
5769
if (['mousedown', 'mouseup', 'click'].includes(type)) {
5870
init.detail = clickCount

0 commit comments

Comments
 (0)
Please sign in to comment.