forked from reactstrap/reactstrap
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathCollapse.js
More file actions
123 lines (111 loc) · 2.8 KB
/
Copy pathCollapse.js
File metadata and controls
123 lines (111 loc) · 2.8 KB
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
import React, { Component, PropTypes } from 'react';
import classNames from 'classnames';
import omit from 'lodash.omit';
import { mapToCssModules } from './utils';
const SHOW = 'SHOW';
const SHOWN = 'SHOWN';
const HIDE = 'HIDE';
const HIDDEN = 'HIDDEN';
const propTypes = {
isOpen: PropTypes.bool,
className: PropTypes.node,
tag: PropTypes.oneOfType([PropTypes.func, PropTypes.string]),
cssModule: PropTypes.object,
};
const defaultProps = {
isOpen: false,
tag: 'div'
};
class Collapse extends Component {
constructor(props) {
super(props);
this.state = {
collapse: props.isOpen ? SHOWN : HIDDEN,
height: props.isOpen ? null : 0
};
this.element = null;
}
componentWillReceiveProps(nextProps) {
const willOpen = nextProps.isOpen;
const collapse = this.state.collapse;
if (willOpen && collapse === HIDDEN) {
// will open
this.setState({ collapse: SHOW }, () => {
// the height transition will work after class "collapsing" applied
this.setState({ height: this.getHeight() });
this.transitionTag = setTimeout(() => {
this.setState({
collapse: SHOWN,
height: null
});
}, 350);
});
} else if (!willOpen && collapse === SHOWN) {
// will hide
this.setState({ height: this.getHeight() }, () => {
this.setState({
collapse: HIDE,
height: this.getHeight()
}, () => {
this.setState({ height: 0 });
});
});
this.transitionTag = setTimeout(() => {
this.setState({
collapse: HIDDEN,
height: null
});
}, 350);
}
// else: do nothing.
}
componentWillUnmount() {
clearTimeout(this.transitionTag);
}
getHeight() {
return this.element.scrollHeight;
}
render() {
const {
className,
cssModule,
tag: Tag,
...attributes
} = omit(this.props, ['isOpen']);
const { collapse, height } = this.state;
let collapseClass;
switch (collapse) {
case SHOW:
collapseClass = 'collapsing';
break;
case SHOWN:
collapseClass = 'collapse in';
break;
case HIDE:
collapseClass = 'collapsing';
break;
case HIDDEN:
collapseClass = 'collapse';
break;
default:
// HIDDEN
collapseClass = 'collapse';
}
const classes = mapToCssModules(classNames(
className,
collapseClass
), cssModule);
const style = height === null ? null : { height };
return (
<Tag
{...attributes}
style={{ ...attributes.style, ...style }}
className={classes}
ref={(c) => { this.element = c; }}
/>
);
}
}
Collapse.propTypes = propTypes;
Collapse.defaultProps = defaultProps;
export default Collapse;