import PropTypes from "prop-types"; import { FormattedMessage } from "react-intl"; import ImmutablePropTypes from "react-immutable-proptypes"; import ImmutablePureComponent from "react-immutable-pure-component"; import { connect } from "react-redux"; import { lookupAccount, fetchAccount } from "mastodon/actions/accounts"; import { openModal } from "mastodon/actions/modal"; import ColumnBackButton from "mastodon/components/column_back_button"; import { LoadMore } from "mastodon/components/load_more"; import { LoadingIndicator } from "mastodon/components/loading_indicator"; import ScrollContainer from "mastodon/containers/scroll_container"; import BundleColumnError from "mastodon/features/ui/components/bundle_column_error"; import { normalizeForLookup } from "mastodon/reducers/accounts_map"; import { getAccountGallery } from "mastodon/selectors"; import { expandAccountMediaTimeline } from "../../actions/timelines"; import HeaderContainer from "../account_timeline/containers/header_container"; import Column from "../ui/components/column"; import MediaItem from "./components/media_item"; const mapStateToProps = (state, { params: { acct, id } }) => { const accountId = id || state.getIn(["accounts_map", normalizeForLookup(acct)]); if (!accountId) { return { isLoading: true, }; } return { accountId, isAccount: !!state.getIn(["accounts", accountId]), attachments: getAccountGallery(state, accountId), isLoading: state.getIn(["timelines", `account:${accountId}:media`, "isLoading"]), hasMore: state.getIn(["timelines", `account:${accountId}:media`, "hasMore"]), suspended: state.getIn(["accounts", accountId, "suspended"], false), blockedBy: state.getIn(["relationships", accountId, "blocked_by"], false), }; }; class LoadMoreMedia extends ImmutablePureComponent { static propTypes = { maxId: PropTypes.string, onLoadMore: PropTypes.func.isRequired, }; handleLoadMore = () => { this.props.onLoadMore(this.props.maxId); }; render () { return ( ); } } class AccountGallery extends ImmutablePureComponent { static propTypes = { params: PropTypes.shape({ acct: PropTypes.string, id: PropTypes.string, }).isRequired, accountId: PropTypes.string, dispatch: PropTypes.func.isRequired, attachments: ImmutablePropTypes.list.isRequired, isLoading: PropTypes.bool, hasMore: PropTypes.bool, isAccount: PropTypes.bool, blockedBy: PropTypes.bool, suspended: PropTypes.bool, multiColumn: PropTypes.bool, }; state = { width: 323, }; _load () { const { accountId, isAccount, dispatch } = this.props; if (!isAccount) { dispatch(fetchAccount(accountId)); } dispatch(expandAccountMediaTimeline(accountId)); } componentDidMount () { const { params: { acct }, accountId, dispatch } = this.props; if (accountId) { this._load(); } else { dispatch(lookupAccount(acct)); } } componentDidUpdate (prevProps) { const { params: { acct }, accountId, dispatch } = this.props; if (prevProps.accountId !== accountId && accountId) { this._load(); } else if (prevProps.params.acct !== acct) { dispatch(lookupAccount(acct)); } } handleScrollToBottom = () => { if (this.props.hasMore) { this.handleLoadMore(this.props.attachments.size > 0 ? this.props.attachments.last().getIn(["status", "id"]) : undefined); } }; handleScroll = e => { const { scrollTop, scrollHeight, clientHeight } = e.target; const offset = scrollHeight - scrollTop - clientHeight; if (150 > offset && !this.props.isLoading) { this.handleScrollToBottom(); } }; handleLoadMore = maxId => { this.props.dispatch(expandAccountMediaTimeline(this.props.accountId, { maxId })); }; handleLoadOlder = e => { e.preventDefault(); this.handleScrollToBottom(); }; handleOpenMedia = attachment => { const { dispatch } = this.props; const statusId = attachment.getIn(["status", "id"]); const lang = attachment.getIn(["status", "language"]); if (attachment.get("type") === "video") { dispatch(openModal({ modalType: "VIDEO", modalProps: { media: attachment, statusId, lang, options: { autoPlay: true } }, })); } else if (attachment.get("type") === "audio") { dispatch(openModal({ modalType: "AUDIO", modalProps: { media: attachment, statusId, lang, options: { autoPlay: true } }, })); } else { const media = attachment.getIn(["status", "media_attachments"]); const index = media.findIndex(x => x.get("id") === attachment.get("id")); dispatch(openModal({ modalType: "MEDIA", modalProps: { media, index, statusId, lang }, })); } }; handleRef = c => { if (c) { this.setState({ width: c.offsetWidth }); } }; render () { const { attachments, isLoading, hasMore, isAccount, multiColumn, blockedBy, suspended } = this.props; const { width } = this.state; if (!isAccount) { return ( ); } if (!attachments && isLoading) { return ( ); } let loadOlder = null; if (hasMore && !(isLoading && attachments.size === 0)) { loadOlder = ; } let emptyMessage; if (suspended) { emptyMessage = ; } else if (blockedBy) { emptyMessage = ; } return (
{(suspended || blockedBy) ? (
{emptyMessage}
) : (
{attachments.map((attachment, index) => attachment === null ? ( 0 ? attachments.getIn(index - 1, "id") : null} onLoadMore={this.handleLoadMore} /> ) : ( ))} {loadOlder}
)} {isLoading && attachments.size === 0 && (
)}
); } } export default connect(mapStateToProps)(AccountGallery);