import React, {Component} from 'react'; import PropTypes from 'prop-types'; import {StyleSheet, requireNativeComponent, NativeModules, View, ViewPropTypes, Image, Platform} from 'react-native'; import resolveAssetSource from 'react-native/Libraries/Image/resolveAssetSource'; import TextTrackType from './TextTrackType'; import VideoResizeMode from './VideoResizeMode.js'; const styles = StyleSheet.create({ base: { overflow: 'hidden', }, }); export { TextTrackType }; export default class Video extends Component { constructor(props) { super(props); this.state = { showPoster: true, }; } setNativeProps(nativeProps) { this._root.setNativeProps(nativeProps); } toTypeString(x) { switch (typeof x) { case "object": return x instanceof Date ? x.toISOString() : JSON.stringify(x); // object, null case "undefined": return ""; default: // boolean, number, string return x.toString(); } } stringsOnlyObject(obj) { const strObj = {}; Object.keys(obj).forEach(x => { strObj[x] = this.toTypeString(obj[x]); }); return strObj; } seek = (time, tolerance = 100) => { if (Platform.OS === 'ios') { this.setNativeProps({ seek: { time, tolerance } }); } else { this.setNativeProps({ seek: time }); } }; presentFullscreenPlayer = () => { this.setNativeProps({ fullscreen: true }); }; dismissFullscreenPlayer = () => { this.setNativeProps({ fullscreen: false }); }; _assignRoot = (component) => { this._root = component; }; _onLoadStart = (event) => { if (this.props.onLoadStart) { this.props.onLoadStart(event.nativeEvent); } }; _onLoad = (event) => { if (this.props.onLoad) { this.props.onLoad(event.nativeEvent); } }; _onError = (event) => { if (this.props.onError) { this.props.onError(event.nativeEvent); } }; _onProgress = (event) => { if (this.props.onProgress) { this.props.onProgress(event.nativeEvent); } }; _onSeek = (event) => { if (this.state.showPoster && !this.props.audioOnly) { this.setState({showPoster: false}); } if (this.props.onSeek) { this.props.onSeek(event.nativeEvent); } }; _onEnd = (event) => { if (this.props.onEnd) { this.props.onEnd(event.nativeEvent); } }; _onTimedMetadata = (event) => { if (this.props.onTimedMetadata) { this.props.onTimedMetadata(event.nativeEvent); } }; _onFullscreenPlayerWillPresent = (event) => { if (this.props.onFullscreenPlayerWillPresent) { this.props.onFullscreenPlayerWillPresent(event.nativeEvent); } }; _onFullscreenPlayerDidPresent = (event) => { if (this.props.onFullscreenPlayerDidPresent) { this.props.onFullscreenPlayerDidPresent(event.nativeEvent); } }; _onFullscreenPlayerWillDismiss = (event) => { if (this.props.onFullscreenPlayerWillDismiss) { this.props.onFullscreenPlayerWillDismiss(event.nativeEvent); } }; _onFullscreenPlayerDidDismiss = (event) => { if (this.props.onFullscreenPlayerDidDismiss) { this.props.onFullscreenPlayerDidDismiss(event.nativeEvent); } }; _onReadyForDisplay = (event) => { if (this.props.onReadyForDisplay) { this.props.onReadyForDisplay(event.nativeEvent); } }; _onPlaybackStalled = (event) => { if (this.props.onPlaybackStalled) { this.props.onPlaybackStalled(event.nativeEvent); } }; _onPlaybackResume = (event) => { if (this.props.onPlaybackResume) { this.props.onPlaybackResume(event.nativeEvent); } }; _onPlaybackRateChange = (event) => { if (this.state.showPoster && event.nativeEvent.playbackRate !== 0 && !this.props.audioOnly) { this.setState({showPoster: false}); } if (this.props.onPlaybackRateChange) { this.props.onPlaybackRateChange(event.nativeEvent); } }; _onExternalPlaybackChange = (event) => { if (this.props.onExternalPlaybackChange) { this.props.onExternalPlaybackChange(event.nativeEvent); } } _onAudioBecomingNoisy = () => { if (this.props.onAudioBecomingNoisy) { this.props.onAudioBecomingNoisy(); } }; _onAudioFocusChanged = (event) => { if (this.props.onAudioFocusChanged) { this.props.onAudioFocusChanged(event.nativeEvent); } }; _onBuffer = (event) => { if (this.props.onBuffer) { this.props.onBuffer(event.nativeEvent); } }; render() { const resizeMode = this.props.resizeMode; const source = resolveAssetSource(this.props.source) || {}; let uri = source.uri || ''; if (uri && uri.match(/^\//)) { uri = `file://${uri}`; } const isNetwork = !!(uri && uri.match(/^https?:/)); const isAsset = !!(uri && uri.match(/^(assets-library|ipod-library|file|content|ms-appx|ms-appdata):/)); let nativeResizeMode; if (resizeMode === VideoResizeMode.stretch) { nativeResizeMode = NativeModules.UIManager.RCTVideo.Constants.ScaleToFill; } else if (resizeMode === VideoResizeMode.contain) { nativeResizeMode = NativeModules.UIManager.RCTVideo.Constants.ScaleAspectFit; } else if (resizeMode === VideoResizeMode.cover) { nativeResizeMode = NativeModules.UIManager.RCTVideo.Constants.ScaleAspectFill; } else { nativeResizeMode = NativeModules.UIManager.RCTVideo.Constants.ScaleNone; } const nativeProps = Object.assign({}, this.props); Object.assign(nativeProps, { style: [styles.base, nativeProps.style], resizeMode: nativeResizeMode, src: { uri, isNetwork, isAsset, type: source.type || '', mainVer: source.mainVer || 0, patchVer: source.patchVer || 0, requestHeaders: source.headers ? this.stringsOnlyObject(source.headers) : {} }, onVideoLoadStart: this._onLoadStart, onVideoLoad: this._onLoad, onVideoError: this._onError, onVideoProgress: this._onProgress, onVideoSeek: this._onSeek, onVideoEnd: this._onEnd, onVideoBuffer: this._onBuffer, onTimedMetadata: this._onTimedMetadata, onVideoAudioBecomingNoisy: this._onAudioBecomingNoisy, onVideoExternalPlaybackChange: this._onExternalPlaybackChange, onVideoFullscreenPlayerWillPresent: this._onFullscreenPlayerWillPresent, onVideoFullscreenPlayerDidPresent: this._onFullscreenPlayerDidPresent, onVideoFullscreenPlayerWillDismiss: this._onFullscreenPlayerWillDismiss, onVideoFullscreenPlayerDidDismiss: this._onFullscreenPlayerDidDismiss, onReadyForDisplay: this._onReadyForDisplay, onPlaybackStalled: this._onPlaybackStalled, onPlaybackResume: this._onPlaybackResume, onPlaybackRateChange: this._onPlaybackRateChange, onAudioFocusChanged: this._onAudioFocusChanged, onAudioBecomingNoisy: this._onAudioBecomingNoisy, }); const posterStyle = { ...StyleSheet.absoluteFillObject, resizeMode: this.props.posterResizeMode || 'contain', }; return ( {this.props.poster && this.state.showPoster && ( )} ); } } Video.filterTypes = ['Normal', 'Country', 'Winter', 'Black N White', 'Sunrise', 'Artistic']; Video.propTypes = { filter: PropTypes.oneOf(Video.filterTypes), /* Native only */ src: PropTypes.object, seek: PropTypes.oneOfType([ PropTypes.number, PropTypes.object ]), fullscreen: PropTypes.bool, onVideoLoadStart: PropTypes.func, onVideoLoad: PropTypes.func, onVideoBuffer: PropTypes.func, onVideoError: PropTypes.func, onVideoProgress: PropTypes.func, onVideoSeek: PropTypes.func, onVideoEnd: PropTypes.func, onTimedMetadata: PropTypes.func, onVideoAudioBecomingNoisy: PropTypes.func, onVideoExternalPlaybackChange: PropTypes.func, onVideoFullscreenPlayerWillPresent: PropTypes.func, onVideoFullscreenPlayerDidPresent: PropTypes.func, onVideoFullscreenPlayerWillDismiss: PropTypes.func, onVideoFullscreenPlayerDidDismiss: PropTypes.func, /* Wrapper component */ source: PropTypes.oneOfType([ PropTypes.shape({ uri: PropTypes.string }), // Opaque type returned by require('./video.mp4') PropTypes.number ]), resizeMode: PropTypes.string, poster: PropTypes.string, posterResizeMode: Image.propTypes.resizeMode, repeat: PropTypes.bool, allowsExternalPlayback: PropTypes.bool, selectedAudioTrack: PropTypes.shape({ type: PropTypes.string.isRequired, value: PropTypes.oneOfType([ PropTypes.string, PropTypes.number ]) }), selectedTextTrack: PropTypes.shape({ type: PropTypes.string.isRequired, value: PropTypes.oneOfType([ PropTypes.string, PropTypes.number ]) }), textTracks: PropTypes.arrayOf( PropTypes.shape({ title: PropTypes.string, uri: PropTypes.string.isRequired, type: PropTypes.oneOf([ TextTrackType.SRT, TextTrackType.TTML, TextTrackType.VTT, ]), language: PropTypes.string.isRequired }) ), paused: PropTypes.bool, muted: PropTypes.bool, volume: PropTypes.number, bufferConfig: PropTypes.shape({ minBufferMs: PropTypes.number, maxBufferMs: PropTypes.number, bufferForPlaybackMs: PropTypes.number, bufferForPlaybackAfterRebufferMs: PropTypes.number, }), stereoPan: PropTypes.number, rate: PropTypes.number, playInBackground: PropTypes.bool, playWhenInactive: PropTypes.bool, ignoreSilentSwitch: PropTypes.oneOf(['ignore', 'obey']), disableFocus: PropTypes.bool, controls: PropTypes.bool, audioOnly: PropTypes.bool, currentTime: PropTypes.number, fullscreenOrientation: PropTypes.oneOf(['all','landscape','portrait']), progressUpdateInterval: PropTypes.number, useTextureView: PropTypes.bool, onLoadStart: PropTypes.func, onLoad: PropTypes.func, onBuffer: PropTypes.func, onError: PropTypes.func, onProgress: PropTypes.func, onSeek: PropTypes.func, onEnd: PropTypes.func, onFullscreenPlayerWillPresent: PropTypes.func, onFullscreenPlayerDidPresent: PropTypes.func, onFullscreenPlayerWillDismiss: PropTypes.func, onFullscreenPlayerDidDismiss: PropTypes.func, onReadyForDisplay: PropTypes.func, onPlaybackStalled: PropTypes.func, onPlaybackResume: PropTypes.func, onPlaybackRateChange: PropTypes.func, onAudioFocusChanged: PropTypes.func, onAudioBecomingNoisy: PropTypes.func, onExternalPlaybackChange: PropTypes.func, /* Required by react-native */ scaleX: PropTypes.number, scaleY: PropTypes.number, translateX: PropTypes.number, translateY: PropTypes.number, rotation: PropTypes.number, ...ViewPropTypes, }; const RCTVideo = requireNativeComponent('RCTVideo', Video, { nativeOnly: { src: true, seek: true, fullscreen: true, }, });