-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathuseSpotlight.js
172 lines (137 loc) · 4.47 KB
/
useSpotlight.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
import Spotlight from '@enact/spotlight';
import utilDOM from '@enact/ui/useScroll/utilDOM';
import {useEffect, useRef} from 'react';
const useSpotlightConfig = (props, instances) => {
// Hooks
useEffect(() => {
function lastFocusedPersist () {
const {spottable: {current: {lastFocusedIndex}}} = instances;
if (lastFocusedIndex != null) {
return {
container: false,
element: true,
key: lastFocusedIndex
};
}
}
function lastFocusedRestore ({key}, all) {
const placeholder = all.find(el => 'vlPlaceholder' in el.dataset);
if (placeholder) {
placeholder.dataset.index = key;
return placeholder;
}
return all.reduce((focused, node) => {
return focused || Number(node.dataset.index) === key && node;
}, null);
}
function configureSpotlight () {
const {spacing, spotlightId} = props;
Spotlight.set(spotlightId, {
enterTo: 'last-focused',
/*
* Returns the data-index as the key for last focused
*/
lastFocusedPersist,
/*
* Restores the data-index into the placeholder if it's the only element. Tries to find a
* matching child otherwise.
*/
lastFocusedRestore,
/*
* Directs spotlight focus to favor straight elements that are within range of `spacing`
* over oblique elements, like scroll buttons.
*/
obliqueMultiplier: spacing > 0 ? spacing : 1
});
}
configureSpotlight();
}, [props, instances]);
};
const getNumberValue = (index) => {
// using '+ operator' for string > number conversion based on performance: https://jsperf.com/convert-string-to-number-techniques/7
let number = +index;
// should return -1 if index is not a number or a negative value
return number >= 0 ? number : -1;
};
const useSpotlightRestore = (props, instances, context) => {
const {scrollContentRef, spottable} = instances;
const {focusByIndex, getItemNode} = context;
// Mutable value
const mutableRef = useRef({
preservedIndex: false,
lastSpotlightDirection: null,
restoreLastFocused: false
});
// Hooks
useEffect(restoreFocus);
// Functions
function handlePlaceholderFocus (ev) {
const placeholder = ev.currentTarget;
if (placeholder) {
const index = placeholder.dataset.index;
if (index) {
mutableRef.current.preservedIndex = getNumberValue(index);
mutableRef.current.lastSpotlightDirection = null;
mutableRef.current.restoreLastFocused = true;
}
}
}
function isPlaceholderFocused () {
const current = Spotlight.getCurrent();
if (current && current.dataset.vlPlaceholder && utilDOM.containsDangerously(scrollContentRef.current, current)) {
return true;
}
return false;
}
function restoreFocus () {
if (
mutableRef.current.restoreLastFocused &&
!isPlaceholderFocused()
) {
const
{spotlightId} = props,
itemNode = getItemNode(mutableRef.current.preservedIndex);
if (itemNode) {
// if we're supposed to restore focus and virtual list has positioned a set of items
// that includes lastFocusedIndex, clear the indicator
mutableRef.current.restoreLastFocused = false;
// try to focus the last focused item
spottable.current.isScrolledByJump = true;
const foundLastFocused = focusByIndex(mutableRef.current.preservedIndex, mutableRef.current.lastSpotlightDirection);
spottable.current.isScrolledByJump = false;
// but if that fails (because it isn't found or is disabled), focus the container so
// spotlight isn't lost
if (!foundLastFocused) {
mutableRef.current.restoreLastFocused = true;
Spotlight.focus(spotlightId);
}
}
}
}
function handleRestoreLastFocus ({firstIndex, lastIndex}) {
if (mutableRef.current.restoreLastFocused && mutableRef.current.preservedIndex >= firstIndex && mutableRef.current.preservedIndex <= lastIndex) {
restoreFocus();
}
}
function updateStatesAndBounds ({dataSize, moreInfo, numOfItems}) {
return (mutableRef.current.restoreLastFocused && numOfItems > 0 && mutableRef.current.preservedIndex < dataSize && (
mutableRef.current.preservedIndex < moreInfo.firstVisibleIndex || mutableRef.current.preservedIndex > moreInfo.lastVisibleIndex
));
}
function setPreservedIndex (index, direction = null) {
mutableRef.current.preservedIndex = index;
mutableRef.current.lastSpotlightDirection = direction;
mutableRef.current.restoreLastFocused = true;
}
// Return
return {
handlePlaceholderFocus,
handleRestoreLastFocus,
setPreservedIndex,
updateStatesAndBounds
};
};
export {
useSpotlightConfig,
useSpotlightRestore
};