mon-entreprise/source/components/AnimatedTargetValue.js

85 lines
2.1 KiB
JavaScript

/* @flow */
import withLanguage from 'Components/utils/withLanguage'
import React, { Component, PureComponent } from 'react'
import ReactCSSTransitionGroup from 'react-addons-css-transition-group'
import './AnimatedTargetValue.css'
type Props = {
value: ?number,
language: string
}
type State = {
difference: number
}
export default withLanguage(
class AnimatedTargetValue extends Component<Props, State> {
previousValue: ?number = null
timeoutId: ?TimeoutID = null
state = { difference: 0 }
componentDidUpdate(prevProps) {
if (prevProps.value === this.props.value) {
return
}
if (this.timeoutId) {
clearTimeout(this.timeoutId)
}
this.previousValue =
this.previousValue === null ? prevProps.value : this.previousValue
this.timeoutId = setTimeout(() => {
this.setState({
difference: (this.props.value || 0) - (this.previousValue || 0)
})
this.previousValue = null
this.timeoutId = null
}, 250)
}
format = value => {
return value == null ? '' : value
}
render() {
const formattedValue = this.format(this.props.value)
const formattedDifference = this.format(this.state.difference)
const shouldDisplayDifference =
Math.abs(this.state.difference) > 1 &&
formattedDifference !== formattedValue &&
this.props.value != null &&
this.state.difference < 0.5 * this.props.value
return (
<>
<span key={this.props.value} className="Rule-value">
{shouldDisplayDifference && (
<Evaporate
style={{
color: this.state.difference > 0 ? 'chartreuse' : 'red'
}}>
{(this.state.difference > 0 ? '+' : '') + formattedDifference}
</Evaporate>
)}{' '}
<span>{this.format(this.props.value)}</span>
</span>
</>
)
}
}
)
class Evaporate extends PureComponent<{ children: string, style: Object }> {
render() {
return (
<ReactCSSTransitionGroup
transitionName="evaporate"
transitionEnterTimeout={1600}
transitionLeaveTimeout={1}>
<span
key={this.props.children}
style={this.props.style}
className="evaporate">
{this.props.children}
</span>
</ReactCSSTransitionGroup>
)
}
}