// Copyright (c) 2015 Mattermost, Inc. All Rights Reserved. // See License.txt for license information. import AudioVideoPreview from './audio_video_preview.jsx'; import CodePreview from './code_preview.jsx'; import PDFPreview from './pdf_preview.jsx'; import FileInfoPreview from './file_info_preview.jsx'; import ViewImagePopoverBar from './view_image_popover_bar.jsx'; import * as GlobalActions from 'actions/global_actions.jsx'; import FileStore from 'stores/file_store.jsx'; import * as AsyncClient from 'utils/async_client.jsx'; import * as Utils from 'utils/utils.jsx'; import Constants from 'utils/constants.jsx'; const KeyCodes = Constants.KeyCodes; import $ from 'jquery'; import React from 'react'; import {intlShape, injectIntl, defineMessages} from 'react-intl'; import {Modal} from 'react-bootstrap'; import loadingGif from 'images/load.gif'; const holders = defineMessages({ loading: { id: 'view_image.loading', defaultMessage: 'Loading ' } }); class ViewImageModal extends React.Component { constructor(props) { super(props); this.showImage = this.showImage.bind(this); this.loadImage = this.loadImage.bind(this); this.handleNext = this.handleNext.bind(this); this.handlePrev = this.handlePrev.bind(this); this.handleKeyPress = this.handleKeyPress.bind(this); this.onModalShown = this.onModalShown.bind(this); this.onModalHidden = this.onModalHidden.bind(this); this.onFileStoreChange = this.onFileStoreChange.bind(this); this.handleGetPublicLink = this.handleGetPublicLink.bind(this); this.onMouseEnterImage = this.onMouseEnterImage.bind(this); this.onMouseLeaveImage = this.onMouseLeaveImage.bind(this); this.state = { imgId: this.props.startId, fileInfo: null, imgHeight: '100%', loaded: Utils.fillArray(false, this.props.filenames.length), progress: Utils.fillArray(0, this.props.filenames.length), showFooter: false }; } handleNext(e) { if (e) { e.stopPropagation(); } let id = this.state.imgId + 1; if (id > this.props.filenames.length - 1) { id = 0; } this.showImage(id); } handlePrev(e) { if (e) { e.stopPropagation(); } let id = this.state.imgId - 1; if (id < 0) { id = this.props.filenames.length - 1; } this.showImage(id); } handleKeyPress(e) { if (e.keyCode === KeyCodes.RIGHT) { this.handleNext(); } else if (e.keyCode === KeyCodes.LEFT) { this.handlePrev(); } } onModalShown(nextProps) { $(window).on('keyup', this.handleKeyPress); this.showImage(nextProps.startId); FileStore.addChangeListener(this.onFileStoreChange); } onModalHidden() { $(window).off('keyup', this.handleKeyPress); if (this.refs.video) { this.refs.video.stop(); } FileStore.removeChangeListener(this.onFileStoreChange); } componentWillReceiveProps(nextProps) { if (nextProps.show === true && this.props.show === false) { this.onModalShown(nextProps); } else if (nextProps.show === false && this.props.show === true) { this.onModalHidden(); } if (!Utils.areObjectsEqual(this.props.filenames, nextProps.filenames)) { this.setState({ loaded: Utils.fillArray(false, nextProps.filenames.length), progress: Utils.fillArray(0, nextProps.filenames.length) }); } } onFileStoreChange(filename) { const id = this.props.filenames.indexOf(filename); if (id !== -1) { if (id === this.state.imgId) { this.setState({ fileInfo: FileStore.getInfo(filename) }); } if (!this.state.loaded[id]) { this.loadImage(id, filename); } } } showImage(id) { this.setState({imgId: id}); const imgHeight = $(window).height() - 100; this.setState({imgHeight}); const filename = this.props.filenames[id]; if (!FileStore.hasInfo(filename)) { // the image will actually be loaded once we know what we need to load AsyncClient.getFileInfo(filename); return; } this.setState({ fileInfo: FileStore.getInfo(filename) }); if (!this.state.loaded[id]) { this.loadImage(id, filename); } } loadImage(id, filename) { const fileInfo = FileStore.getInfo(filename); const fileType = Utils.getFileType(fileInfo.extension); if (fileType === 'image') { let previewUrl; if (fileInfo.has_image_preview) { previewUrl = Utils.getPreviewImagePath(filename); } else { // some images (eg animated gifs) just show the file itself and not a preview previewUrl = Utils.getFileUrl(filename); } const img = new Image(); img.load( previewUrl, () => { const progress = this.state.progress; progress[id] = img.completedPercentage; this.setState({progress}); } ); img.onload = () => { const loaded = this.state.loaded; loaded[id] = true; this.setState({loaded}); }; } else { // there's nothing to load for non-image files var loaded = this.state.loaded; loaded[id] = true; this.setState({loaded}); } } handleGetPublicLink() { this.props.onModalDismissed(); GlobalActions.showGetPublicLinkModal(this.props.filenames[this.state.imgId]); } onMouseEnterImage() { this.setState({showFooter: true}); } onMouseLeaveImage() { this.setState({showFooter: false}); } render() { if (this.props.filenames.length < 1 || this.props.filenames.length - 1 < this.state.imgId) { return
; } const filename = this.props.filenames[this.state.imgId]; const fileUrl = Utils.getFileUrl(filename); var content; if (this.state.loaded[this.state.imgId]) { // this.state.fileInfo is for the current image and we shoudl have it before we load the image const fileInfo = this.state.fileInfo; const fileType = Utils.getFileType(fileInfo.extension); if (fileType === 'image') { content = (