This repository was archived by the owner on Feb 23, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 131
/
Copy pathProvider.js
104 lines (86 loc) · 2.68 KB
/
Provider.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
import React from "react";
import PropTypes from "prop-types";
import throttle from "lodash/throttle";
import { findActiveNode } from "helpers/dom";
const sortByPosition = (a, b) => {
const aY = a.ref.current.getBoundingClientRect().top;
const bY = b.ref.current.getBoundingClientRect().top;
return aY - bY;
};
let lastScrollPosition = 0;
export const SideNavProgressContext = React.createContext({
activeContent: null,
stopTrackingElement: () => {},
trackElement: () => {},
setActiveNode: () => {},
setIsNavClicked: () => {},
});
export const Provider = ({ children }) => {
const [activeContent, setActiveNode] = React.useState({ id: "", ref: null });
const [trackedElements, setTrackedElements] = React.useState([]);
const [isNavClicked, setIsNavClicked] = React.useState(false);
// We need to search just the refs pretty frequently, so memoize it so we
// don't generate a ton of garbage.
const elementRefs = React.useMemo(() => trackedElements.map((e) => e.ref), [
trackedElements,
]);
React.useEffect(() => {
const handler = throttle(() => {
// If we haven't scrolled at least 100 pixels, just bail.
if (Math.abs(window.scrollY - lastScrollPosition) < 100) {
return;
}
const isScrollingDown = window.scrollY > lastScrollPosition;
lastScrollPosition = window.scrollY;
const newActiveRef = findActiveNode(elementRefs, isScrollingDown);
if (isNavClicked) {
return;
}
const newActiveNode = trackedElements.find((e) => e.ref === newActiveRef);
if (newActiveNode && newActiveNode !== activeContent) {
setActiveNode(newActiveNode);
}
}, 60);
window.addEventListener("scroll", handler);
handler();
return () => {
window.removeEventListener("scroll", handler);
};
}, [activeContent, elementRefs, trackedElements, isNavClicked]);
const trackElement = React.useCallback(
(ref) => {
setTrackedElements((state) => [...state, ref].sort(sortByPosition));
},
[setTrackedElements],
);
const stopTrackingElement = React.useCallback(
(ref) => {
setTrackedElements((state) => state.filter((x) => x === ref));
},
[setTrackedElements],
);
const contextValue = React.useMemo(
() => ({
activeContent,
stopTrackingElement,
trackElement,
setActiveNode,
setIsNavClicked,
}),
[
activeContent,
stopTrackingElement,
trackElement,
setActiveNode,
setIsNavClicked,
],
);
return (
<SideNavProgressContext.Provider value={contextValue}>
{children}
</SideNavProgressContext.Provider>
);
};
Provider.propTypes = {
children: PropTypes.node.isRequired,
};