[Glitch] Add featured tags selector for WebUI
Port 4c7b5fb6c1 to glitch-soc
Signed-off-by: Claire <claire.github-309c@sitedethib.com>
			
			
This commit is contained in:
		
							parent
							
								
									a2942fd0b8
								
							
						
					
					
						commit
						8be350cc82
					
				| 
						 | 
					@ -0,0 +1,34 @@
 | 
				
			||||||
 | 
					import api from '../api';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export const FEATURED_TAGS_FETCH_REQUEST = 'FEATURED_TAGS_FETCH_REQUEST';
 | 
				
			||||||
 | 
					export const FEATURED_TAGS_FETCH_SUCCESS = 'FEATURED_TAGS_FETCH_SUCCESS';
 | 
				
			||||||
 | 
					export const FEATURED_TAGS_FETCH_FAIL    = 'FEATURED_TAGS_FETCH_FAIL';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export const fetchFeaturedTags = (id) => (dispatch, getState) => {
 | 
				
			||||||
 | 
					  if (getState().getIn(['user_lists', 'featured_tags', id, 'items'])) {
 | 
				
			||||||
 | 
					    return;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  dispatch(fetchFeaturedTagsRequest(id));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  api(getState).get(`/api/v1/accounts/${id}/featured_tags`)
 | 
				
			||||||
 | 
					    .then(({ data }) => dispatch(fetchFeaturedTagsSuccess(id, data)))
 | 
				
			||||||
 | 
					    .catch(err => dispatch(fetchFeaturedTagsFail(id, err)));
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export const fetchFeaturedTagsRequest = (id) => ({
 | 
				
			||||||
 | 
					  type: FEATURED_TAGS_FETCH_REQUEST,
 | 
				
			||||||
 | 
					  id,
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export const fetchFeaturedTagsSuccess = (id, tags) => ({
 | 
				
			||||||
 | 
					  type: FEATURED_TAGS_FETCH_SUCCESS,
 | 
				
			||||||
 | 
					  id,
 | 
				
			||||||
 | 
					  tags,
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export const fetchFeaturedTagsFail = (id, error) => ({
 | 
				
			||||||
 | 
					  type: FEATURED_TAGS_FETCH_FAIL,
 | 
				
			||||||
 | 
					  id,
 | 
				
			||||||
 | 
					  error,
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
| 
						 | 
					@ -156,8 +156,8 @@ export const expandHomeTimeline            = ({ maxId } = {}, done = noOp) => ex
 | 
				
			||||||
export const expandPublicTimeline          = ({ maxId, onlyMedia, onlyRemote, allowLocalOnly } = {}, done = noOp) => expandTimeline(`public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { remote: !!onlyRemote, allow_local_only: !!allowLocalOnly, max_id: maxId, only_media: !!onlyMedia }, done);
 | 
					export const expandPublicTimeline          = ({ maxId, onlyMedia, onlyRemote, allowLocalOnly } = {}, done = noOp) => expandTimeline(`public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { remote: !!onlyRemote, allow_local_only: !!allowLocalOnly, max_id: maxId, only_media: !!onlyMedia }, done);
 | 
				
			||||||
export const expandCommunityTimeline       = ({ maxId, onlyMedia } = {}, done = noOp) => expandTimeline(`community${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { local: true, max_id: maxId, only_media: !!onlyMedia }, done);
 | 
					export const expandCommunityTimeline       = ({ maxId, onlyMedia } = {}, done = noOp) => expandTimeline(`community${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { local: true, max_id: maxId, only_media: !!onlyMedia }, done);
 | 
				
			||||||
export const expandDirectTimeline          = ({ maxId } = {}, done = noOp) => expandTimeline('direct', '/api/v1/timelines/direct', { max_id: maxId }, done);
 | 
					export const expandDirectTimeline          = ({ maxId } = {}, done = noOp) => expandTimeline('direct', '/api/v1/timelines/direct', { max_id: maxId }, done);
 | 
				
			||||||
export const expandAccountTimeline         = (accountId, { maxId, withReplies } = {}) => expandTimeline(`account:${accountId}${withReplies ? ':with_replies' : ''}`, `/api/v1/accounts/${accountId}/statuses`, { exclude_replies: !withReplies, max_id: maxId });
 | 
					export const expandAccountTimeline         = (accountId, { maxId, withReplies, tagged } = {}) => expandTimeline(`account:${accountId}${withReplies ? ':with_replies' : ''}${tagged ? `:${tagged}` : ''}`, `/api/v1/accounts/${accountId}/statuses`, { exclude_replies: !withReplies, tagged, max_id: maxId });
 | 
				
			||||||
export const expandAccountFeaturedTimeline = accountId => expandTimeline(`account:${accountId}:pinned`, `/api/v1/accounts/${accountId}/statuses`, { pinned: true });
 | 
					export const expandAccountFeaturedTimeline = (accountId, { tagged } = {}) => expandTimeline(`account:${accountId}:pinned`, `/api/v1/accounts/${accountId}/statuses`, { pinned: true, tagged });
 | 
				
			||||||
export const expandAccountMediaTimeline    = (accountId, { maxId } = {}) => expandTimeline(`account:${accountId}:media`, `/api/v1/accounts/${accountId}/statuses`, { max_id: maxId, only_media: true, limit: 40 });
 | 
					export const expandAccountMediaTimeline    = (accountId, { maxId } = {}) => expandTimeline(`account:${accountId}:media`, `/api/v1/accounts/${accountId}/statuses`, { max_id: maxId, only_media: true, limit: 40 });
 | 
				
			||||||
export const expandListTimeline            = (id, { maxId } = {}, done = noOp) => expandTimeline(`list:${id}`, `/api/v1/timelines/list/${id}`, { max_id: maxId }, done);
 | 
					export const expandListTimeline            = (id, { maxId } = {}, done = noOp) => expandTimeline(`list:${id}`, `/api/v1/timelines/list/${id}`, { max_id: maxId }, done);
 | 
				
			||||||
export const expandHashtagTimeline         = (hashtag, { maxId, tags, local } = {}, done = noOp) => {
 | 
					export const expandHashtagTimeline         = (hashtag, { maxId, tags, local } = {}, done = noOp) => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,71 @@
 | 
				
			||||||
 | 
					import React from 'react';
 | 
				
			||||||
 | 
					import { connect } from 'react-redux';
 | 
				
			||||||
 | 
					import PropTypes from 'prop-types';
 | 
				
			||||||
 | 
					import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
				
			||||||
 | 
					import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
				
			||||||
 | 
					import { defineMessages, injectIntl } from 'react-intl';
 | 
				
			||||||
 | 
					import classNames from 'classnames';
 | 
				
			||||||
 | 
					import Permalink from 'flavours/glitch/components/permalink';
 | 
				
			||||||
 | 
					import ShortNumber from 'flavours/glitch/components/short_number';
 | 
				
			||||||
 | 
					import { List as ImmutableList } from 'immutable';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const messages = defineMessages({
 | 
				
			||||||
 | 
					  hashtag_all: { id: 'account.hashtag_all', defaultMessage: 'All' },
 | 
				
			||||||
 | 
					  hashtag_all_description: { id: 'account.hashtag_all_description', defaultMessage: 'All posts (deselect hashtags)' },
 | 
				
			||||||
 | 
					  hashtag_select_description: { id: 'account.hashtag_select_description', defaultMessage: 'Select hashtag #{name}' },
 | 
				
			||||||
 | 
					  statuses_counter: { id: 'account.statuses_counter', defaultMessage: '{count, plural, one {{counter} Post} other {{counter} Posts}}' },
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const mapStateToProps = (state, { account }) => ({
 | 
				
			||||||
 | 
					  featuredTags: state.getIn(['user_lists', 'featured_tags', account.get('id'), 'items'], ImmutableList()),
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default @connect(mapStateToProps)
 | 
				
			||||||
 | 
					@injectIntl
 | 
				
			||||||
 | 
					class FeaturedTags extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  static contextTypes = {
 | 
				
			||||||
 | 
					    router: PropTypes.object,
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  static propTypes = {
 | 
				
			||||||
 | 
					    account: ImmutablePropTypes.map,
 | 
				
			||||||
 | 
					    featuredTags: ImmutablePropTypes.list,
 | 
				
			||||||
 | 
					    tagged: PropTypes.string,
 | 
				
			||||||
 | 
					    intl: PropTypes.object.isRequired,
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  render () {
 | 
				
			||||||
 | 
					    const { account, featuredTags, tagged, intl } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (!account || featuredTags.isEmpty()) {
 | 
				
			||||||
 | 
					      return null;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const suspended = account.get('suspended');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return (
 | 
				
			||||||
 | 
					      <div className={classNames('account__header', 'advanced', { inactive: !!account.get('moved') })}>
 | 
				
			||||||
 | 
					        <div className='account__header__extra'>
 | 
				
			||||||
 | 
					          <div className='account__header__extra__hashtag-links'>
 | 
				
			||||||
 | 
					            <Permalink key='all' className={classNames('account__hashtag-link', { active: !tagged })} title={intl.formatMessage(messages.hashtag_all_description)} href={account.get('url')} to={`/@${account.get('acct')}`}>{intl.formatMessage(messages.hashtag_all)}</Permalink>
 | 
				
			||||||
 | 
					            {!suspended && featuredTags.map(featuredTag => {
 | 
				
			||||||
 | 
					              const name  = featuredTag.get('name');
 | 
				
			||||||
 | 
					              const url   = featuredTag.get('url');
 | 
				
			||||||
 | 
					              const to    = `/@${account.get('acct')}/tagged/${name}`;
 | 
				
			||||||
 | 
					              const desc  = intl.formatMessage(messages.hashtag_select_description, { name });
 | 
				
			||||||
 | 
					              const count = featuredTag.get('statuses_count');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					              return (
 | 
				
			||||||
 | 
					                <Permalink key={`#${name}`} className={classNames('account__hashtag-link', { active: this.context.router.history.location.pathname === to })} title={desc} href={url} to={to}>
 | 
				
			||||||
 | 
					                  #{name} <span title={intl.formatMessage(messages.statuses_counter, { count: count, counter: intl.formatNumber(count) })}>({<ShortNumber value={count} />})</span>
 | 
				
			||||||
 | 
					                </Permalink>
 | 
				
			||||||
 | 
					              );
 | 
				
			||||||
 | 
					            })}
 | 
				
			||||||
 | 
					          </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					      </div>
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -28,6 +28,7 @@ export default class Header extends ImmutablePureComponent {
 | 
				
			||||||
    hideTabs: PropTypes.bool,
 | 
					    hideTabs: PropTypes.bool,
 | 
				
			||||||
    domain: PropTypes.string.isRequired,
 | 
					    domain: PropTypes.string.isRequired,
 | 
				
			||||||
    hidden: PropTypes.bool,
 | 
					    hidden: PropTypes.bool,
 | 
				
			||||||
 | 
					    tagged: PropTypes.string,
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static contextTypes = {
 | 
					  static contextTypes = {
 | 
				
			||||||
| 
						 | 
					@ -103,7 +104,7 @@ export default class Header extends ImmutablePureComponent {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { account, hidden, hideTabs } = this.props;
 | 
					    const { account, hidden, hideTabs, tagged } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (account === null) {
 | 
					    if (account === null) {
 | 
				
			||||||
      return null;
 | 
					      return null;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -18,10 +18,11 @@ import TimelineHint from 'flavours/glitch/components/timeline_hint';
 | 
				
			||||||
import LimitedAccountHint from './components/limited_account_hint';
 | 
					import LimitedAccountHint from './components/limited_account_hint';
 | 
				
			||||||
import { getAccountHidden } from 'flavours/glitch/selectors';
 | 
					import { getAccountHidden } from 'flavours/glitch/selectors';
 | 
				
			||||||
import { normalizeForLookup } from 'flavours/glitch/reducers/accounts_map';
 | 
					import { normalizeForLookup } from 'flavours/glitch/reducers/accounts_map';
 | 
				
			||||||
 | 
					import { fetchFeaturedTags } from '../../actions/featured_tags';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const emptyList = ImmutableList();
 | 
					const emptyList = ImmutableList();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const mapStateToProps = (state, { params: { acct, id }, withReplies = false }) => {
 | 
					const mapStateToProps = (state, { params: { acct, id, tagged }, withReplies = false }) => {
 | 
				
			||||||
  const accountId = id || state.getIn(['accounts_map', normalizeForLookup(acct)]);
 | 
					  const accountId = id || state.getIn(['accounts_map', normalizeForLookup(acct)]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (!accountId) {
 | 
					  if (!accountId) {
 | 
				
			||||||
| 
						 | 
					@ -31,7 +32,7 @@ const mapStateToProps = (state, { params: { acct, id }, withReplies = false }) =
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const path = withReplies ? `${accountId}:with_replies` : accountId;
 | 
					  const path = withReplies ? `${accountId}:with_replies` : `${accountId}${tagged ? `:${tagged}` : ''}`;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return {
 | 
					  return {
 | 
				
			||||||
    accountId,
 | 
					    accountId,
 | 
				
			||||||
| 
						 | 
					@ -39,7 +40,7 @@ const mapStateToProps = (state, { params: { acct, id }, withReplies = false }) =
 | 
				
			||||||
    remoteUrl: state.getIn(['accounts', accountId, 'url']),
 | 
					    remoteUrl: state.getIn(['accounts', accountId, 'url']),
 | 
				
			||||||
    isAccount: !!state.getIn(['accounts', accountId]),
 | 
					    isAccount: !!state.getIn(['accounts', accountId]),
 | 
				
			||||||
    statusIds: state.getIn(['timelines', `account:${path}`, 'items'], ImmutableList()),
 | 
					    statusIds: state.getIn(['timelines', `account:${path}`, 'items'], ImmutableList()),
 | 
				
			||||||
    featuredStatusIds: withReplies ? ImmutableList() : state.getIn(['timelines', `account:${accountId}:pinned`, 'items'], ImmutableList()),
 | 
					    featuredStatusIds: withReplies ? ImmutableList() : state.getIn(['timelines', `account:${accountId}:pinned${tagged ? `:${tagged}` : ''}`, 'items'], ImmutableList()),
 | 
				
			||||||
    isLoading: state.getIn(['timelines', `account:${path}`, 'isLoading']),
 | 
					    isLoading: state.getIn(['timelines', `account:${path}`, 'isLoading']),
 | 
				
			||||||
    hasMore:   state.getIn(['timelines', `account:${path}`, 'hasMore']),
 | 
					    hasMore:   state.getIn(['timelines', `account:${path}`, 'hasMore']),
 | 
				
			||||||
    suspended: state.getIn(['accounts', accountId, 'suspended'], false),
 | 
					    suspended: state.getIn(['accounts', accountId, 'suspended'], false),
 | 
				
			||||||
| 
						 | 
					@ -62,6 +63,7 @@ class AccountTimeline extends ImmutablePureComponent {
 | 
				
			||||||
    params: PropTypes.shape({
 | 
					    params: PropTypes.shape({
 | 
				
			||||||
      acct: PropTypes.string,
 | 
					      acct: PropTypes.string,
 | 
				
			||||||
      id: PropTypes.string,
 | 
					      id: PropTypes.string,
 | 
				
			||||||
 | 
					      tagged: PropTypes.string,
 | 
				
			||||||
    }).isRequired,
 | 
					    }).isRequired,
 | 
				
			||||||
    accountId: PropTypes.string,
 | 
					    accountId: PropTypes.string,
 | 
				
			||||||
    dispatch: PropTypes.func.isRequired,
 | 
					    dispatch: PropTypes.func.isRequired,
 | 
				
			||||||
| 
						 | 
					@ -79,14 +81,16 @@ class AccountTimeline extends ImmutablePureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  _load () {
 | 
					  _load () {
 | 
				
			||||||
    const { accountId, withReplies, dispatch } = this.props;
 | 
					    const { accountId, withReplies, params: { tagged }, dispatch } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    dispatch(fetchAccount(accountId));
 | 
					    dispatch(fetchAccount(accountId));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!withReplies) {
 | 
					    if (!withReplies) {
 | 
				
			||||||
      dispatch(expandAccountFeaturedTimeline(accountId));
 | 
					      dispatch(expandAccountFeaturedTimeline(accountId, { tagged }));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    dispatch(expandAccountTimeline(accountId, { withReplies }));
 | 
					
 | 
				
			||||||
 | 
					    dispatch(fetchFeaturedTags(accountId));
 | 
				
			||||||
 | 
					    dispatch(expandAccountTimeline(accountId, { withReplies, tagged }));
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  componentDidMount () {
 | 
					  componentDidMount () {
 | 
				
			||||||
| 
						 | 
					@ -100,12 +104,17 @@ class AccountTimeline extends ImmutablePureComponent {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  componentDidUpdate (prevProps) {
 | 
					  componentDidUpdate (prevProps) {
 | 
				
			||||||
    const { params: { acct }, accountId, dispatch } = this.props;
 | 
					    const { params: { acct, tagged }, accountId, withReplies, dispatch } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (prevProps.accountId !== accountId && accountId) {
 | 
					    if (prevProps.accountId !== accountId && accountId) {
 | 
				
			||||||
      this._load();
 | 
					      this._load();
 | 
				
			||||||
    } else if (prevProps.params.acct !== acct) {
 | 
					    } else if (prevProps.params.acct !== acct) {
 | 
				
			||||||
      dispatch(lookupAccount(acct));
 | 
					      dispatch(lookupAccount(acct));
 | 
				
			||||||
 | 
					    } else if (prevProps.params.tagged !== tagged) {
 | 
				
			||||||
 | 
					      if (!withReplies) {
 | 
				
			||||||
 | 
					        dispatch(expandAccountFeaturedTimeline(accountId, { tagged }));
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      dispatch(expandAccountTimeline(accountId, { withReplies, tagged }));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -128,7 +137,7 @@ class AccountTimeline extends ImmutablePureComponent {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  handleLoadMore = maxId => {
 | 
					  handleLoadMore = maxId => {
 | 
				
			||||||
    this.props.dispatch(expandAccountTimeline(this.props.accountId, { maxId, withReplies: this.props.withReplies }));
 | 
					    this.props.dispatch(expandAccountTimeline(this.props.accountId, { maxId, withReplies: this.props.withReplies, tagged: this.props.params.tagged }));
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  setRef = c => {
 | 
					  setRef = c => {
 | 
				
			||||||
| 
						 | 
					@ -174,7 +183,7 @@ class AccountTimeline extends ImmutablePureComponent {
 | 
				
			||||||
        <ProfileColumnHeader onClick={this.handleHeaderClick} multiColumn={multiColumn} />
 | 
					        <ProfileColumnHeader onClick={this.handleHeaderClick} multiColumn={multiColumn} />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        <StatusList
 | 
					        <StatusList
 | 
				
			||||||
          prepend={<HeaderContainer accountId={this.props.accountId} hideTabs={forceEmptyState} />}
 | 
					          prepend={<HeaderContainer accountId={this.props.accountId} hideTabs={forceEmptyState} tagged={this.props.params.tagged} />}
 | 
				
			||||||
          alwaysPrepend
 | 
					          alwaysPrepend
 | 
				
			||||||
          append={remoteMessage}
 | 
					          append={remoteMessage}
 | 
				
			||||||
          scrollKey='account_timeline'
 | 
					          scrollKey='account_timeline'
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -211,6 +211,7 @@ class SwitchingColumnsArea extends React.PureComponent {
 | 
				
			||||||
          <WrappedRoute path={['/publish', '/statuses/new']} component={Compose} content={children} />
 | 
					          <WrappedRoute path={['/publish', '/statuses/new']} component={Compose} content={children} />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
          <WrappedRoute path={['/@:acct', '/accounts/:id']} exact component={AccountTimeline} content={children} />
 | 
					          <WrappedRoute path={['/@:acct', '/accounts/:id']} exact component={AccountTimeline} content={children} />
 | 
				
			||||||
 | 
					          <WrappedRoute path='/@:acct/tagged/:tagged?' exact component={AccountTimeline} content={children} />
 | 
				
			||||||
          <WrappedRoute path={['/@:acct/with_replies', '/accounts/:id/with_replies']} component={AccountTimeline} content={children} componentParams={{ withReplies: true }} />
 | 
					          <WrappedRoute path={['/@:acct/with_replies', '/accounts/:id/with_replies']} component={AccountTimeline} content={children} componentParams={{ withReplies: true }} />
 | 
				
			||||||
          <WrappedRoute path={['/accounts/:id/followers', '/users/:acct/followers', '/@:acct/followers']} component={Followers} content={children} />
 | 
					          <WrappedRoute path={['/accounts/:id/followers', '/users/:acct/followers', '/@:acct/followers']} component={Followers} content={children} />
 | 
				
			||||||
          <WrappedRoute path={['/accounts/:id/following', '/users/:acct/following', '/@:acct/following']} component={Following} content={children} />
 | 
					          <WrappedRoute path={['/accounts/:id/following', '/users/:acct/following', '/@:acct/following']} component={Following} content={children} />
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -51,7 +51,12 @@ import {
 | 
				
			||||||
  DIRECTORY_EXPAND_SUCCESS,
 | 
					  DIRECTORY_EXPAND_SUCCESS,
 | 
				
			||||||
  DIRECTORY_EXPAND_FAIL,
 | 
					  DIRECTORY_EXPAND_FAIL,
 | 
				
			||||||
} from 'flavours/glitch/actions/directory';
 | 
					} from 'flavours/glitch/actions/directory';
 | 
				
			||||||
import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
 | 
					import {
 | 
				
			||||||
 | 
					  FEATURED_TAGS_FETCH_REQUEST,
 | 
				
			||||||
 | 
					  FEATURED_TAGS_FETCH_SUCCESS,
 | 
				
			||||||
 | 
					  FEATURED_TAGS_FETCH_FAIL,
 | 
				
			||||||
 | 
					} from 'flavours/glitch/actions/featured_tags';
 | 
				
			||||||
 | 
					import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const initialListState = ImmutableMap({
 | 
					const initialListState = ImmutableMap({
 | 
				
			||||||
  next: null,
 | 
					  next: null,
 | 
				
			||||||
| 
						 | 
					@ -67,6 +72,7 @@ const initialState = ImmutableMap({
 | 
				
			||||||
  follow_requests: initialListState,
 | 
					  follow_requests: initialListState,
 | 
				
			||||||
  blocks: initialListState,
 | 
					  blocks: initialListState,
 | 
				
			||||||
  mutes: initialListState,
 | 
					  mutes: initialListState,
 | 
				
			||||||
 | 
					  featured_tags: initialListState,
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const normalizeList = (state, path, accounts, next) => {
 | 
					const normalizeList = (state, path, accounts, next) => {
 | 
				
			||||||
| 
						 | 
					@ -89,6 +95,18 @@ const normalizeFollowRequest = (state, notification) => {
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const normalizeFeaturedTag = (featuredTags, accountId) => {
 | 
				
			||||||
 | 
					  const normalizeFeaturedTag = { ...featuredTags, accountId: accountId };
 | 
				
			||||||
 | 
					  return fromJS(normalizeFeaturedTag);
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const normalizeFeaturedTags = (state, path, featuredTags, accountId) => {
 | 
				
			||||||
 | 
					  return state.setIn(path, ImmutableMap({
 | 
				
			||||||
 | 
					    items: ImmutableList(featuredTags.map(featuredTag => normalizeFeaturedTag(featuredTag, accountId)).sort((a, b) => b.get('statuses_count') - a.get('statuses_count'))),
 | 
				
			||||||
 | 
					    isLoading: false,
 | 
				
			||||||
 | 
					  }));
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default function userLists(state = initialState, action) {
 | 
					export default function userLists(state = initialState, action) {
 | 
				
			||||||
  switch(action.type) {
 | 
					  switch(action.type) {
 | 
				
			||||||
  case FOLLOWERS_FETCH_SUCCESS:
 | 
					  case FOLLOWERS_FETCH_SUCCESS:
 | 
				
			||||||
| 
						 | 
					@ -160,6 +178,12 @@ export default function userLists(state = initialState, action) {
 | 
				
			||||||
  case DIRECTORY_FETCH_FAIL:
 | 
					  case DIRECTORY_FETCH_FAIL:
 | 
				
			||||||
  case DIRECTORY_EXPAND_FAIL:
 | 
					  case DIRECTORY_EXPAND_FAIL:
 | 
				
			||||||
    return state.setIn(['directory', 'isLoading'], false);
 | 
					    return state.setIn(['directory', 'isLoading'], false);
 | 
				
			||||||
 | 
					  case FEATURED_TAGS_FETCH_SUCCESS:
 | 
				
			||||||
 | 
					    return normalizeFeaturedTags(state, ['featured_tags', action.id], action.tags, action.id);
 | 
				
			||||||
 | 
					  case FEATURED_TAGS_FETCH_REQUEST:
 | 
				
			||||||
 | 
					    return state.setIn(['featured_tags', action.id, 'isLoading'], true);
 | 
				
			||||||
 | 
					  case FEATURED_TAGS_FETCH_FAIL:
 | 
				
			||||||
 | 
					    return state.setIn(['featured_tags', action.id, 'isLoading'], false);
 | 
				
			||||||
  default:
 | 
					  default:
 | 
				
			||||||
    return state;
 | 
					    return state;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue