Merge branch 'next-polls' into 'next'
Next: Polls See merge request soapbox-pub/soapbox-fe!1153
This commit is contained in:
commit
37a40e9f67
|
@ -9,6 +9,7 @@ import spring from 'react-motion/lib/spring';
|
||||||
import { openModal } from 'soapbox/actions/modals';
|
import { openModal } from 'soapbox/actions/modals';
|
||||||
import { vote, fetchPoll } from 'soapbox/actions/polls';
|
import { vote, fetchPoll } from 'soapbox/actions/polls';
|
||||||
import Icon from 'soapbox/components/icon';
|
import Icon from 'soapbox/components/icon';
|
||||||
|
import { Text } from 'soapbox/components/ui';
|
||||||
import Motion from 'soapbox/features/ui/util/optional_motion';
|
import Motion from 'soapbox/features/ui/util/optional_motion';
|
||||||
import SoapboxPropTypes from 'soapbox/utils/soapbox_prop_types';
|
import SoapboxPropTypes from 'soapbox/utils/soapbox_prop_types';
|
||||||
|
|
||||||
|
@ -106,7 +107,7 @@ class Poll extends ImmutablePureComponent {
|
||||||
{showResults && (
|
{showResults && (
|
||||||
<Motion defaultStyle={{ width: 0 }} style={{ width: spring(percent, { stiffness: 180, damping: 12 }) }}>
|
<Motion defaultStyle={{ width: 0 }} style={{ width: spring(percent, { stiffness: 180, damping: 12 }) }}>
|
||||||
{({ width }) =>
|
{({ width }) =>
|
||||||
<span className={classNames('poll__chart', { leading })} style={{ width: `${width}%` }} />
|
<span className={classNames('poll__chart bg-gray-300 dark:bg-slate-900', { 'bg-primary-300 dark:bg-primary-400': leading })} style={{ width: `${width}%` }} />
|
||||||
}
|
}
|
||||||
</Motion>
|
</Motion>
|
||||||
)}
|
)}
|
||||||
|
@ -163,9 +164,15 @@ class Poll extends ImmutablePureComponent {
|
||||||
|
|
||||||
<div className='poll__footer'>
|
<div className='poll__footer'>
|
||||||
{!showResults && <button className='button button-secondary' disabled={disabled} onClick={this.handleVote}><FormattedMessage id='poll.vote' defaultMessage='Vote' /></button>}
|
{!showResults && <button className='button button-secondary' disabled={disabled} onClick={this.handleVote}><FormattedMessage id='poll.vote' defaultMessage='Vote' /></button>}
|
||||||
{showResults && !this.props.disabled && <span><button className='poll__link' onClick={this.handleRefresh}><FormattedMessage id='poll.refresh' defaultMessage='Refresh' /></button> · </span>}
|
<Text>
|
||||||
|
{showResults && !this.props.disabled && (
|
||||||
|
<span><button className='poll__link' onClick={this.handleRefresh}>
|
||||||
|
<Text><FormattedMessage id='poll.refresh' defaultMessage='Refresh' /></Text>
|
||||||
|
</button> · </span>
|
||||||
|
)}
|
||||||
<FormattedMessage id='poll.total_votes' defaultMessage='{count, plural, one {# vote} other {# votes}}' values={{ count: poll.get('votes_count') }} />
|
<FormattedMessage id='poll.total_votes' defaultMessage='{count, plural, one {# vote} other {# votes}}' values={{ count: poll.get('votes_count') }} />
|
||||||
{poll.get('expires_at') && <span> · {timeRemaining}</span>}
|
{poll.get('expires_at') && <span> · {timeRemaining}</span>}
|
||||||
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
|
@ -1,28 +0,0 @@
|
||||||
import { Map as ImmutableMap } from 'immutable';
|
|
||||||
|
|
||||||
import { POLLS_IMPORT } from 'soapbox/actions/importer';
|
|
||||||
import { normalizeStatus } from 'soapbox/normalizers/status';
|
|
||||||
|
|
||||||
// HOTFIX: Convert the poll into a fake status to normalize it...
|
|
||||||
// TODO: get rid of POLLS_IMPORT and use STATUS_IMPORT here.
|
|
||||||
const normalizePoll = poll => {
|
|
||||||
const status = { poll };
|
|
||||||
return normalizeStatus(status).poll;
|
|
||||||
};
|
|
||||||
|
|
||||||
const importPolls = (state, polls) => {
|
|
||||||
return state.withMutations(map => {
|
|
||||||
return polls.forEach(poll => map.set(poll.id, normalizePoll(poll)));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const initialState = ImmutableMap();
|
|
||||||
|
|
||||||
export default function polls(state = initialState, action) {
|
|
||||||
switch(action.type) {
|
|
||||||
case POLLS_IMPORT:
|
|
||||||
return importPolls(state, action.polls);
|
|
||||||
default:
|
|
||||||
return state;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
import { Map as ImmutableMap } from 'immutable';
|
||||||
|
|
||||||
|
import { POLLS_IMPORT } from 'soapbox/actions/importer';
|
||||||
|
import { normalizeStatus } from 'soapbox/normalizers/status';
|
||||||
|
|
||||||
|
import type { AnyAction } from 'redux';
|
||||||
|
import type { Poll, APIEntity, EmbeddedEntity } from 'soapbox/types/entities';
|
||||||
|
|
||||||
|
type State = ImmutableMap<string, Poll>;
|
||||||
|
|
||||||
|
// HOTFIX: Convert the poll into a fake status to normalize it...
|
||||||
|
// TODO: get rid of POLLS_IMPORT and use STATUS_IMPORT here.
|
||||||
|
const normalizePoll = (poll: any): EmbeddedEntity<Poll> => {
|
||||||
|
const status = { poll };
|
||||||
|
return normalizeStatus(status).poll;
|
||||||
|
};
|
||||||
|
|
||||||
|
const importPolls = (state: State, polls: Array<APIEntity>) => {
|
||||||
|
return state.withMutations(map => {
|
||||||
|
return polls.forEach(poll => {
|
||||||
|
const normalPoll = normalizePoll(poll);
|
||||||
|
|
||||||
|
if (normalPoll && typeof normalPoll === 'object') {
|
||||||
|
map.set(normalPoll.id, normalPoll);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const initialState: State = ImmutableMap();
|
||||||
|
|
||||||
|
export default function polls(state: State = initialState, action: AnyAction): State {
|
||||||
|
switch(action.type) {
|
||||||
|
case POLLS_IMPORT:
|
||||||
|
return importPolls(state, action.polls);
|
||||||
|
default:
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,6 +1,5 @@
|
||||||
import escapeTextContentForBrowser from 'escape-html';
|
import escapeTextContentForBrowser from 'escape-html';
|
||||||
import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
|
import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
|
||||||
import { AnyAction } from 'redux';
|
|
||||||
|
|
||||||
import emojify from 'soapbox/features/emoji/emoji';
|
import emojify from 'soapbox/features/emoji/emoji';
|
||||||
import { normalizeStatus } from 'soapbox/normalizers';
|
import { normalizeStatus } from 'soapbox/normalizers';
|
||||||
|
@ -32,6 +31,8 @@ import {
|
||||||
} from '../actions/statuses';
|
} from '../actions/statuses';
|
||||||
import { TIMELINE_DELETE } from '../actions/timelines';
|
import { TIMELINE_DELETE } from '../actions/timelines';
|
||||||
|
|
||||||
|
import type { AnyAction } from 'redux';
|
||||||
|
|
||||||
const domParser = new DOMParser();
|
const domParser = new DOMParser();
|
||||||
|
|
||||||
type StatusRecord = ReturnType<typeof normalizeStatus>;
|
type StatusRecord = ReturnType<typeof normalizeStatus>;
|
||||||
|
|
|
@ -27,6 +27,7 @@ type PollOption = ReturnType<typeof PollOptionRecord>;
|
||||||
type Status = ReturnType<typeof StatusRecord>;
|
type Status = ReturnType<typeof StatusRecord>;
|
||||||
|
|
||||||
// Utility types
|
// Utility types
|
||||||
|
type APIEntity = Record<string, any>;
|
||||||
type EmbeddedEntity<T extends object> = null | string | ReturnType<ImmutableRecord.Factory<T>>;
|
type EmbeddedEntity<T extends object> = null | string | ReturnType<ImmutableRecord.Factory<T>>;
|
||||||
|
|
||||||
export {
|
export {
|
||||||
|
@ -43,5 +44,6 @@ export {
|
||||||
Status,
|
Status,
|
||||||
|
|
||||||
// Utility types
|
// Utility types
|
||||||
|
APIEntity,
|
||||||
EmbeddedEntity,
|
EmbeddedEntity,
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue