2018-11-20 18:00:08 +00:00
|
|
|
/* @flow */
|
2018-08-04 10:20:08 +00:00
|
|
|
import withLanguage from 'Components/utils/withLanguage'
|
2018-11-20 18:00:08 +00:00
|
|
|
import React, { Component, PureComponent } from 'react'
|
2018-07-05 15:33:49 +00:00
|
|
|
import ReactCSSTransitionGroup from 'react-addons-css-transition-group'
|
2018-11-14 15:51:37 +00:00
|
|
|
import './AnimatedTargetValue.css'
|
2018-07-05 15:33:49 +00:00
|
|
|
|
2018-11-20 18:00:08 +00:00
|
|
|
type Props = {
|
|
|
|
value: ?number,
|
|
|
|
language: string
|
|
|
|
}
|
|
|
|
type State = {
|
|
|
|
difference: number
|
|
|
|
}
|
2018-11-14 15:51:37 +00:00
|
|
|
export default withLanguage(
|
2018-11-20 18:00:08 +00:00
|
|
|
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
|
2018-11-20 18:34:45 +00:00
|
|
|
}, 250)
|
2018-11-20 18:00:08 +00:00
|
|
|
}
|
|
|
|
format = value => {
|
|
|
|
return value == null
|
|
|
|
? ''
|
|
|
|
: Intl.NumberFormat(this.props.language, {
|
|
|
|
style: 'currency',
|
|
|
|
currency: 'EUR',
|
|
|
|
maximumFractionDigits: 0,
|
|
|
|
minimumFractionDigits: 0
|
|
|
|
}).format(value)
|
|
|
|
}
|
2018-11-14 15:51:37 +00:00
|
|
|
render() {
|
2018-11-20 18:00:08 +00:00
|
|
|
const formattedValue = this.format(this.props.value)
|
|
|
|
const formattedDifference = this.format(this.state.difference)
|
2018-11-21 15:33:03 +00:00
|
|
|
const shouldDisplayDifference =
|
|
|
|
Math.abs(this.state.difference) > 1 &&
|
|
|
|
formattedDifference !== formattedValue &&
|
|
|
|
this.props.value != null &&
|
|
|
|
this.state.difference < 0.5 * this.props.value
|
2018-11-14 15:51:37 +00:00
|
|
|
return (
|
2018-11-20 18:00:08 +00:00
|
|
|
<>
|
|
|
|
<span key={this.props.value} className="Rule-value">
|
2018-11-21 15:33:03 +00:00
|
|
|
{shouldDisplayDifference && (
|
|
|
|
<Evaporate
|
|
|
|
style={{
|
|
|
|
color: this.state.difference > 0 ? 'chartreuse' : 'red'
|
|
|
|
}}>
|
|
|
|
{(this.state.difference > 0 ? '+' : '') + formattedDifference}
|
|
|
|
</Evaporate>
|
|
|
|
)}{' '}
|
2018-11-20 18:00:08 +00:00
|
|
|
<span>{this.format(this.props.value)}</span>
|
2018-11-14 15:51:37 +00:00
|
|
|
</span>
|
2018-11-20 18:00:08 +00:00
|
|
|
</>
|
2018-11-14 15:51:37 +00:00
|
|
|
)
|
|
|
|
}
|
2018-07-05 15:33:49 +00:00
|
|
|
}
|
2018-11-14 15:51:37 +00:00
|
|
|
)
|
2018-11-20 18:00:08 +00:00
|
|
|
|
|
|
|
class Evaporate extends PureComponent<{ children: string, style: Object }> {
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<ReactCSSTransitionGroup
|
|
|
|
transitionName="evaporate"
|
2018-11-21 15:33:03 +00:00
|
|
|
transitionEnterTimeout={1600}
|
2018-11-20 18:00:08 +00:00
|
|
|
transitionLeaveTimeout={1}>
|
|
|
|
<span
|
|
|
|
key={this.props.children}
|
|
|
|
style={this.props.style}
|
|
|
|
className="evaporate">
|
|
|
|
{this.props.children}
|
|
|
|
</span>
|
|
|
|
</ReactCSSTransitionGroup>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|