Working fullscreen view

master
Håvar Aambø Fosstveit 2018-06-22 11:45:02 +02:00
parent fa94a62d9d
commit 3b03b24cc4
13 changed files with 311 additions and 23 deletions

View File

@ -0,0 +1,91 @@
import React from 'react';
import { connect } from 'react-redux';
import PropTypes from 'prop-types';
import classnames from 'classnames';
import * as appPropTypes from './appPropTypes';
import * as stateActions from '../redux/stateActions';
import FullView from './FullView';
const FullScreenView = (props) =>
{
const {
advancedMode,
consumer,
toggleConsumerFullscreen
} = props;
if (!consumer)
return null;
const consumerVisible = (
Boolean(consumer) &&
!consumer.locallyPaused &&
!consumer.remotelyPaused
);
let consumerProfile;
if (consumer)
consumerProfile = consumer.profile;
return (
<div data-component='FullScreenView'>
{consumerVisible && !consumer.supported ?
<div className='incompatible-video'>
<p>incompatible video</p>
</div>
:null
}
<div className='controls'>
<div
className={classnames('button', 'fullscreen')}
onClick={(e) =>
{
e.stopPropagation();
toggleConsumerFullscreen(consumer);
}}
/>
</div>
<FullView
advancedMode={advancedMode}
videoTrack={consumer ? consumer.track : null}
videoVisible={consumerVisible}
videoProfile={consumerProfile}
/>
</div>
);
};
FullScreenView.propTypes =
{
advancedMode : PropTypes.bool,
consumer : appPropTypes.Consumer,
toggleConsumerFullscreen : PropTypes.func.isRequired
};
const mapStateToProps = (state) =>
{
return {
consumer : state.consumers[state.room.fullScreenConsumer]
};
};
const mapDispatchToProps = (dispatch) =>
{
return {
toggleConsumerFullscreen : (consumer) =>
{
if (consumer)
dispatch(stateActions.toggleConsumerFullscreen(consumer.id));
}
};
};
const FullScreenViewContainer = connect(
mapStateToProps,
mapDispatchToProps
)(FullScreenView);
export default FullScreenViewContainer;

View File

@ -22,7 +22,7 @@ export default class FullView extends React.Component
} = this.props; } = this.props;
return ( return (
<div data-component='ScreenView'> <div data-component='FullView'>
<video <video
ref='video' ref='video'
className={classnames({ className={classnames({

View File

@ -78,11 +78,6 @@ class Me extends React.Component
if (!me.displayNameSet) if (!me.displayNameSet)
tip = 'Click on your name to change it'; tip = 'Click on your name to change it';
const style =
{
'width' : screenProducer ? '40vmin' : ''
};
return ( return (
<div <div
data-component='Me' data-component='Me'
@ -90,7 +85,6 @@ class Me extends React.Component
data-tip={tip} data-tip={tip}
data-tip-disable={!tip} data-tip-disable={!tip}
data-type='dark' data-type='dark'
style={style}
> >
<div className={classnames('view-container', 'webcam')}> <div className={classnames('view-container', 'webcam')}>
{connected ? {connected ?

View File

@ -226,7 +226,8 @@ const mapDispatchToProps = (dispatch) =>
}, },
toggleConsumerFullscreen : (consumer) => toggleConsumerFullscreen : (consumer) =>
{ {
dispatch(stateActions.toggleConsumerFullscreen(consumer)); if (consumer)
dispatch(stateActions.toggleConsumerFullscreen(consumer.id));
} }
}; };
}; };

View File

@ -74,7 +74,6 @@ class Peers extends React.Component
componentWillReceiveProps(nextProps) componentWillReceiveProps(nextProps)
{ {
if (nextProps.videoStreams)
this.updateDimensions(nextProps); this.updateDimensions(nextProps);
} }
@ -83,7 +82,8 @@ class Peers extends React.Component
const { const {
advancedMode, advancedMode,
activeSpeakerName, activeSpeakerName,
peers peers,
toolAreaOpen
} = this.props; } = this.props;
const style = const style =
@ -108,6 +108,7 @@ class Peers extends React.Component
advancedMode={advancedMode} advancedMode={advancedMode}
name={peer.name} name={peer.name}
style={style} style={style}
toolAreaOpen={toolAreaOpen}
/> />
</div> </div>
</Appear> </Appear>
@ -124,7 +125,8 @@ Peers.propTypes =
advancedMode : PropTypes.bool, advancedMode : PropTypes.bool,
peers : PropTypes.arrayOf(appPropTypes.Peer).isRequired, peers : PropTypes.arrayOf(appPropTypes.Peer).isRequired,
videoStreams : PropTypes.any, videoStreams : PropTypes.any,
activeSpeakerName : PropTypes.string activeSpeakerName : PropTypes.string,
toolAreaOpen : PropTypes.bool
}; };
const mapStateToProps = (state) => const mapStateToProps = (state) =>
@ -140,7 +142,8 @@ const mapStateToProps = (state) =>
return { return {
peers : peersArray, peers : peersArray,
videoStreams : videoStreams, videoStreams : videoStreams,
activeSpeakerName : state.room.activeSpeakerName activeSpeakerName : state.room.activeSpeakerName,
toolAreaOpen : state.toolarea.toolAreaOpen
}; };
}; };

View File

@ -12,6 +12,7 @@ import Peers from './Peers';
import Notifications from './Notifications'; import Notifications from './Notifications';
import ToolAreaButton from './ToolArea/ToolAreaButton'; import ToolAreaButton from './ToolArea/ToolAreaButton';
import ToolArea from './ToolArea/ToolArea'; import ToolArea from './ToolArea/ToolArea';
import FullScreenView from './FullScreenView';
class Room extends React.Component class Room extends React.Component
{ {
@ -59,6 +60,7 @@ class Room extends React.Component
return ( return (
<Appear duration={300}> <Appear duration={300}>
<div data-component='Room'> <div data-component='Room'>
<FullScreenView advancedMode={room.advancedMode} />
<div <div
className='room-wrapper' className='room-wrapper'
style={{ style={{

View File

@ -4,7 +4,8 @@ const initialState =
state : 'new', // new/connecting/connected/disconnected/closed, state : 'new', // new/connecting/connected/disconnected/closed,
activeSpeakerName : null, activeSpeakerName : null,
showSettings : false, showSettings : false,
advancedMode : false advancedMode : false,
fullScreenConsumer : null // ConsumerID
}; };
const room = (state = initialState, action) => const room = (state = initialState, action) =>
@ -49,6 +50,14 @@ const room = (state = initialState, action) =>
return { ...state, advancedMode }; return { ...state, advancedMode };
} }
case 'TOGGLE_FULLSCREEN_CONSUMER':
{
const { consumerId } = action.payload;
const currentConsumer = state.fullScreenConsumer;
return { ...state, fullScreenConsumer: currentConsumer ? null : consumerId };
}
default: default:
return state; return state;
} }

View File

@ -361,6 +361,14 @@ export const toggleChat = () =>
}; };
}; };
export const toggleConsumerFullscreen = (consumerId) =>
{
return {
type : 'TOGGLE_FULLSCREEN_CONSUMER',
payload : { consumerId }
};
};
export const increaseBadge = () => export const increaseBadge = () =>
{ {
return { return {

View File

@ -7,7 +7,7 @@
max-width: 300px; max-width: 300px;
right: 0; right: 0;
width: 90vw; width: 90vw;
z-index: 9999; z-index: 100;
> .launcher { > .launcher {
align-self: flex-end; align-self: flex-end;

View File

@ -0,0 +1,75 @@
[data-component='FullScreenView'] {
position: absolute;
top: 0;
left: 0;
height: 100%;
width: 100%;
z-index: 200;
> .controls {
position: absolute;
z-index: 201;
right: 0;
top: 0;
display: flex;
flex-direction:; row;
justify-content: flex-start;
align-items: center;
padding: 0.4vmin;
> .button {
flex: 0 0 auto;
margin: 0.2vmin;
border-radius: 2px;
background-position: center;
background-size: 75%;
background-repeat: no-repeat;
background-color: rgba(#000, 0.5);
cursor: pointer;
transition-property: opacity, background-color;
transition-duration: 0.15s;
+desktop() {
width: 5vmin;
height: 5vmin;
opacity: 0.85;
&:hover {
opacity: 1;
}
}
+mobile() {
width: 5vmin;
height: 5vmin;
}
&.fullscreen {
background-image: url('/resources/images/icon_fullscreen_exit_black.svg');
background-color: rgba(#fff, 0.7);
}
}
}
.incompatible-video {
position: absolute;
z-index: 2
top: 0;
bottom: 0;
left: 0;
right: 0;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
> p {
padding: 6px 12px;
border-radius: 6px;
user-select: none;
pointer-events: none;
font-size: 15px;
color: rgba(#fff, 0.55);
}
}
}

View File

@ -0,0 +1,105 @@
[data-component='FullView'] {
position: relative;
flex: 100 100 auto;
height: 100%;
width: 100%;
display: flex;
flex-direction: column;
overflow: hidden;
background-color: rgba(#2a4b58, 0.9);
background-image: url('/resources/images/buddy.svg');
background-position: bottom;
background-size: auto 85%;
background-repeat: no-repeat;
> .info {
$backgroundTint = #000;
position: absolute;
z-index: 5
top: 0.6vmin;
left: 0.6vmin;
bottom: 0;
right: 0;
display: flex;
flex-direction: column;
justify-content: space-between;
> .media {
flex: 0 0 auto;
display: flex;
flex-direction: row;
> .box {
padding: 0.4vmin;
border-radius: 2px;
background-color: rgba(#000, 0.25);
> p {
user-select: none;
pointer-events: none;
margin-bottom: 2px;
color: rgba(#fff, 0.7);
font-size: 10px;
&:last-child {
margin-bottom: 0;
}
}
}
}
}
> video {
flex: 100 100 auto;
height: 100%;
width: 100%;
object-fit: contain;
user-select: none;
transition-property: opacity;
transition-duration: .15s;
background-color: rgba(#000, 0.75);
&.hidden {
opacity: 0;
transition-duration: 0s;
}
&.loading {
filter: blur(5px);
}
}
> .spinner-container {
position: absolute;
top: 0
bottom: 0;
left: 0;
right: 0;
background-color: rgba(#000, 0.75);
.react-spinner {
position: relative;
width: 48px;
height: 48px;
top: 50%;
left: 50%;
.react-spinner_bar {
position: absolute;
width: 20%;
height: 7.8%;
top: -3.9%;
left: -10%;
animation: PeerView-spinner 1.2s linear infinite;
border-radius: 5px;
background-color: rgba(#fff, 0.5);
}
}
}
}
@keyframes FullView-spinner {
0% { opacity: 1; }
100% { opacity: 0.15; }
}

View File

@ -1,15 +1,13 @@
[data-component='Me'] { [data-component='Me'] {
flex: 100 100 auto; flex: 100 100 auto;
position: relative; position: relative;
height: 15vmin;
width: 20vmin;
flex-direction: row; flex-direction: row;
display: flex; display: flex;
> .view-container { > .view-container {
position: relative; position: relative;
height: 100%; width: 20vmin;
width: 100%; height: 15vmin;
&.webcam { &.webcam {
order: 2; order: 2;

View File

@ -46,6 +46,8 @@ body {
@import './components/Settings'; @import './components/Settings';
@import './components/ToolArea'; @import './components/ToolArea';
@import './components/ParticipantList'; @import './components/ParticipantList';
@import './components/FullScreenView';
@import './components/FullView';
} }
// Hack to detect in JS the current media query // Hack to detect in JS the current media query