Change output of api/accounts/:id/follow and unfollow to return relationship
Track relationship in redux state. Display follow/unfollow and following-back information on account view (unstyled)
This commit is contained in:
		| @@ -124,10 +124,10 @@ export function followAccountRequest(id) { | |||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export function followAccountSuccess(account) { | export function followAccountSuccess(relationship) { | ||||||
|   return { |   return { | ||||||
|     type: ACCOUNT_FOLLOW_SUCCESS, |     type: ACCOUNT_FOLLOW_SUCCESS, | ||||||
|     account: account |     relationship: relationship | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -145,10 +145,10 @@ export function unfollowAccountRequest(id) { | |||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export function unfollowAccountSuccess(account) { | export function unfollowAccountSuccess(relationship) { | ||||||
|   return { |   return { | ||||||
|     type: ACCOUNT_UNFOLLOW_SUCCESS, |     type: ACCOUNT_UNFOLLOW_SUCCESS, | ||||||
|     account: account |     relationship: relationship | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,15 +14,24 @@ const StatusContent = React.createClass({ | |||||||
|   mixins: [PureRenderMixin], |   mixins: [PureRenderMixin], | ||||||
|  |  | ||||||
|   componentDidMount () { |   componentDidMount () { | ||||||
|     const node = ReactDOM.findDOMNode(this); |     const node  = ReactDOM.findDOMNode(this); | ||||||
|  |     const links = node.querySelectorAll('a'); | ||||||
|  |  | ||||||
|     this.props.status.get('mentions').forEach(mention => { |     for (var i = 0; i < links.length; ++i) { | ||||||
|       const links = node.querySelector(`a[href="${mention.get('url')}"]`); |       let link    = links[i]; | ||||||
|       links.addEventListener('click', this.onLinkClick.bind(this, mention)); |       let mention = this.props.status.get('mentions').find(item => link.href === item.get('url')); | ||||||
|     }); |  | ||||||
|  |       if (mention) { | ||||||
|  |         link.addEventListener('click', this.onMentionClick.bind(this, mention)); | ||||||
|  |       } else { | ||||||
|  |         link.setAttribute('target', '_blank'); | ||||||
|  |         link.setAttribute('rel', 'noopener'); | ||||||
|  |         link.addEventListener('click', this.onNormalClick); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
|   onLinkClick (mention, e) { |   onMentionClick (mention, e) { | ||||||
|     if (e.button === 0) { |     if (e.button === 0) { | ||||||
|       e.preventDefault(); |       e.preventDefault(); | ||||||
|       this.context.router.push(`/accounts/${mention.get('id')}`); |       this.context.router.push(`/accounts/${mention.get('id')}`); | ||||||
| @@ -31,6 +40,10 @@ const StatusContent = React.createClass({ | |||||||
|     e.stopPropagation(); |     e.stopPropagation(); | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
|  |   onNormalClick (e) { | ||||||
|  |     e.stopPropagation(); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|   render () { |   render () { | ||||||
|     const content = { __html: this.props.status.get('content') }; |     const content = { __html: this.props.status.get('content') }; | ||||||
|     return <div className='status__content' dangerouslySetInnerHTML={content} />; |     return <div className='status__content' dangerouslySetInnerHTML={content} />; | ||||||
|   | |||||||
| @@ -0,0 +1,48 @@ | |||||||
|  | import PureRenderMixin    from 'react-addons-pure-render-mixin'; | ||||||
|  | import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||||
|  | import Button             from '../../../components/button'; | ||||||
|  |  | ||||||
|  | const ActionBar = React.createClass({ | ||||||
|  |  | ||||||
|  |   propTypes: { | ||||||
|  |     account: ImmutablePropTypes.map.isRequired, | ||||||
|  |     me: React.PropTypes.number.isRequired, | ||||||
|  |     onFollow: React.PropTypes.func.isRequired, | ||||||
|  |     onUnfollow: React.PropTypes.func.isRequired | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   mixins: [PureRenderMixin], | ||||||
|  |  | ||||||
|  |   render () { | ||||||
|  |     const { account, me } = this.props; | ||||||
|  |      | ||||||
|  |     let followBack   = ''; | ||||||
|  |     let actionButton = ''; | ||||||
|  |  | ||||||
|  |     if (account.get('id') === me) { | ||||||
|  |       actionButton = 'This is you!'; | ||||||
|  |     } else { | ||||||
|  |       if (account.getIn(['relationship', 'following'])) { | ||||||
|  |         actionButton = <Button text='Unfollow' onClick={this.props.onUnfollow} /> | ||||||
|  |       } else { | ||||||
|  |         actionButton = <Button text='Follow' onClick={this.props.onFollow} /> | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (account.getIn(['relationship', 'followed_by'])) { | ||||||
|  |         followBack = 'follows you'; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return ( | ||||||
|  |       <div> | ||||||
|  |         {actionButton} | ||||||
|  |         {account.get('followers_count')} followers | ||||||
|  |         {account.get('following_count')} following | ||||||
|  |         {followBack} | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | export default ActionBar; | ||||||
| @@ -1,13 +1,10 @@ | |||||||
| import PureRenderMixin    from 'react-addons-pure-render-mixin'; | import PureRenderMixin    from 'react-addons-pure-render-mixin'; | ||||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||||
| import Button             from '../../../components/button'; |  | ||||||
|  |  | ||||||
| const Header = React.createClass({ | const Header = React.createClass({ | ||||||
|  |  | ||||||
|   propTypes: { |   propTypes: { | ||||||
|     account: ImmutablePropTypes.map.isRequired, |     account: ImmutablePropTypes.map.isRequired | ||||||
|     onFollow: React.PropTypes.func.isRequired, |  | ||||||
|     onUnfollow: React.PropTypes.func.isRequired |  | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
|   mixins: [PureRenderMixin], |   mixins: [PureRenderMixin], | ||||||
|   | |||||||
| @@ -11,13 +11,13 @@ import { | |||||||
| import { replyCompose }      from '../../actions/compose'; | import { replyCompose }      from '../../actions/compose'; | ||||||
| import { favourite, reblog } from '../../actions/interactions'; | import { favourite, reblog } from '../../actions/interactions'; | ||||||
| import Header                from './components/header'; | import Header                from './components/header'; | ||||||
| import { selectStatus }      from '../../reducers/timelines'; | import { | ||||||
|  |   selectStatus, | ||||||
|  |   selectAccount | ||||||
|  | }                            from '../../reducers/timelines'; | ||||||
| import StatusList            from '../../components/status_list'; | import StatusList            from '../../components/status_list'; | ||||||
| import Immutable             from 'immutable'; | import Immutable             from 'immutable'; | ||||||
|  | import ActionBar             from './components/action_bar'; | ||||||
| function selectAccount(state, id) { |  | ||||||
|   return state.getIn(['timelines', 'accounts', id], null); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| function selectStatuses(state, accountId) { | function selectStatuses(state, accountId) { | ||||||
|   return state.getIn(['timelines', 'accounts_timelines', accountId], Immutable.List()).map(id => selectStatus(state, id)).filterNot(status => status === null); |   return state.getIn(['timelines', 'accounts_timelines', accountId], Immutable.List()).map(id => selectStatus(state, id)).filterNot(status => status === null); | ||||||
| @@ -25,7 +25,8 @@ function selectStatuses(state, accountId) { | |||||||
|  |  | ||||||
| const mapStateToProps = (state, props) => ({ | const mapStateToProps = (state, props) => ({ | ||||||
|   account: selectAccount(state, Number(props.params.accountId)), |   account: selectAccount(state, Number(props.params.accountId)), | ||||||
|   statuses: selectStatuses(state, Number(props.params.accountId)) |   statuses: selectStatuses(state, Number(props.params.accountId)), | ||||||
|  |   me: state.getIn(['timelines', 'me']) | ||||||
| }); | }); | ||||||
|  |  | ||||||
| const Account = React.createClass({ | const Account = React.createClass({ | ||||||
| @@ -76,7 +77,7 @@ const Account = React.createClass({ | |||||||
|   }, |   }, | ||||||
|  |  | ||||||
|   render () { |   render () { | ||||||
|     const { account, statuses } = this.props; |     const { account, statuses, me } = this.props; | ||||||
|  |  | ||||||
|     if (account === null) { |     if (account === null) { | ||||||
|       return <div>Loading {this.props.params.accountId}...</div>; |       return <div>Loading {this.props.params.accountId}...</div>; | ||||||
| @@ -84,7 +85,8 @@ const Account = React.createClass({ | |||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|       <div style={{ display: 'flex', flexDirection: 'column', 'flex': '0 0 auto', height: '100%' }}> |       <div style={{ display: 'flex', flexDirection: 'column', 'flex': '0 0 auto', height: '100%' }}> | ||||||
|         <Header account={account} onFollow={this.handleFollow} onUnfollow={this.handleUnfollow} /> |         <Header account={account} /> | ||||||
|  |         <ActionBar account={account} me={me} onFollow={this.handleFollow} onUnfollow={this.handleUnfollow} /> | ||||||
|         <StatusList statuses={statuses} onScrollToBottom={this.handleScrollToBottom} onReply={this.handleReply} onReblog={this.handleReblog} onFavourite={this.handleFavourite} /> |         <StatusList statuses={statuses} onScrollToBottom={this.handleScrollToBottom} onReply={this.handleReply} onReblog={this.handleReblog} onFavourite={this.handleFavourite} /> | ||||||
|       </div> |       </div> | ||||||
|     ); |     ); | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ export function selectStatus(state, id) { | |||||||
|     return null; |     return null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   status = status.set('account', state.getIn(['timelines', 'accounts', status.get('account')])); |   status = status.set('account', selectAccount(state, status.get('account'))); | ||||||
|  |  | ||||||
|   if (status.get('reblog') !== null) { |   if (status.get('reblog') !== null) { | ||||||
|     status = status.set('reblog', selectStatus(state, status.get('reblog'))); |     status = status.set('reblog', selectStatus(state, status.get('reblog'))); | ||||||
| @@ -48,6 +48,16 @@ export function selectStatus(state, id) { | |||||||
|   return status; |   return status; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export function selectAccount(state, id) { | ||||||
|  |   let account = state.getIn(['timelines', 'accounts', id], null); | ||||||
|  |  | ||||||
|  |   if (account === null) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return account.set('relationship', state.getIn(['timelines', 'relationships', id])); | ||||||
|  | }; | ||||||
|  |  | ||||||
| function normalizeStatus(state, status) { | function normalizeStatus(state, status) { | ||||||
|   // Separate account |   // Separate account | ||||||
|   let account = status.get('account'); |   let account = status.get('account'); | ||||||
| @@ -139,10 +149,18 @@ function deleteStatus(state, id) { | |||||||
|   return state.deleteIn(['statuses', id]); |   return state.deleteIn(['statuses', id]); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| function normalizeAccount(state, account) { | function normalizeAccount(state, account, relationship) { | ||||||
|  |   if (relationship) { | ||||||
|  |     state = normalizeRelationship(state, relationship); | ||||||
|  |   } | ||||||
|  |    | ||||||
|   return state.setIn(['accounts', account.get('id')], account); |   return state.setIn(['accounts', account.get('id')], account); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | function normalizeRelationship(state, relationship) { | ||||||
|  |   return state.setIn(['relationships', relationship.get('id')], relationship); | ||||||
|  | }; | ||||||
|  |  | ||||||
| function setSelf(state, account) { | function setSelf(state, account) { | ||||||
|   state = normalizeAccount(state, account); |   state = normalizeAccount(state, account); | ||||||
|   return state.set('me', account.get('id')); |   return state.set('me', account.get('id')); | ||||||
| @@ -184,9 +202,10 @@ export default function timelines(state = initialState, action) { | |||||||
|       return setSelf(state, Immutable.fromJS(action.account)); |       return setSelf(state, Immutable.fromJS(action.account)); | ||||||
|     case ACCOUNT_FETCH_SUCCESS: |     case ACCOUNT_FETCH_SUCCESS: | ||||||
|     case FOLLOW_SUBMIT_SUCCESS: |     case FOLLOW_SUBMIT_SUCCESS: | ||||||
|  |       return normalizeAccount(state, Immutable.fromJS(action.account), Immutable.fromJS(action.relationship)); | ||||||
|     case ACCOUNT_FOLLOW_SUCCESS: |     case ACCOUNT_FOLLOW_SUCCESS: | ||||||
|     case ACCOUNT_UNFOLLOW_SUCCESS: |     case ACCOUNT_UNFOLLOW_SUCCESS: | ||||||
|       return normalizeAccount(state, Immutable.fromJS(action.account)); |       return normalizeRelationship(state, Immutable.fromJS(action.relationship)); | ||||||
|     case STATUS_FETCH_SUCCESS: |     case STATUS_FETCH_SUCCESS: | ||||||
|       return normalizeContext(state, Immutable.fromJS(action.status), Immutable.fromJS(action.context.ancestors), Immutable.fromJS(action.context.descendants)); |       return normalizeContext(state, Immutable.fromJS(action.status), Immutable.fromJS(action.context.ancestors), Immutable.fromJS(action.context.descendants)); | ||||||
|     case ACCOUNT_TIMELINE_FETCH_SUCCESS: |     case ACCOUNT_TIMELINE_FETCH_SUCCESS: | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| class Api::AccountsController < ApiController | class Api::AccountsController < ApiController | ||||||
|   before_action :set_account |  | ||||||
|   before_action :doorkeeper_authorize! |   before_action :doorkeeper_authorize! | ||||||
|  |   before_action :set_account | ||||||
|   respond_to    :json |   respond_to    :json | ||||||
|  |  | ||||||
|   def show |   def show | ||||||
| @@ -20,12 +20,14 @@ class Api::AccountsController < ApiController | |||||||
|  |  | ||||||
|   def follow |   def follow | ||||||
|     @follow = FollowService.new.(current_user.account, @account.acct) |     @follow = FollowService.new.(current_user.account, @account.acct) | ||||||
|     render action: :show |     set_relationship | ||||||
|  |     render action: :relationship | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def unfollow |   def unfollow | ||||||
|     @unfollow = UnfollowService.new.(current_user.account, @account) |     @unfollow = UnfollowService.new.(current_user.account, @account) | ||||||
|     render action: :show |     set_relationship | ||||||
|  |     render action: :relationship | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def relationships |   def relationships | ||||||
| @@ -41,4 +43,10 @@ class Api::AccountsController < ApiController | |||||||
|   def set_account |   def set_account | ||||||
|     @account = Account.find(params[:id]) |     @account = Account.find(params[:id]) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  |   def set_relationship | ||||||
|  |     @following   = Account.following_map([@account.id], current_user.account_id) | ||||||
|  |     @followed_by = Account.followed_by_map([@account.id], current_user.account_id) | ||||||
|  |     @blocking    = {} | ||||||
|  |   end | ||||||
| end | end | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								app/views/api/accounts/relationship.rabl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								app/views/api/accounts/relationship.rabl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | object @account | ||||||
|  | attribute :id | ||||||
|  | node(:following)   { |account| @following[account.id]   || false } | ||||||
|  | node(:followed_by) { |account| @followed_by[account.id] || false } | ||||||
|  | node(:blocking)    { |account| @blocking[account.id]    || false } | ||||||
| @@ -1,5 +1,2 @@ | |||||||
| collection @accounts | collection @accounts | ||||||
| attribute :id | extends 'api/accounts/relationship' | ||||||
| node(:following)   { |account| @following[account.id]   || false } |  | ||||||
| node(:followed_by) { |account| @followed_by[account.id] || false } |  | ||||||
| node(:blocking)    { |account| @blocking[account.id]    || false } |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user