import { OrderedSet as ImmutableOrderedSet } from 'immutable'; import debounce from 'lodash/debounce'; import React, { useCallback, useEffect, useState } from 'react'; import { defineMessages, FormattedMessage, useIntl } from 'react-intl'; import { fetchAccount, fetchFollowers, expandFollowers, fetchAccountByUsername, } from 'soapbox/actions/accounts'; import MissingIndicator from 'soapbox/components/missing_indicator'; import ScrollableList from 'soapbox/components/scrollable_list'; import { Spinner } from 'soapbox/components/ui'; import AccountContainer from 'soapbox/containers/account_container'; import { useAppDispatch, useAppSelector, useFeatures, useOwnAccount } from 'soapbox/hooks'; import { findAccountByUsername } from 'soapbox/selectors'; import Column from '../ui/components/column'; const messages = defineMessages({ heading: { id: 'column.followers', defaultMessage: 'Followers' }, }); interface IFollowers { params?: { username?: string, } } /** Displays a list of accounts who follow the given account. */ const Followers: React.FC = (props) => { const intl = useIntl(); const dispatch = useAppDispatch(); const features = useFeatures(); const ownAccount = useOwnAccount(); const [loading, setLoading] = useState(true); const username = props.params?.username || ''; const account = useAppSelector(state => findAccountByUsername(state, username)); const isOwnAccount = username.toLowerCase() === ownAccount?.username?.toLowerCase(); const accountIds = useAppSelector(state => state.user_lists.followers.get(account!?.id)?.items || ImmutableOrderedSet()); const hasMore = useAppSelector(state => !!state.user_lists.followers.get(account!?.id)?.next); const unavailable = useAppSelector(state => { const blockedBy = state.relationships.getIn([account?.id, 'blocked_by']) === true; return isOwnAccount ? false : (blockedBy && !features.blockersVisible); }); useEffect(() => { let promises = []; if (account) { promises = [ dispatch(fetchAccount(account.id)), dispatch(fetchFollowers(account.id)), ]; } else { promises = [ dispatch(fetchAccountByUsername(username)), ]; } Promise.all(promises) .then(() => setLoading(false)) .catch(() => setLoading(false)); }, [account?.id, username]); const handleLoadMore = useCallback(debounce(() => { if (account) { dispatch(expandFollowers(account.id)); } }, 300, { leading: true }), [account?.id]); if (loading && accountIds.isEmpty()) { return ( ); } if (!account) { return ( ); } if (unavailable) { return (
); } return ( } itemClassName='pb-4' > {accountIds.map(id => , )} ); }; export default Followers;