jiti-meet/react/features/base/media/components/native/VideoTrack.js

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);