164 lines
4.7 KiB
JavaScript
164 lines
4.7 KiB
JavaScript
|
import React from 'react';
|
||
|
import { Animated } from 'react-native';
|
||
|
import { connect } from 'react-redux';
|
||
|
|
||
|
import { AbstractVideoTrack } from '../AbstractVideoTrack';
|
||
|
import { styles } from './styles';
|
||
|
|
||
|
/**
|
||
|
* Component that renders video element for a specified video track.
|
||
|
*
|
||
|
* @extends AbstractVideoTrack
|
||
|
*/
|
||
|
class VideoTrack extends AbstractVideoTrack {
|
||
|
/**
|
||
|
* Initializes a new VideoTrack instance.
|
||
|
*
|
||
|
* @param {Object} props - The read-only properties with which the new
|
||
|
* instance is to be initialized.
|
||
|
*/
|
||
|
constructor(props) {
|
||
|
super(props);
|
||
|
|
||
|
/**
|
||
|
* Reference to currently running animation if any.
|
||
|
*
|
||
|
* @private
|
||
|
*/
|
||
|
this._animation = null;
|
||
|
|
||
|
/**
|
||
|
* Extend Component's state with additional animation-related vars.
|
||
|
*
|
||
|
* @type {Object}
|
||
|
*/
|
||
|
this.state = {
|
||
|
...this.state,
|
||
|
fade: new Animated.Value(1),
|
||
|
flip: new Animated.Value(1)
|
||
|
};
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Renders video element with animation.
|
||
|
*
|
||
|
* @override
|
||
|
* @returns {ReactElement}
|
||
|
*/
|
||
|
render() {
|
||
|
return (
|
||
|
<Animated.View
|
||
|
style = { [ styles.video, this._getAnimationStyles() ] }>
|
||
|
{ super.render() }
|
||
|
</Animated.View>
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Animates setting a new video track to be rendered by this instance.
|
||
|
*
|
||
|
* @param {Track} oldValue - The old video track rendered by this instance.
|
||
|
* @param {Track} newValue - The new video track to be rendered by this
|
||
|
* instance.
|
||
|
* @private
|
||
|
* @returns {Promise}
|
||
|
*/
|
||
|
_animateSetVideoTrack(oldValue, newValue) {
|
||
|
// If we're in the middle of an animation and a new animation is about
|
||
|
// to start, stop the previous one first.
|
||
|
if (this._animation) {
|
||
|
this._animation.stop();
|
||
|
this._animation = null;
|
||
|
this.state.fade.setValue(1);
|
||
|
this.state.flip.setValue(1);
|
||
|
}
|
||
|
|
||
|
// If we're switching between two local video tracks, that actually
|
||
|
// means we're switching local cameras, so we'll use a flip animation.
|
||
|
// Otherwise, we'll use fade animation.
|
||
|
const animation
|
||
|
= oldValue && newValue && oldValue.local && newValue.local
|
||
|
? 'flip'
|
||
|
: 'fade';
|
||
|
|
||
|
return this._animateVideoTrack(animation, 0)
|
||
|
.then(() => {
|
||
|
super._setVideoTrack(newValue);
|
||
|
|
||
|
return this._animateVideoTrack(animation, 1);
|
||
|
})
|
||
|
.catch(() => {
|
||
|
console.log('Animation was stopped');
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Animates the display of the state videoTrack.
|
||
|
*
|
||
|
* @param {string} animatedValue - The name of the state property which
|
||
|
* specifies the Animated.Value to be animated.
|
||
|
* @param {number} toValue - The value to which the specified animatedValue
|
||
|
* is to be animated.
|
||
|
* @private
|
||
|
* @returns {Promise}
|
||
|
*/
|
||
|
_animateVideoTrack(animatedValue, toValue) {
|
||
|
return new Promise((resolve, reject) => {
|
||
|
this._animation
|
||
|
= Animated.timing(this.state[animatedValue], { toValue });
|
||
|
this._animation.start(result => {
|
||
|
this._animation = null;
|
||
|
result.finished ? resolve() : reject();
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns animation styles for Animated.View.
|
||
|
*
|
||
|
* @private
|
||
|
* @returns {Object}
|
||
|
*/
|
||
|
_getAnimationStyles() {
|
||
|
return {
|
||
|
opacity: this.state.fade,
|
||
|
transform: [ {
|
||
|
rotateY: this.state.flip.interpolate({
|
||
|
inputRange: [ 0, 1 ],
|
||
|
outputRange: [ '90deg', '0deg' ]
|
||
|
})
|
||
|
} ]
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// eslint-disable-next-line valid-jsdoc
|
||
|
/**
|
||
|
* @inheritdoc
|
||
|
*
|
||
|
* Animate the setting of the video track to be rendered by this instance.
|
||
|
*/
|
||
|
_setVideoTrack(videoTrack) {
|
||
|
// If JitsiTrack instance didn't change, that means some other track's
|
||
|
// props were changed and we don't need to animate.
|
||
|
const oldValue = this.state.videoTrack;
|
||
|
const oldJitsiTrack = oldValue ? oldValue.jitsiTrack : null;
|
||
|
const newValue = videoTrack;
|
||
|
const newJitsiTrack = newValue ? newValue.jitsiTrack : null;
|
||
|
|
||
|
if (oldJitsiTrack === newJitsiTrack) {
|
||
|
super._setVideoTrack(newValue);
|
||
|
} else {
|
||
|
this._animateSetVideoTrack(oldValue, newValue);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* VideoTrack component's property types.
|
||
|
*
|
||
|
* @static
|
||
|
*/
|
||
|
VideoTrack.propTypes = AbstractVideoTrack.propTypes;
|
||
|
|
||
|
export default connect()(VideoTrack);
|