jiti-meet/react/features/welcome/components/AbstractWelcomePage.js

218 lines
6.0 KiB
JavaScript
Raw Normal View History

import React, { Component } from 'react';
import { appNavigate } from '../../app';
import { isRoomValid } from '../../base/conference';
import { VideoTrack } from '../../base/media';
import { getLocalVideoTrack } from '../../base/tracks';
import { generateRoomWithoutSeparator } from '../roomnameGenerator';
/**
* Base (abstract) class for container component rendering the welcome page.
*
* @abstract
*/
export class AbstractWelcomePage extends Component {
/**
2017-07-19 21:25:06 +00:00
* {@code AbstractWelcomePage} component's property types.
*
* @static
*/
static propTypes = {
_localVideoTrack: React.PropTypes.object,
_room: React.PropTypes.string,
dispatch: React.PropTypes.func
};
/**
2017-07-19 21:25:06 +00:00
* Initializes a new {@code AbstractWelcomePage} instance.
*
2017-07-19 21:25:06 +00:00
* @param {Object} props - The React {@code Component} props to initialize
* the new {@code AbstractWelcomePage} instance with.
*/
constructor(props) {
super(props);
/**
* Save room name into component's local state.
*
* @type {Object}
2017-07-19 21:25:06 +00:00
* @property {number|null} animateTimeoutId - Identifier of the letter
* animation timeout.
2016-12-16 03:00:06 +00:00
* @property {string} generatedRoomname - Automatically generated
* room name.
* @property {string} room - Room name.
* @property {string} roomPlaceholder - Room placeholder
* that's used as a placeholder for input.
2017-07-19 21:25:06 +00:00
* @property {nubmer|null} updateTimeoutId - Identifier of the timeout
* updating the generated room name.
*/
this.state = {
2016-12-16 03:00:06 +00:00
animateTimeoutId: null,
generatedRoomname: '',
room: '',
roomPlaceholder: '',
updateTimeoutId: null
};
2017-07-19 21:25:06 +00:00
// Bind event handlers so they are only bound once per instance.
2016-12-16 03:00:06 +00:00
this._animateRoomnameChanging
= this._animateRoomnameChanging.bind(this);
this._onJoin = this._onJoin.bind(this);
this._onRoomChange = this._onRoomChange.bind(this);
this._updateRoomname = this._updateRoomname.bind(this);
}
/**
* This method is executed when component receives new properties.
*
* @inheritdoc
* @param {Object} nextProps - New props component will receive.
*/
componentWillReceiveProps(nextProps) {
this.setState({ room: nextProps._room });
}
/**
2016-12-16 03:00:06 +00:00
* This method is executed when method will be unmounted from DOM.
*
2016-12-16 03:00:06 +00:00
* @inheritdoc
*/
2016-12-16 03:00:06 +00:00
componentWillUnmount() {
this._clearTimeouts();
}
/**
2016-12-16 03:00:06 +00:00
* Animates the changing of the room name.
*
2016-12-16 03:00:06 +00:00
* @param {string} word - The part of room name that should be added to
* placeholder.
* @private
* @returns {void}
*/
_animateRoomnameChanging(word) {
let animateTimeoutId = null;
2016-12-16 03:00:06 +00:00
const roomPlaceholder = this.state.roomPlaceholder + word.substr(0, 1);
if (word.length > 1) {
2016-12-16 03:00:06 +00:00
animateTimeoutId
= setTimeout(
2017-06-15 00:40:51 +00:00
() => {
this._animateRoomnameChanging(
word.substring(1, word.length));
},
70);
}
this.setState({
animateTimeoutId,
roomPlaceholder
});
}
2016-12-16 03:00:06 +00:00
/**
* Method that clears timeouts for animations and updates of room name.
*
* @private
* @returns {void}
*/
_clearTimeouts() {
clearTimeout(this.state.animateTimeoutId);
clearTimeout(this.state.updateTimeoutId);
}
/**
* Determines whether the 'Join' button is (to be) disabled i.e. there's no
* valid room name typed into the respective text input field.
*
* @protected
* @returns {boolean} If the 'Join' button is (to be) disabled, true;
* otherwise, false.
*/
_isJoinDisabled() {
return !isRoomValid(this.state.room);
}
/**
* Handles joining. Either by clicking on 'Join' button
* or by pressing 'Enter' in room name input field.
*
* @protected
* @returns {void}
*/
_onJoin() {
2016-12-16 03:00:06 +00:00
const room = this.state.room || this.state.generatedRoomname;
2017-07-19 21:25:06 +00:00
room && this.props.dispatch(appNavigate(room));
}
/**
* Handles 'change' event for the room name text input field.
*
* @param {string} value - The text typed into the respective text input
* field.
* @protected
* @returns {void}
*/
_onRoomChange(value) {
this.setState({ room: value });
}
/**
* Renders a local video if any.
*
* @protected
* @returns {(ReactElement|null)}
*/
_renderLocalVideo() {
return (
<VideoTrack videoTrack = { this.props._localVideoTrack } />
);
}
2016-12-16 03:00:06 +00:00
/**
* Triggers the generation of a new room name and initiates an animation of
* its changing.
*
* @protected
* @returns {void}
*/
_updateRoomname() {
const generatedRoomname = generateRoomWithoutSeparator();
const roomPlaceholder = '';
const updateTimeoutId = setTimeout(this._updateRoomname, 10000);
this._clearTimeouts();
this.setState(
{
generatedRoomname,
roomPlaceholder,
updateTimeoutId
},
() => this._animateRoomnameChanging(generatedRoomname));
}
}
/**
* Selects local video track from tracks in state, local participant and room
* and maps them to component props. It seems it's not possible to 'connect'
* base component and then extend from it. So we export this function in order
* to be used in child classes for 'connect'.
*
* @param {Object} state - Redux state.
* @protected
* @returns {{
* _localVideoTrack: (Track|undefined),
* _room: string
* }}
*/
export function _mapStateToProps(state) {
const conference = state['features/base/conference'];
const tracks = state['features/base/tracks'];
return {
_localVideoTrack: getLocalVideoTrack(tracks),
_room: conference.room
};
}