|
|
@ -8,6 +8,11 @@ const messages = defineMessages({ |
|
|
|
minutes: { id: 'relative_time.minutes', defaultMessage: '{number}m' }, |
|
|
|
minutes: { id: 'relative_time.minutes', defaultMessage: '{number}m' }, |
|
|
|
hours: { id: 'relative_time.hours', defaultMessage: '{number}h' }, |
|
|
|
hours: { id: 'relative_time.hours', defaultMessage: '{number}h' }, |
|
|
|
days: { id: 'relative_time.days', defaultMessage: '{number}d' }, |
|
|
|
days: { id: 'relative_time.days', defaultMessage: '{number}d' }, |
|
|
|
|
|
|
|
moments_remaining: { id: 'time_remaining.moments', defaultMessage: 'Moments remaining' }, |
|
|
|
|
|
|
|
seconds_remaining: { id: 'time_remaining.seconds', defaultMessage: '{number, plural, one {# second} other {# seconds}} left' }, |
|
|
|
|
|
|
|
minutes_remaining: { id: 'time_remaining.minutes', defaultMessage: '{number, plural, one {# minute} other {# minutes}} left' }, |
|
|
|
|
|
|
|
hours_remaining: { id: 'time_remaining.hours', defaultMessage: '{number, plural, one {# hour} other {# hours}} left' }, |
|
|
|
|
|
|
|
days_remaining: { id: 'time_remaining.days', defaultMessage: '{number, plural, one {# day} other {# days}} left' }, |
|
|
|
}); |
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
const dateFormatOptions = { |
|
|
|
const dateFormatOptions = { |
|
|
@ -86,6 +91,26 @@ export const timeAgoString = (intl, date, now, year) => { |
|
|
|
return relativeTime; |
|
|
|
return relativeTime; |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const timeRemainingString = (intl, date, now) => { |
|
|
|
|
|
|
|
const delta = date.getTime() - now; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let relativeTime; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (delta < 10 * SECOND) { |
|
|
|
|
|
|
|
relativeTime = intl.formatMessage(messages.moments_remaining); |
|
|
|
|
|
|
|
} else if (delta < MINUTE) { |
|
|
|
|
|
|
|
relativeTime = intl.formatMessage(messages.seconds_remaining, { number: Math.floor(delta / SECOND) }); |
|
|
|
|
|
|
|
} else if (delta < HOUR) { |
|
|
|
|
|
|
|
relativeTime = intl.formatMessage(messages.minutes_remaining, { number: Math.floor(delta / MINUTE) }); |
|
|
|
|
|
|
|
} else if (delta < DAY) { |
|
|
|
|
|
|
|
relativeTime = intl.formatMessage(messages.hours_remaining, { number: Math.floor(delta / HOUR) }); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
relativeTime = intl.formatMessage(messages.days_remaining, { number: Math.floor(delta / DAY) }); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return relativeTime; |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
export default @injectIntl |
|
|
|
export default @injectIntl |
|
|
|
class RelativeTimestamp extends React.Component { |
|
|
|
class RelativeTimestamp extends React.Component { |
|
|
|
|
|
|
|
|
|
|
@ -93,6 +118,7 @@ class RelativeTimestamp extends React.Component { |
|
|
|
intl: PropTypes.object.isRequired, |
|
|
|
intl: PropTypes.object.isRequired, |
|
|
|
timestamp: PropTypes.string.isRequired, |
|
|
|
timestamp: PropTypes.string.isRequired, |
|
|
|
year: PropTypes.number.isRequired, |
|
|
|
year: PropTypes.number.isRequired, |
|
|
|
|
|
|
|
futureDate: PropTypes.bool, |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
state = { |
|
|
|
state = { |
|
|
@ -145,10 +171,10 @@ class RelativeTimestamp extends React.Component { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
render () { |
|
|
|
render () { |
|
|
|
const { timestamp, intl, year } = this.props; |
|
|
|
const { timestamp, intl, year, futureDate } = this.props; |
|
|
|
|
|
|
|
|
|
|
|
const date = new Date(timestamp); |
|
|
|
const date = new Date(timestamp); |
|
|
|
const relativeTime = timeAgoString(intl, date, this.state.now, year); |
|
|
|
const relativeTime = futureDate ? timeRemainingString(intl, date, this.state.now) : timeAgoString(intl, date, this.state.now, year); |
|
|
|
|
|
|
|
|
|
|
|
return ( |
|
|
|
return ( |
|
|
|
<time dateTime={timestamp} title={intl.formatDate(date, dateFormatOptions)}> |
|
|
|
<time dateTime={timestamp} title={intl.formatDate(date, dateFormatOptions)}> |
|
|
|