diff options
Diffstat (limited to 'web/react/components/posts_view.jsx')
-rw-r--r-- | web/react/components/posts_view.jsx | 604 |
1 files changed, 0 insertions, 604 deletions
diff --git a/web/react/components/posts_view.jsx b/web/react/components/posts_view.jsx deleted file mode 100644 index 0a9232850..000000000 --- a/web/react/components/posts_view.jsx +++ /dev/null @@ -1,604 +0,0 @@ -// Copyright (c) 2015 Mattermost, Inc. All Rights Reserved. -// See License.txt for license information. - -import PreferenceStore from '../stores/preference_store.jsx'; -import * as GlobalActions from '../action_creators/global_actions.jsx'; -import * as Utils from '../utils/utils.jsx'; -import Post from './post.jsx'; -import Constants from '../utils/constants.jsx'; -import DelayedAction from '../utils/delayed_action.jsx'; - -import {FormattedDate, FormattedMessage} from 'mm-intl'; - -const Preferences = Constants.Preferences; - -export default class PostsView extends React.Component { - constructor(props) { - super(props); - - this.updateState = this.updateState.bind(this); - this.handleScroll = this.handleScroll.bind(this); - this.handleScrollStop = this.handleScrollStop.bind(this); - this.isAtBottom = this.isAtBottom.bind(this); - this.loadMorePostsTop = this.loadMorePostsTop.bind(this); - this.loadMorePostsBottom = this.loadMorePostsBottom.bind(this); - this.createPosts = this.createPosts.bind(this); - this.updateScrolling = this.updateScrolling.bind(this); - this.handleResize = this.handleResize.bind(this); - this.scrollToBottom = this.scrollToBottom.bind(this); - this.scrollToBottomAnimated = this.scrollToBottomAnimated.bind(this); - - this.jumpToPostNode = null; - this.wasAtBottom = true; - this.scrollHeight = 0; - - this.scrollStopAction = new DelayedAction(this.handleScrollStop); - - this.state = { - displayNameType: PreferenceStore.get(Preferences.CATEGORY_DISPLAY_SETTINGS, 'name_format', 'false'), - isScrolling: false, - topPostId: null - }; - } - static get SCROLL_TYPE_FREE() { - return 1; - } - static get SCROLL_TYPE_BOTTOM() { - return 2; - } - static get SCROLL_TYPE_SIDEBAR_OPEN() { - return 3; - } - static get SCROLL_TYPE_NEW_MESSAGE() { - return 4; - } - static get SCROLL_TYPE_POST() { - return 5; - } - updateState() { - this.setState({displayNameType: PreferenceStore.get(Preferences.CATEGORY_DISPLAY_SETTINGS, 'name_format', 'false')}); - } - isAtBottom() { - // consider the view to be at the bottom if it's within this many pixels of the bottom - const atBottomMargin = 10; - - return this.refs.postlist.clientHeight + this.refs.postlist.scrollTop >= this.refs.postlist.scrollHeight - atBottomMargin; - } - handleScroll() { - // HACK FOR RHS -- REMOVE WHEN RHS DIES - const childNodes = this.refs.postlistcontent.childNodes; - for (let i = 0; i < childNodes.length; i++) { - // If the node is 1/3 down the page - if (childNodes[i].offsetTop > (this.refs.postlist.scrollTop + (this.refs.postlist.offsetHeight / 3))) { - this.jumpToPostNode = childNodes[i]; - break; - } - } - this.wasAtBottom = this.isAtBottom(); - if (!this.jumpToPostNode && childNodes.length > 0) { - this.jumpToPostNode = childNodes[childNodes.length - 1]; - } - - // --- -------- - - this.props.postViewScrolled(this.isAtBottom()); - this.prevScrollHeight = this.refs.postlist.scrollHeight; - this.prevOffsetTop = this.jumpToPostNode.offsetTop; - - this.updateFloatingTimestamp(); - - if (!this.state.isScrolling) { - this.setState({ - isScrolling: true - }); - } - - this.scrollStopAction.fireAfter(2000); - } - handleScrollStop() { - this.setState({ - isScrolling: false - }); - } - updateFloatingTimestamp() { - // skip this in non-mobile view since that's when the timestamp is visible - if ($(window).width() > 768) { - return; - } - - if (this.props.postList) { - // iterate through posts starting at the bottom since users are more likely to be viewing newer posts - for (let i = 0; i < this.props.postList.order.length; i++) { - const id = this.props.postList.order[i]; - const element = ReactDOM.findDOMNode(this.refs[id]); - - if (!element || element.offsetTop + element.clientHeight <= this.refs.postlist.scrollTop) { - // this post is off the top of the screen so the last one is at the top of the screen - let topPostId; - - if (i > 0) { - topPostId = this.props.postList.order[i - 1]; - } else { - // the first post we look at should always be on the screen, but handle that case anyway - topPostId = id; - } - - if (topPostId !== this.state.topPostId) { - this.setState({ - topPostId - }); - } - - break; - } - } - } - } - loadMorePostsTop() { - this.props.loadMorePostsTopClicked(); - } - loadMorePostsBottom() { - this.props.loadMorePostsBottomClicked(); - } - createPosts(posts, order) { - const postCtls = []; - let previousPostDay = new Date(0); - const userId = this.props.currentUser.id; - const profiles = this.props.profiles || {}; - - let renderedLastViewed = false; - - for (let i = order.length - 1; i >= 0; i--) { - const post = posts[order[i]]; - const parentPost = posts[post.parent_id]; - const prevPost = posts[order[i + 1]]; - const postUserId = Utils.isSystemMessage(post) ? '' : post.user_id; - - // If the post is a comment whose parent has been deleted, don't add it to the list. - if (parentPost && parentPost.state === Constants.POST_DELETED) { - continue; - } - - let sameUser = false; - let sameRoot = false; - let hideProfilePic = false; - - if (prevPost) { - const postIsComment = Utils.isComment(post); - const prevPostIsComment = Utils.isComment(prevPost); - const postFromWebhook = Boolean(post.props && post.props.from_webhook); - const prevPostFromWebhook = Boolean(prevPost.props && prevPost.props.from_webhook); - const prevPostUserId = Utils.isSystemMessage(prevPost) ? '' : prevPost.user_id; - let prevWebhookName = ''; - if (prevPost.props && prevPost.props.override_username) { - prevWebhookName = prevPost.props.override_username; - } - let curWebhookName = ''; - if (post.props && post.props.override_username) { - curWebhookName = post.props.override_username; - } - - // consider posts from the same user if: - // the previous post was made by the same user as the current post, - // the previous post was made within 5 minutes of the current post, - // the previous post and current post are both from webhooks or both not, - // the previous post and current post have the same webhook usernames - if (prevPostUserId === postUserId && - post.create_at - prevPost.create_at <= 1000 * 60 * 5 && - postFromWebhook === prevPostFromWebhook && - prevWebhookName === curWebhookName) { - sameUser = true; - } - - // consider posts from the same root if: - // the current post is a comment, - // the current post has the same root as the previous post - if (postIsComment && (prevPost.id === post.root_id || prevPost.root_id === post.root_id)) { - sameRoot = true; - } - - // consider posts from the same root if: - // the current post is not a comment, - // the previous post is not a comment, - // the previous post is from the same user - if (!postIsComment && !prevPostIsComment && sameUser) { - sameRoot = true; - } - - // hide the profile pic if: - // the previous post was made by the same user as the current post, - // the previous post is not a comment, - // the current post is not a comment, - // the previous post and current post are both from webhooks or both not, - // the previous post and current post have the same webhook usernames - if (prevPostUserId === postUserId && - !prevPostIsComment && - !postIsComment && - postFromWebhook === prevPostFromWebhook && - prevWebhookName === curWebhookName) { - hideProfilePic = true; - } - } - - // check if it's the last comment in a consecutive string of comments on the same post - // it is the last comment if it is last post in the channel or the next post has a different root post - const isLastComment = Utils.isComment(post) && (i === 0 || posts[order[i - 1]].root_id !== post.root_id); - - const keyPrefix = post.id ? post.id : i; - - const shouldHighlight = this.props.postsToHighlight && this.props.postsToHighlight.hasOwnProperty(post.id); - - let profile; - if (this.props.currentUser.id === post.user_id) { - profile = this.props.currentUser; - } else { - profile = profiles[post.user_id]; - } - - const postCtl = ( - <Post - key={keyPrefix + 'postKey'} - ref={post.id} - sameUser={sameUser} - sameRoot={sameRoot} - post={post} - parentPost={parentPost} - posts={posts} - hideProfilePic={hideProfilePic} - isLastComment={isLastComment} - shouldHighlight={shouldHighlight} - onClick={() => GlobalActions.emitPostFocusEvent(post.id)} //eslint-disable-line no-loop-func - displayNameType={this.state.displayNameType} - user={profile} - currentUser={this.props.currentUser} - /> - ); - - const currentPostDay = Utils.getDateForUnixTicks(post.create_at); - if (currentPostDay.toDateString() !== previousPostDay.toDateString()) { - postCtls.push( - <div - key={currentPostDay.toDateString()} - className='date-separator' - > - <hr className='separator__hr'/> - <div className='separator__text'> - <FormattedDate - value={currentPostDay} - weekday='short' - month='short' - day='2-digit' - year='numeric' - /> - </div> - </div> - ); - } - - if (postUserId !== userId && - this.props.messageSeparatorTime !== 0 && - post.create_at > this.props.messageSeparatorTime && - !renderedLastViewed) { - renderedLastViewed = true; - - // Temporary fix to solve ie11 rendering issue - let newSeparatorId = ''; - if (!Utils.isBrowserIE()) { - newSeparatorId = 'new_message_' + post.id; - } - postCtls.push( - <div - id={newSeparatorId} - key='unviewed' - ref='newMessageSeparator' - className='new-separator' - > - <hr - className='separator__hr' - /> - <div className='separator__text'> - <FormattedMessage - id='posts_view.newMsg' - defaultMessage='New Messages' - /> - </div> - </div> - ); - } - postCtls.push(postCtl); - previousPostDay = currentPostDay; - } - - return postCtls; - } - updateScrolling() { - if (this.props.scrollType === PostsView.SCROLL_TYPE_BOTTOM) { - this.scrollToBottom(); - } else if (this.props.scrollType === PostsView.SCROLL_TYPE_NEW_MESSAGE) { - window.requestAnimationFrame(() => { - // If separator exists scroll to it. Otherwise scroll to bottom. - if (this.refs.newMessageSeparator) { - var objDiv = this.refs.postlist; - objDiv.scrollTop = this.refs.newMessageSeparator.offsetTop; //scrolls node to top of Div - } else if (this.refs.postlist) { - this.refs.postlist.scrollTop = this.refs.postlist.scrollHeight; - } - }); - } else if (this.props.scrollType === PostsView.SCROLL_TYPE_POST && this.props.scrollPostId) { - window.requestAnimationFrame(() => { - const postNode = ReactDOM.findDOMNode(this.refs[this.props.scrollPostId]); - if (postNode == null) { - return; - } - postNode.scrollIntoView(); - if (this.refs.postlist.scrollTop === postNode.offsetTop) { - this.refs.postlist.scrollTop -= (this.refs.postlist.offsetHeight / 3); - } else { - this.refs.postlist.scrollTop -= (this.refs.postlist.offsetHeight / 3) + (this.refs.postlist.scrollTop - postNode.offsetTop); - } - }); - } else if (this.props.scrollType === PostsView.SCROLL_TYPE_SIDEBAR_OPEN) { - // If we are at the bottom then stay there - if (this.wasAtBottom) { - this.refs.postlist.scrollTop = this.refs.postlist.scrollHeight; - } else { - window.requestAnimationFrame(() => { - this.jumpToPostNode.scrollIntoView(); - if (this.refs.postlist.scrollTop === this.jumpToPostNode.offsetTop) { - this.refs.postlist.scrollTop -= (this.refs.postlist.offsetHeight / 3); - } else { - this.refs.postlist.scrollTop -= (this.refs.postlist.offsetHeight / 3) + (this.refs.postlist.scrollTop - this.jumpToPostNode.offsetTop); - } - }); - } - } else if (this.refs.postlist.scrollHeight !== this.prevScrollHeight) { - window.requestAnimationFrame(() => { - // Only need to jump if we added posts to the top. - if (this.jumpToPostNode && (this.jumpToPostNode.offsetTop !== this.prevOffsetTop)) { - this.refs.postlist.scrollTop += (this.refs.postlist.scrollHeight - this.prevScrollHeight); - } - }); - } - } - handleResize() { - this.updateScrolling(); - } - scrollToBottom() { - window.requestAnimationFrame(() => { - this.refs.postlist.scrollTop = this.refs.postlist.scrollHeight; - }); - } - scrollToBottomAnimated() { - var postList = $(this.refs.postlist); - postList.animate({scrollTop: this.refs.postlist.scrollHeight}, '500'); - } - componentDidMount() { - if (this.props.postList != null) { - this.updateScrolling(); - } - window.addEventListener('resize', this.handleResize); - } - componentWillUnmount() { - window.removeEventListener('resize', this.handleResize); - } - componentDidUpdate() { - if (this.props.postList != null) { - this.updateScrolling(); - } - } - componentWillReceiveProps(nextProps) { - if (!this.props.isActive && nextProps.isActive) { - this.updateState(); - PreferenceStore.addChangeListener(this.updateState); - } else if (this.props.isActive && !nextProps.isActive) { - PreferenceStore.removeChangeListener(this.updateState); - } - } - shouldComponentUpdate(nextProps, nextState) { - if (this.props.isActive !== nextProps.isActive) { - return true; - } - if (this.props.postList !== nextProps.postList) { - return true; - } - if (this.props.scrollPostId !== nextProps.scrollPostId) { - return true; - } - if (this.props.scrollType !== nextProps.scrollType && nextProps.scrollType !== PostsView.SCROLL_TYPE_FREE) { - return true; - } - if (this.props.messageSeparatorTime !== nextProps.messageSeparatorTime) { - return true; - } - if (!Utils.areObjectsEqual(this.props.postList, nextProps.postList)) { - return true; - } - if (nextState.displayNameType !== this.state.displayNameType) { - return true; - } - if (this.state.topPostId !== nextState.topPostId) { - return true; - } - if (this.state.isScrolling !== nextState.isScrolling) { - return true; - } - if (!Utils.areObjectsEqual(this.props.profiles, nextProps.profiles)) { - return true; - } - - return false; - } - render() { - let posts = []; - let order = []; - let moreMessagesTop; - let moreMessagesBottom; - let postElements; - let activeClass = 'inactive'; - if (this.props.postList != null) { - posts = this.props.postList.posts; - order = this.props.postList.order; - - // Create intro message or top loadmore link - if (this.props.showMoreMessagesTop) { - moreMessagesTop = ( - <a - ref='loadmoretop' - className='more-messages-text theme' - href='#' - onClick={this.loadMorePostsTop} - > - <FormattedMessage - id='posts_view.loadMore' - defaultMessage='Load more messages' - /> - </a> - ); - } else { - moreMessagesTop = this.props.introText; - } - - // Give option to load more posts at bottom if nessisary - if (this.props.showMoreMessagesBottom) { - moreMessagesBottom = ( - <a - ref='loadmorebottom' - className='more-messages-text theme' - href='#' - onClick={this.loadMorePostsBottom} - > - <FormattedMessage id='posts_view.loadMore'/> - </a> - ); - } else { - moreMessagesBottom = null; - } - - // Create post elements - postElements = this.createPosts(posts, order); - - // Show ourselves if we are marked active - if (this.props.isActive) { - activeClass = ''; - } - } - - let topPost = null; - if (this.state.topPostId) { - topPost = this.props.postList.posts[this.state.topPostId]; - } - - return ( - <div className={activeClass}> - <FloatingTimestamp - isScrolling={this.state.isScrolling} - post={topPost} - /> - <ScrollToBottomArrows - isScrolling={this.state.isScrolling} - atBottom={this.wasAtBottom} - onClick={this.scrollToBottomAnimated} - /> - <div - ref='postlist' - className='post-list-holder-by-time' - onScroll={this.handleScroll} - > - <div className='post-list__table'> - <div - ref='postlistcontent' - className='post-list__content' - > - {moreMessagesTop} - {postElements} - {moreMessagesBottom} - </div> - </div> - </div> - </div> - ); - } -} -PostsView.defaultProps = { -}; - -PostsView.propTypes = { - isActive: React.PropTypes.bool, - postList: React.PropTypes.object, - profiles: React.PropTypes.object.isRequired, - scrollPostId: React.PropTypes.string, - scrollType: React.PropTypes.number, - postViewScrolled: React.PropTypes.func.isRequired, - loadMorePostsTopClicked: React.PropTypes.func.isRequired, - loadMorePostsBottomClicked: React.PropTypes.func.isRequired, - showMoreMessagesTop: React.PropTypes.bool, - showMoreMessagesBottom: React.PropTypes.bool, - introText: React.PropTypes.element, - messageSeparatorTime: React.PropTypes.number, - postsToHighlight: React.PropTypes.object, - currentUser: React.PropTypes.object.isRequired -}; - -function FloatingTimestamp({isScrolling, post}) { - // only show on mobile - if ($(window).width() > 768) { - return <noscript/>; - } - - if (!post) { - return <noscript/>; - } - - const dateString = ( - <FormattedDate - value={post.create_at} - weekday='short' - day='2-digit' - month='short' - year='numeric' - /> - ); - - let className = 'post-list__timestamp'; - if (isScrolling) { - className += ' scrolling'; - } - - return ( - <div className={className}> - <span>{dateString}</span> - </div> - ); -} - -FloatingTimestamp.propTypes = { - isScrolling: React.PropTypes.bool.isRequired, - post: React.PropTypes.object -}; - -function ScrollToBottomArrows({isScrolling, atBottom, onClick}) { - // only show on mobile - if ($(window).width() > 768) { - return <noscript/>; - } - - let className = 'post-list__arrows'; - if (isScrolling && !atBottom) { - className += ' scrolling'; - } - - return ( - <div - className={className} - onClick={onClick} - > - <span dangerouslySetInnerHTML={{__html: Constants.SCROLL_BOTTOM_ICON}}/> - </div> - ); -} - -ScrollToBottomArrows.propTypes = { - isScrolling: React.PropTypes.bool.isRequired, - atBottom: React.PropTypes.bool.isRequired, - onClick: React.PropTypes.func.isRequired -}; |