2019-03-03 21:18:23 +00:00
import PropTypes from 'prop-types' ;
2023-05-23 15:15:17 +00:00
2019-03-03 21:18:23 +00:00
import { defineMessages , injectIntl , FormattedMessage } from 'react-intl' ;
2023-05-23 15:15:17 +00:00
2019-03-03 21:18:23 +00:00
import classNames from 'classnames' ;
2023-05-23 15:15:17 +00:00
import ImmutablePropTypes from 'react-immutable-proptypes' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
2019-03-06 04:35:52 +00:00
import escapeTextContentForBrowser from 'escape-html' ;
2023-05-23 15:15:17 +00:00
import spring from 'react-motion/lib/spring' ;
2024-01-16 10:27:26 +00:00
import CheckIcon from '@/material-icons/400-24px/check.svg?react' ;
2023-05-23 15:15:17 +00:00
import { Icon } from 'mastodon/components/icon' ;
2019-03-06 04:35:52 +00:00
import emojify from 'mastodon/features/emoji/emoji' ;
2023-05-23 15:15:17 +00:00
import Motion from 'mastodon/features/ui/util/optional_motion' ;
2024-05-19 17:07:32 +00:00
import { identityContextPropShape , withIdentity } from 'mastodon/identity_context' ;
2023-05-23 15:15:17 +00:00
2023-05-09 01:11:56 +00:00
import { RelativeTimestamp } from './relative_timestamp' ;
2019-03-03 21:18:23 +00:00
const messages = defineMessages ( {
2021-10-13 02:59:31 +00:00
closed : {
id : 'poll.closed' ,
defaultMessage : 'Closed' ,
} ,
voted : {
id : 'poll.voted' ,
defaultMessage : 'You voted for this answer' ,
} ,
votes : {
id : 'poll.votes' ,
defaultMessage : '{votes, plural, one {# vote} other {# votes}}' ,
} ,
2019-03-03 21:18:23 +00:00
} ) ;
2019-03-20 16:29:12 +00:00
const makeEmojiMap = record => record . get ( 'emojis' ) . reduce ( ( obj , emoji ) => {
obj [ ` : ${ emoji . get ( 'shortcode' ) } : ` ] = emoji . toJS ( ) ;
return obj ;
} , { } ) ;
2019-03-03 21:18:23 +00:00
class Poll extends ImmutablePureComponent {
static propTypes = {
2024-05-19 17:07:32 +00:00
identity : identityContextPropShape ,
2019-03-03 22:45:02 +00:00
poll : ImmutablePropTypes . map ,
2023-02-26 19:13:27 +00:00
lang : PropTypes . string ,
2019-03-03 21:18:23 +00:00
intl : PropTypes . object . isRequired ,
disabled : PropTypes . bool ,
2020-04-16 18:16:20 +00:00
refresh : PropTypes . func ,
2020-04-17 19:54:25 +00:00
onVote : PropTypes . func ,
2019-03-03 21:18:23 +00:00
} ;
state = {
selected : { } ,
2019-09-16 12:32:26 +00:00
expired : null ,
2019-03-03 21:18:23 +00:00
} ;
2019-09-16 12:32:26 +00:00
static getDerivedStateFromProps ( props , state ) {
2023-06-01 11:46:19 +00:00
const { poll } = props ;
2019-10-27 11:45:55 +00:00
const expires _at = poll . get ( 'expires_at' ) ;
2023-06-01 11:46:19 +00:00
const expired = poll . get ( 'expired' ) || expires _at !== null && ( new Date ( expires _at ) ) . getTime ( ) < Date . now ( ) ;
2019-09-16 12:32:26 +00:00
return ( expired === state . expired ) ? null : { expired } ;
}
componentDidMount ( ) {
this . _setupTimer ( ) ;
}
componentDidUpdate ( ) {
this . _setupTimer ( ) ;
}
componentWillUnmount ( ) {
clearTimeout ( this . _timer ) ;
}
_setupTimer ( ) {
2023-06-01 11:46:19 +00:00
const { poll } = this . props ;
2019-09-16 12:32:26 +00:00
clearTimeout ( this . _timer ) ;
if ( ! this . state . expired ) {
2023-06-01 11:46:19 +00:00
const delay = ( new Date ( poll . get ( 'expires_at' ) ) ) . getTime ( ) - Date . now ( ) ;
2019-09-16 12:32:26 +00:00
this . _timer = setTimeout ( ( ) => {
this . setState ( { expired : true } ) ;
} , delay ) ;
}
}
2019-12-03 18:53:16 +00:00
_toggleOption = value => {
2019-03-03 21:18:23 +00:00
if ( this . props . poll . get ( 'multiple' ) ) {
const tmp = { ... this . state . selected } ;
2019-03-04 00:54:14 +00:00
if ( tmp [ value ] ) {
delete tmp [ value ] ;
} else {
tmp [ value ] = true ;
}
2019-03-03 21:18:23 +00:00
this . setState ( { selected : tmp } ) ;
} else {
const tmp = { } ;
tmp [ value ] = true ;
this . setState ( { selected : tmp } ) ;
}
2023-01-30 00:45:35 +00:00
} ;
2019-12-03 18:53:16 +00:00
handleOptionChange = ( { target : { value } } ) => {
this . _toggleOption ( value ) ;
2019-03-03 21:18:23 +00:00
} ;
2019-12-03 18:53:16 +00:00
handleOptionKeyPress = ( e ) => {
if ( e . key === 'Enter' || e . key === ' ' ) {
this . _toggleOption ( e . target . getAttribute ( 'data-index' ) ) ;
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
}
2023-01-30 00:45:35 +00:00
} ;
2019-12-03 18:53:16 +00:00
2019-03-03 21:18:23 +00:00
handleVote = ( ) => {
if ( this . props . disabled ) {
return ;
}
2020-04-17 19:54:25 +00:00
this . props . onVote ( Object . keys ( this . state . selected ) ) ;
2019-03-03 21:18:23 +00:00
} ;
handleRefresh = ( ) => {
if ( this . props . disabled ) {
return ;
}
2020-04-16 18:16:20 +00:00
this . props . refresh ( ) ;
2019-03-03 21:18:23 +00:00
} ;
2023-07-05 08:32:04 +00:00
handleReveal = ( ) => {
this . setState ( { revealed : true } ) ;
2023-10-09 11:38:29 +00:00
} ;
2023-07-05 08:32:04 +00:00
2019-09-16 12:32:26 +00:00
renderOption ( option , optionIndex , showResults ) {
2023-02-26 19:13:27 +00:00
const { poll , lang , disabled , intl } = this . props ;
2019-09-29 20:58:01 +00:00
const pollVotesCount = poll . get ( 'voters_count' ) || poll . get ( 'votes_count' ) ;
const percent = pollVotesCount === 0 ? 0 : ( option . get ( 'votes_count' ) / pollVotesCount ) * 100 ;
const leading = poll . get ( 'options' ) . filterNot ( other => other . get ( 'title' ) === option . get ( 'title' ) ) . every ( other => option . get ( 'votes_count' ) >= other . get ( 'votes_count' ) ) ;
const active = ! ! this . state . selected [ ` ${ optionIndex } ` ] ;
const voted = option . get ( 'voted' ) || ( poll . get ( 'own_votes' ) && poll . get ( 'own_votes' ) . includes ( optionIndex ) ) ;
2019-03-03 21:18:23 +00:00
2023-05-31 22:10:21 +00:00
const title = option . getIn ( [ 'translation' , 'title' ] ) || option . get ( 'title' ) ;
let titleHtml = option . getIn ( [ 'translation' , 'titleHtml' ] ) || option . get ( 'titleHtml' ) ;
if ( ! titleHtml ) {
2019-03-20 16:29:12 +00:00
const emojiMap = makeEmojiMap ( poll ) ;
2023-05-31 22:10:21 +00:00
titleHtml = emojify ( escapeTextContentForBrowser ( title ) , emojiMap ) ;
2019-03-20 16:29:12 +00:00
}
2019-03-03 21:18:23 +00:00
return (
< li key = { option . get ( 'title' ) } >
2020-04-02 15:10:55 +00:00
< label className = { classNames ( 'poll__option' , { selectable : ! showResults } ) } >
2019-03-03 21:18:23 +00:00
< input
name = 'vote-options'
type = { poll . get ( 'multiple' ) ? 'checkbox' : 'radio' }
value = { optionIndex }
checked = { active }
onChange = { this . handleOptionChange }
2019-03-04 00:54:14 +00:00
disabled = { disabled }
2019-03-03 21:18:23 +00:00
/ >
2019-12-03 18:53:16 +00:00
{ ! showResults && (
< span
className = { classNames ( 'poll__input' , { checkbox : poll . get ( 'multiple' ) , active } ) }
2023-04-04 14:33:44 +00:00
tabIndex = { 0 }
2019-12-03 18:53:16 +00:00
role = { poll . get ( 'multiple' ) ? 'checkbox' : 'radio' }
onKeyPress = { this . handleOptionKeyPress }
aria - checked = { active }
2023-05-31 22:10:21 +00:00
aria - label = { title }
2023-02-26 19:13:27 +00:00
lang = { lang }
2019-12-03 18:53:16 +00:00
data - index = { optionIndex }
/ >
) }
2021-10-13 02:59:31 +00:00
{ showResults && (
< span
className = 'poll__number'
title = { intl . formatMessage ( messages . votes , {
votes : option . get ( 'votes_count' ) ,
} ) }
>
{ Math . round ( percent ) } %
< / span >
) }
2019-03-03 21:18:23 +00:00
2020-04-02 15:10:55 +00:00
< span
2021-01-22 09:09:23 +00:00
className = 'poll__option__text translate'
2023-02-26 19:13:27 +00:00
lang = { lang }
2023-05-31 22:10:21 +00:00
dangerouslySetInnerHTML = { { _ _html : titleHtml } }
2020-04-02 15:10:55 +00:00
/ >
{ ! ! voted && < span className = 'poll__voted' >
2023-10-24 17:45:08 +00:00
< Icon id = 'check' icon = { CheckIcon } className = 'poll__voted__mark' title = { intl . formatMessage ( messages . voted ) } / >
2020-04-02 15:10:55 +00:00
< / span > }
2019-03-03 21:18:23 +00:00
< / label >
2020-04-02 15:10:55 +00:00
{ showResults && (
< Motion defaultStyle = { { width : 0 } } style = { { width : spring ( percent , { stiffness : 180 , damping : 12 } ) } } >
{ ( { width } ) =>
< span className = { classNames ( 'poll__chart' , { leading } ) } style = { { width : ` ${ width } % ` } } / >
}
< / Motion >
) }
2019-03-03 21:18:23 +00:00
< / li >
) ;
}
render ( ) {
const { poll , intl } = this . props ;
2023-07-05 08:32:04 +00:00
const { revealed , expired } = this . state ;
2019-03-03 22:45:02 +00:00
if ( ! poll ) {
return null ;
}
2019-09-16 12:32:26 +00:00
const timeRemaining = expired ? intl . formatMessage ( messages . closed ) : < RelativeTimestamp timestamp = { poll . get ( 'expires_at' ) } futureDate / > ;
2023-07-05 08:32:04 +00:00
const showResults = poll . get ( 'voted' ) || revealed || expired ;
2019-03-03 22:45:02 +00:00
const disabled = this . props . disabled || Object . entries ( this . state . selected ) . every ( item => ! item ) ;
2019-03-03 21:18:23 +00:00
2019-09-29 20:58:01 +00:00
let votesCount = null ;
if ( poll . get ( 'voters_count' ) !== null && poll . get ( 'voters_count' ) !== undefined ) {
votesCount = < FormattedMessage id = 'poll.total_people' defaultMessage = '{count, plural, one {# person} other {# people}}' values = { { count : poll . get ( 'voters_count' ) } } / > ;
} else {
votesCount = < FormattedMessage id = 'poll.total_votes' defaultMessage = '{count, plural, one {# vote} other {# votes}}' values = { { count : poll . get ( 'votes_count' ) } } / > ;
}
2019-03-03 21:18:23 +00:00
return (
< div className = 'poll' >
< ul >
2019-09-16 12:32:26 +00:00
{ poll . get ( 'options' ) . map ( ( option , i ) => this . renderOption ( option , i , showResults ) ) }
2019-03-03 21:18:23 +00:00
< / ul >
< div className = 'poll__footer' >
2024-05-19 17:07:32 +00:00
{ ! showResults && < button className = 'button button-secondary' disabled = { disabled || ! this . props . identity . signedIn } onClick = { this . handleVote } > < FormattedMessage id = 'poll.vote' defaultMessage = 'Vote' / > < / button > }
2023-07-05 08:32:04 +00:00
{ ! showResults && < > < button className = 'poll__link' onClick = { this . handleReveal } > < FormattedMessage id = 'poll.reveal' defaultMessage = 'See results' / > < / button > · < / > }
{ showResults && ! this . props . disabled && < > < button className = 'poll__link' onClick = { this . handleRefresh } > < FormattedMessage id = 'poll.refresh' defaultMessage = 'Refresh' / > < / button > · < / > }
2019-09-29 20:58:01 +00:00
{ votesCount }
2023-07-05 08:32:04 +00:00
{ poll . get ( 'expires_at' ) && < > · { timeRemaining } < / > }
2019-03-03 21:18:23 +00:00
< / div >
< / div >
) ;
}
}
2023-03-24 02:17:53 +00:00
2024-05-19 17:07:32 +00:00
export default injectIntl ( withIdentity ( Poll ) ) ;