2019-12-07 13:28:52 +02:00
|
|
|
import React from 'react';
|
2017-07-22 18:57:38 +03:00
|
|
|
import { Motion, spring } from 'react-motion';
|
2019-12-07 21:02:00 +02:00
|
|
|
import MeasureHeight from 'app/components/MeasureHeight';
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2017-11-19 20:16:15 +02:00
|
|
|
import styles from './slide-motion.scss';
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
interface State {
|
|
|
|
[stepHeight: string]: number;
|
|
|
|
version: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
class SlideMotion extends React.Component<
|
2019-11-27 11:03:32 +02:00
|
|
|
{
|
2019-12-07 13:28:52 +02:00
|
|
|
activeStep: number;
|
|
|
|
children: React.ReactNode;
|
2019-11-27 11:03:32 +02:00
|
|
|
},
|
2019-12-07 13:28:52 +02:00
|
|
|
State
|
2019-11-27 11:03:32 +02:00
|
|
|
> {
|
2019-12-07 13:28:52 +02:00
|
|
|
state: State = {
|
2019-11-27 11:03:32 +02:00
|
|
|
version: 0,
|
|
|
|
};
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
isHeightMeasured: boolean;
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
componentWillReceiveProps() {
|
|
|
|
// mark this view as dirty to re-measure height
|
|
|
|
this.setState({
|
|
|
|
version: this.state.version + 1,
|
|
|
|
});
|
|
|
|
}
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
render() {
|
|
|
|
const { activeStep, children } = this.props;
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
const { version } = this.state;
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
const activeStepHeight = this.state[`step${activeStep}Height`] || 0;
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
// a hack to disable height animation on first render
|
|
|
|
const { isHeightMeasured } = this;
|
|
|
|
this.isHeightMeasured = isHeightMeasured || activeStepHeight > 0;
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
const motionStyle = {
|
|
|
|
transform: spring(activeStep * 100, {
|
|
|
|
stiffness: 500,
|
|
|
|
damping: 50,
|
|
|
|
precision: 0.5,
|
|
|
|
}),
|
|
|
|
height: isHeightMeasured
|
|
|
|
? spring(activeStepHeight, {
|
|
|
|
stiffness: 500,
|
|
|
|
damping: 20,
|
|
|
|
precision: 0.5,
|
|
|
|
})
|
|
|
|
: activeStepHeight,
|
|
|
|
};
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
|
|
|
<Motion style={motionStyle}>
|
2019-12-07 13:28:52 +02:00
|
|
|
{(interpolatingStyle: { height: number; transform: string }) => (
|
2019-11-27 11:03:32 +02:00
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
overflow: 'hidden',
|
|
|
|
height: `${interpolatingStyle.height}px`,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className={styles.container}
|
|
|
|
style={{
|
|
|
|
WebkitTransform: `translateX(-${interpolatingStyle.transform}%)`,
|
|
|
|
transform: `translateX(-${interpolatingStyle.transform}%)`,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{React.Children.map(children, (child, index) => (
|
|
|
|
<MeasureHeight
|
|
|
|
className={styles.item}
|
|
|
|
onMeasure={this.onStepMeasure(index)}
|
|
|
|
state={version}
|
|
|
|
key={index}
|
|
|
|
>
|
|
|
|
{child}
|
|
|
|
</MeasureHeight>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Motion>
|
|
|
|
);
|
|
|
|
}
|
2017-07-22 18:57:38 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
onStepMeasure(step: number) {
|
|
|
|
return (height: number) =>
|
|
|
|
this.setState({
|
|
|
|
[`step${step}Height`]: height,
|
|
|
|
});
|
|
|
|
}
|
2017-07-22 18:57:38 +03:00
|
|
|
}
|
2019-12-07 13:28:52 +02:00
|
|
|
|
|
|
|
export default SlideMotion;
|