import React from 'react' import { connect } from 'react-redux' import { FormattedMessage } from 'react-intl' import { Map as ImmutableMap, List as ImmutableList, fromJS, } from 'immutable'; import { replyCompose, mentionCompose, quoteCompose, } from '../actions/compose'; import { repost, favorite, unrepost, unfavorite, } from '../actions/interactions'; import { hideStatus, revealStatus, fetchComments, fetchContext, } from '../actions/statuses'; import { openModal } from '../actions/modal'; import { openPopover } from '../actions/popover'; import { me, boostModal, } from '../initial_state' import { MODAL_BOOST, MODAL_CONFIRM, MODAL_PRO_UPGRADE, POPOVER_COMMENT_SORTING_OPTIONS, POPOVER_STATUS_SHARE, COMMENT_SORTING_TYPE_OLDEST, COMMENT_SORTING_TYPE_NEWEST, COMMENT_SORTING_TYPE_TOP, } from '../constants' import { makeGetStatus } from '../selectors' import Status from '../components/status'; const sortReplies = (replyIds, state, type) => { if (!replyIds) return replyIds if (type === COMMENT_SORTING_TYPE_OLDEST || !type) { return replyIds // default } else if (type === COMMENT_SORTING_TYPE_NEWEST) { return replyIds.reverse() } else if (type === COMMENT_SORTING_TYPE_TOP) { let statusList = [] replyIds.forEach((replyId) => { const status = state.getIn(['statuses', replyId]) if (status) { statusList.push({ id: replyId, likeCount: status.get('favourites_count'), }) } }) statusList.sort((a, b) => parseFloat(b.likeCount) - parseFloat(a.likeCount)) let newReplyIds = ImmutableList() for (let i = 0; i < statusList.length; i++) { const block = statusList[i]; newReplyIds = newReplyIds.set(i, block.id) } return newReplyIds } return replyIds } const getDescendants = (state, status, highlightStatusId, commentSortingType) => { let descendantsIds = ImmutableList() let index = 0 const MAX_INDENT = 2 descendantsIds = descendantsIds.withMutations((mutable) => { const ids = [{ id: status.get('id'), indent: -1, }] while (ids.length > 0) { let block = ids.shift() let id = block.id let indent = block.indent let replies = state.getIn(['contexts', 'replies', id]) // Sort only Top-level replies (if original status not comment) if (index === 0) { replies = sortReplies(replies, state, commentSortingType) } if (status.get('id') !== id) { mutable.push(ImmutableMap({ statusId: id, indent: indent, isHighlighted: !!highlightStatusId && highlightStatusId === id, })) } if (replies) { indent++ indent = Math.min(MAX_INDENT, indent) replies.reverse().forEach((reply) => { ids.unshift({ id: reply, indent: indent, }) }) } index++ } }) return descendantsIds } const makeMapStateToProps = () => { const getStatus = makeGetStatus() const mapStateToProps = (state, props) => { const statusId = props.id || props.params.statusId const username = props.params ? props.params.username : undefined const commentSortingType = state.getIn(['settings', 'commentSorting']) const status = getStatus(state, { id: statusId, username: username, }) let fetchedContext = false let descendantsIds = ImmutableList() let ancestorStatus // if (status && status.get('in_reply_to_account_id') && !props.isChild) { fetchedContext = true let inReplyTos = state.getIn(['contexts', 'inReplyTos']) let ancestorsIds = ImmutableList() ancestorsIds = ancestorsIds.withMutations(mutable => { let id = statusId; while (id) { mutable.unshift(id) id = inReplyTos.get(id) } }) const ancestorStatusId = ancestorsIds.get(0) if (ancestorStatusId !== statusId) { ancestorStatus = getStatus(state, { id: ancestorStatusId, }) if (!!ancestorStatus) descendantsIds = getDescendants(state, ancestorStatus, statusId) } } // if (status && status.get('replies_count') > 0 && !fetchedContext) { descendantsIds = getDescendants(state, status, null, commentSortingType) } const isComment = !!status ? !!status.get('in_reply_to_id') : false return { status, ancestorStatus, descendantsIds, isComment, commentSortingType, isComposeModalOpen: state.getIn(['modal', 'modalType']) === 'COMPOSE', isCompact: state.getIn(['settings', 'isCompact']), } } return mapStateToProps }; const mapDispatchToProps = (dispatch) => ({ onReply (status, router, showModal) { if (!me) return dispatch(openModal('UNAUTHORIZED')) dispatch((_, getState) => { const state = getState(); if (state.getIn(['compose', 'text']).trim().length !== 0) { dispatch(openModal('CONFIRM', { message: , confirm: , onConfirm: () => dispatch(replyCompose(status, router)), })) } else { dispatch(replyCompose(status, router, showModal)); } }) }, onShowRevisions (status) { if (!me) return dispatch(openModal('UNAUTHORIZED')) dispatch(openModal('STATUS_REVISION', { status })); }, onFavorite (status) { if (!me) return dispatch(openModal('UNAUTHORIZED')) if (status.get('favourited')) { dispatch(unfavorite(status)); } else { dispatch(favorite(status)); } }, onMention (account, router) { if (!me) return dispatch(openModal('UNAUTHORIZED')) dispatch(mentionCompose(account, router)); }, onOpenMedia (media, index) { dispatch(openModal('MEDIA', { media, index })); }, onOpenVideo (media, time) { dispatch(openModal('VIDEO', { media, time })); }, onToggleHidden (status) { if (!me) return dispatch(openModal('UNAUTHORIZED')) if (status.get('hidden')) { dispatch(revealStatus(status.get('id'))); } else { dispatch(hideStatus(status.get('id'))); } }, onOpenLikes(status) { if (!me) return dispatch(openModal('UNAUTHORIZED')) dispatch(openModal('STATUS_LIKES', { status })) }, onOpenReposts(status) { if (!me) return dispatch(openModal('UNAUTHORIZED')) dispatch(openModal('STATUS_REPOSTS', { status })) }, onFetchComments(statusId) { dispatch(fetchComments(statusId)) }, onFetchContext(statusId) { dispatch(fetchContext(statusId)) }, onQuote (status, router) { if (!me) return dispatch(openModal('UNAUTHORIZED')) dispatch((_, getState) => { const state = getState() if (state.getIn(['compose', 'text']).trim().length !== 0) { dispatch(openModal(MODAL_CONFIRM, { message: , confirm: , onConfirm: () => dispatch(quoteCompose(status, router)), })) } else { dispatch(quoteCompose(status, router)) } }) }, onRepost (status) { if (!me) return dispatch(openModal('UNAUTHORIZED')) const alreadyReposted = status.get('reblogged') if (boostModal && !alreadyReposted) { dispatch(openModal(MODAL_BOOST, { status, onRepost: () => dispatch(repost(status)), })) } else { if (alreadyReposted) { dispatch(unrepost(status)) } else { dispatch(repost(status)) } } }, onShare (targetRef, status) { if (!!navigator && navigator.share) { if (!!navigator.canShare && !navigator.canShare()) { // } else { const url = status.get('url') return navigator.share({ url, title: 'Check out this post on Gab', text: 'Check out this post on Gab', }).then(() => { // }).catch((error) => { console.log('Error sharing', error) }) } } dispatch(openPopover(POPOVER_STATUS_SHARE, { targetRef, status, position: 'top', })) }, onCommentSortOpen(targetRef, callback) { dispatch(openPopover(POPOVER_COMMENT_SORTING_OPTIONS, { targetRef, callback, position: 'top', })) }, onOpenProModal() { dispatch(openModal(MODAL_PRO_UPGRADE)) }, }); export default connect(makeMapStateToProps, mapDispatchToProps)(Status);