forked from hmmgo/react-floating-scrollbar
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscrollbar.jsx
118 lines (102 loc) · 3.52 KB
/
scrollbar.jsx
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
import React, { Fragment, Component } from 'react';
import PropTypes from 'prop-types';
class ReactFlScrollbar extends Component {
constructor(props) {
super(props);
this.scrollRef = React.createRef();
this.scrollBlockRef = React.createRef();
this.state = {
contentRefEl: props.contentRef.current,
scrollRefEl: this.scrollRef.current,
visible: true,
preventSyncCont: false,
preventSyncSbar: false,
};
}
componentDidMount() {
const contentRefEl = this.props.contentRef.current;
const elToScroll = document.getElementsByClassName(this.props.innerScrollableClass)[0];
const scrollWidth = contentRefEl && elToScroll ? elToScroll.scrollWidth : 0;
const scrollRefEl = this.scrollRef.current;
const scrollBlockRefEl = this.scrollBlockRef.current;
const lots = this;
if (contentRefEl && elToScroll) {
elToScroll.onscroll = (e) => {
lots.syncSbar(e.target, true);
};
elToScroll.onfocus = () => {
setTimeout(lots.syncSbar.bind(lots, contentRefEl), 0);
};
}
if (scrollRefEl) {
const elToScrolls = document.getElementsByClassName('react-fl-scrollbar')[0];
elToScrolls.onscroll = (e) => {
lots.syncCont(e.target, true);
};
}
if (scrollWidth > 0) {
scrollBlockRefEl.style.width = `${scrollWidth}px`;
}
lots.setState({ contentRefEl, scrollRefEl, scrollBlockRefEl });
}
componentDidUpdate() {
const contentRefEl = this.props.contentRef.current;
const scrollBlockRefEl = this.scrollBlockRef.current;
const elToScroll = document.getElementsByClassName(this.props.innerScrollableClass)[0];
const scrollWidth = contentRefEl && elToScroll ? elToScroll.scrollWidth : 0;
scrollBlockRefEl.style.width = `${scrollWidth}px`;
}
componentWillUnmount() {
const contentRefEl = this.props.contentRef.current;
const elToScroll = document.getElementsByClassName(this.props.innerScrollableClass)[0];
const scrollRefEl = this.scrollRef.current;
if (contentRefEl && elToScroll) {
elToScroll.onscroll = null;
elToScroll.onfocus = null;
}
if (scrollRefEl) {
scrollRefEl.onscroll = null;
}
}
syncSbar(sender, preventSyncCont) {
if (this.state.preventSyncSbar) {
this.setState({ preventSyncSbar: false });
return;
}
this.setState({ preventSyncCont: !!preventSyncCont });
this.scrollRef.current.scrollLeft = sender.scrollLeft;
}
syncCont(sender, preventSyncSbar) {
const contentRefEl = this.props.contentRef.current;
const elToScroll = document.getElementsByClassName(this.props.innerScrollableClass)[0];
if (this.state.preventSyncCont) {
this.setState({ preventSyncCont: false });
return;
}
this.setState({ preventSyncSbar: !!preventSyncSbar });
if (contentRefEl && elToScroll) {
elToScroll.scrollLeft = sender.scrollLeft;
}
}
render() {
const { contentRefEl } = this.state;
return (
<Fragment>
{this.props.children}
<div
className={'react-fl-scrollbar'}
style={{ width: `${contentRefEl ? contentRefEl.offsetWidth : 0}px`,
left: `${contentRefEl ? contentRefEl.getBoundingClientRect().left : 0}px` }}
ref={this.scrollRef}
>
<div ref={this.scrollBlockRef} />
</div>
</Fragment>);
}
}
ReactFlScrollbar.propTypes = {
contentRef: PropTypes.object.isRequired,
innerScrollableClass: PropTypes.string.isRequired,
children: PropTypes.node.isRequired,
};
export default ReactFlScrollbar;