diff options
author | Joram Wilander <jwawilander@gmail.com> | 2017-06-18 14:42:32 -0400 |
---|---|---|
committer | GitHub <noreply@github.com> | 2017-06-18 14:42:32 -0400 |
commit | ab67f6e257f6e8f08145a02a7b93550f99641be4 (patch) | |
tree | d33d1c58a3d229f7e37db58bc2c397ac3806c503 /webapp/components/post_view/post_attachment_opengraph | |
parent | 0231e95f1c5a8c42ba97875f0d2301815f552974 (diff) | |
download | chat-ab67f6e257f6e8f08145a02a7b93550f99641be4.tar.gz chat-ab67f6e257f6e8f08145a02a7b93550f99641be4.tar.bz2 chat-ab67f6e257f6e8f08145a02a7b93550f99641be4.zip |
PLT-6215 Major post list refactor (#6501)
* Major post list refactor
* Fix post and thread deletion
* Fix preferences not selecting correctly
* Fix military time displaying
* Fix UP key for editing posts
* Fix ESLint error
* Various fixes and updates per feedback
* Fix for permalink view
* Revert to old scrolling method and various fixes
* Add floating timestamp, new message indicator, scroll arrows
* Update post loading for focus mode and add visibility limit
* Fix pinning posts and a react warning
* Add loading UI updates from Asaad
* Fix refreshing loop
* Temporarily bump post visibility limit
* Update infinite scrolling
* Remove infinite scrolling
Diffstat (limited to 'webapp/components/post_view/post_attachment_opengraph')
-rw-r--r-- | webapp/components/post_view/post_attachment_opengraph/index.js | 26 | ||||
-rw-r--r-- | webapp/components/post_view/post_attachment_opengraph/post_attachment_opengraph.jsx | 259 |
2 files changed, 285 insertions, 0 deletions
diff --git a/webapp/components/post_view/post_attachment_opengraph/index.js b/webapp/components/post_view/post_attachment_opengraph/index.js new file mode 100644 index 000000000..e0bec8f36 --- /dev/null +++ b/webapp/components/post_view/post_attachment_opengraph/index.js @@ -0,0 +1,26 @@ +// Copyright (c) 2017 Mattermost, Inc. All Rights Reserved. +// See License.txt for license information. + +import {connect} from 'react-redux'; +import {bindActionCreators} from 'redux'; +import {getOpenGraphMetadata} from 'mattermost-redux/actions/posts'; +import {getOpenGraphMetadataForUrl} from 'mattermost-redux/selectors/entities/posts'; + +import PostAttachmentOpenGraph from './post_attachment_opengraph.jsx'; + +function mapStateToProps(state, ownProps) { + return { + ...ownProps, + openGraphData: getOpenGraphMetadataForUrl(state, ownProps.link) + }; +} + +function mapDispatchToProps(dispatch) { + return { + actions: bindActionCreators({ + getOpenGraphMetadata + }, dispatch) + }; +} + +export default connect(mapStateToProps, mapDispatchToProps)(PostAttachmentOpenGraph); diff --git a/webapp/components/post_view/post_attachment_opengraph/post_attachment_opengraph.jsx b/webapp/components/post_view/post_attachment_opengraph/post_attachment_opengraph.jsx new file mode 100644 index 000000000..dbf8f6049 --- /dev/null +++ b/webapp/components/post_view/post_attachment_opengraph/post_attachment_opengraph.jsx @@ -0,0 +1,259 @@ +// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. +// See License.txt for license information. + +import React from 'react'; +import PropTypes from 'prop-types'; + +import * as Utils from 'utils/utils.jsx'; +import * as CommonUtils from 'utils/commons.jsx'; + +export default class PostAttachmentOpenGraph extends React.PureComponent { + static propTypes = { + + /** + * The link to display the open graph data for + */ + link: PropTypes.string.isRequired, + + /** + * The open graph data to render + */ + openGraphData: PropTypes.object.isRequired, + + /** + * Set to collapse the preview + */ + previewCollapsed: PropTypes.string, + actions: PropTypes.shape({ + + /** + * The function to get open graph data for a link + */ + getOpenGraphMetadata: PropTypes.func.isRequired + }).isRequired + } + + constructor(props) { + super(props); + this.largeImageMinWidth = 150; + this.imageDimentions = { // Image dimentions in pixels. + height: 80, + width: 80 + }; + this.textMaxLenght = 300; + this.textEllipsis = '...'; + this.largeImageMinRatio = 16 / 9; + this.smallImageContainerLeftPadding = 15; + + this.imageRatio = null; + + this.smallImageContainer = null; + this.smallImageElement = null; + + this.fetchData = this.fetchData.bind(this); + this.toggleImageVisibility = this.toggleImageVisibility.bind(this); + this.onImageLoad = this.onImageLoad.bind(this); + this.onImageError = this.onImageError.bind(this); + this.truncateText = this.truncateText.bind(this); + } + + IMAGE_LOADED = { + LOADING: 'loading', + YES: 'yes', + ERROR: 'error' + } + + componentWillMount() { + this.setState({ + imageLoaded: this.IMAGE_LOADED.LOADING, + imageVisible: this.props.previewCollapsed.startsWith('false'), + hasLargeImage: false + }); + this.fetchData(this.props.link); + } + + componentWillReceiveProps(nextProps) { + if (nextProps.link !== this.props.link) { + this.fetchData(nextProps.link); + } + } + + fetchData(url) { + if (!this.props.openGraphData) { + this.props.actions.getOpenGraphMetadata(url); + } + } + + getBestImageUrl() { + if (Utils.isEmptyObject(this.props.openGraphData.images)) { + return null; + } + + const bestImage = CommonUtils.getNearestPoint(this.imageDimentions, this.props.openGraphData.images, 'width', 'height'); + return bestImage.secure_url || bestImage.url; + } + + toggleImageVisibility() { + this.setState({imageVisible: !this.state.imageVisible}); + } + + onImageLoad(image) { + this.imageRatio = image.target.naturalWidth / image.target.naturalHeight; + if ( + image.target.naturalWidth >= this.largeImageMinWidth && + this.imageRatio >= this.largeImageMinRatio && + !this.state.hasLargeImage + ) { + this.setState({ + hasLargeImage: true + }); + } + this.setState({ + imageLoaded: this.IMAGE_LOADED.YES + }); + } + + onImageError() { + this.setState({imageLoaded: this.IMAGE_LOADED.ERROR}); + } + + loadImage(src) { + const img = new Image(); + img.onload = this.onImageLoad; + img.onerror = this.onImageError; + img.src = src; + } + + imageToggleAnchoreTag(imageUrl) { + if (imageUrl && this.state.hasLargeImage) { + return ( + <a + className={'post__embed-visibility'} + data-expanded={this.state.imageVisible} + aria-label='Toggle Embed Visibility' + onClick={this.toggleImageVisibility} + /> + ); + } + return null; + } + + wrapInSmallImageContainer(imageElement) { + return ( + <div + className='attachment__image__container--openraph' + ref={(div) => { + this.smallImageContainer = div; + }} + > + {imageElement} + </div> + ); + } + + imageTag(imageUrl, renderingForLargeImage = false) { + var element = null; + if ( + imageUrl && renderingForLargeImage === this.state.hasLargeImage && + (!renderingForLargeImage || (renderingForLargeImage && this.state.imageVisible)) + ) { + if (this.state.imageLoaded === this.IMAGE_LOADED.LOADING) { + if (renderingForLargeImage) { + element = <img className={'attachment__image attachment__image--openraph loading large_image'}/>; + } else { + element = this.wrapInSmallImageContainer( + <img className={'attachment__image attachment__image--openraph loading '}/> + ); + } + } else if (this.state.imageLoaded === this.IMAGE_LOADED.YES) { + if (renderingForLargeImage) { + element = ( + <img + className={'attachment__image attachment__image--openraph large_image'} + src={imageUrl} + /> + ); + } else { + element = this.wrapInSmallImageContainer( + <img + className={'attachment__image attachment__image--openraph'} + src={imageUrl} + ref={(img) => { + this.smallImageElement = img; + }} + /> + ); + } + } else if (this.state.imageLoaded === this.IMAGE_LOADED.ERROR) { + return null; + } + } + return element; + } + + truncateText(text, maxLength = this.textMaxLenght, ellipsis = this.textEllipsis) { + if (text.length > maxLength) { + return text.substring(0, maxLength - ellipsis.length) + ellipsis; + } + return text; + } + + render() { + if (!this.props.openGraphData || Utils.isEmptyObject(this.props.openGraphData.description)) { + return null; + } + + const data = this.props.openGraphData; + const imageUrl = this.getBestImageUrl(); + + if (imageUrl) { + this.loadImage(imageUrl); + } + + return ( + <div + className='attachment attachment--opengraph' + ref='attachment' + > + <div className='attachment__content'> + <div + className={'clearfix attachment__container attachment__container--opengraph'} + > + <div + className={'attachment__body__wrap attachment__body__wrap--opengraph'} + > + <span className='sitename'>{this.truncateText(data.site_name)}</span> + <h1 + className={'attachment__title attachment__title--opengraph' + (data.title ? '' : ' is-url')} + > + <a + className='attachment__title-link attachment__title-link--opengraph' + href={data.url || this.props.link} + target='_blank' + rel='noopener noreferrer' + title={data.title || data.url || this.props.link} + > + {this.truncateText(data.title || data.url || this.props.link)} + </a> + </h1> + <div > + <div + className={'attachment__body attachment__body--opengraph'} + > + <div> + <div> + {this.truncateText(data.description)} + {this.imageToggleAnchoreTag(imageUrl)} + </div> + {this.imageTag(imageUrl, true)} + </div> + </div> + </div> + </div> + {this.imageTag(imageUrl, false)} + </div> + </div> + </div> + ); + } +} |