Placed icon and begun work with code.

master
Håvar Aambø Fosstveit 2018-06-21 15:25:40 +02:00
parent 81fcd136f4
commit fa94a62d9d
9 changed files with 447 additions and 232 deletions

View File

@ -604,7 +604,7 @@ export default class RoomClient
{ {
for (const consumer of peer.consumers) for (const consumer of peer.consumers)
{ {
if (consumer.kind !== 'audio') if (consumer.appData.source !== 'mic')
continue; continue;
consumer.pause('mute-audio'); consumer.pause('mute-audio');
@ -640,7 +640,7 @@ export default class RoomClient
{ {
for (const consumer of peer.consumers) for (const consumer of peer.consumers)
{ {
if (consumer.kind !== 'audio' || !consumer.supported) if (consumer.appData.source !== 'mic' || !consumer.supported)
continue; continue;
consumer.resume(); consumer.resume();
@ -676,7 +676,7 @@ export default class RoomClient
{ {
for (const consumer of peer.consumers) for (const consumer of peer.consumers)
{ {
if (consumer.kind !== 'video') if (consumer.appData.source !== 'webcam')
continue; continue;
consumer.pause('pause-video'); consumer.pause('pause-video');
@ -712,7 +712,7 @@ export default class RoomClient
{ {
for (const consumer of peer.consumers) for (const consumer of peer.consumers)
{ {
if (consumer.kind !== 'video' || !consumer.supported) if (consumer.appData.source !== 'webcam' || !consumer.supported)
continue; continue;
consumer.resume(); consumer.resume();

View File

@ -0,0 +1,90 @@
import React from 'react';
import PropTypes from 'prop-types';
import classnames from 'classnames';
import Spinner from 'react-spinner';
export default class FullView extends React.Component
{
constructor(props)
{
super(props);
// Latest received video track.
// @type {MediaStreamTrack}
this._videoTrack = null;
}
render()
{
const {
videoVisible,
videoProfile
} = this.props;
return (
<div data-component='ScreenView'>
<video
ref='video'
className={classnames({
hidden : !videoVisible,
loading : videoProfile === 'none'
})}
autoPlay
muted={Boolean(true)}
/>
{videoProfile === 'none' ?
<div className='spinner-container'>
<Spinner />
</div>
:null
}
</div>
);
}
componentDidMount()
{
const { videoTrack } = this.props;
this._setTracks(videoTrack);
}
componentWillReceiveProps(nextProps)
{
const { videoTrack } = nextProps;
this._setTracks(videoTrack);
}
_setTracks(videoTrack)
{
if (this._videoTrack === videoTrack)
return;
this._videoTrack = videoTrack;
const { video } = this.refs;
if (videoTrack)
{
const stream = new MediaStream;
if (videoTrack)
stream.addTrack(videoTrack);
video.srcObject = stream;
}
else
{
video.srcObject = null;
}
}
}
FullView.propTypes =
{
videoTrack : PropTypes.any,
videoVisible : PropTypes.bool,
videoProfile : PropTypes.string
};

View File

@ -78,6 +78,11 @@ 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'
@ -85,52 +90,57 @@ 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}
> >
{connected ? <div className={classnames('view-container', 'webcam')}>
<div className='controls'> {connected ?
<div <div className='controls'>
className={classnames('button', 'mic', micState, { <div
disabled : me.audioInProgress className={classnames('button', 'mic', micState, {
})} disabled : me.audioInProgress
onClick={() => })}
{ onClick={() =>
micState === 'on' ? onMuteMic() : onUnmuteMic(); {
}} micState === 'on' ? onMuteMic() : onUnmuteMic();
/> }}
/>
<div <div
className={classnames('button', 'webcam', webcamState, { className={classnames('button', 'webcam', webcamState, {
disabled : me.webcamInProgress disabled : me.webcamInProgress
})} })}
onClick={() => onClick={() =>
{ {
webcamState === 'on' ? onDisableWebcam() : onEnableWebcam(); webcamState === 'on' ? onDisableWebcam() : onEnableWebcam();
}} }}
/> />
</div> </div>
:null :null
} }
<PeerView <PeerView
isMe
advancedMode={advancedMode}
peer={me}
audioTrack={micProducer ? micProducer.track : null}
videoTrack={webcamProducer ? webcamProducer.track : null}
videoVisible={videoVisible}
audioCodec={micProducer ? micProducer.codec : null}
videoCodec={webcamProducer ? webcamProducer.codec : null}
onChangeDisplayName={(displayName) => onChangeDisplayName(displayName)}
/>
{screenProducer ?
<ScreenView
isMe isMe
advancedMode={advancedMode} advancedMode={advancedMode}
screenTrack={screenProducer ? screenProducer.track : null} peer={me}
screenVisible={screenVisible} audioTrack={micProducer ? micProducer.track : null}
screenCodec={screenProducer ? screenProducer.codec : null} videoTrack={webcamProducer ? webcamProducer.track : null}
videoVisible={videoVisible}
audioCodec={micProducer ? micProducer.codec : null}
videoCodec={webcamProducer ? webcamProducer.codec : null}
onChangeDisplayName={(displayName) => onChangeDisplayName(displayName)}
/> />
</div>
{screenProducer ?
<div className={classnames('view-container', 'screen')}>
<ScreenView
isMe
advancedMode={advancedMode}
screenTrack={screenProducer ? screenProducer.track : null}
screenVisible={screenVisible}
screenCodec={screenProducer ? screenProducer.codec : null}
/>
</div>
:null :null
} }

View File

@ -4,6 +4,7 @@ import PropTypes from 'prop-types';
import classnames from 'classnames'; import classnames from 'classnames';
import * as appPropTypes from './appPropTypes'; import * as appPropTypes from './appPropTypes';
import * as requestActions from '../redux/requestActions'; import * as requestActions from '../redux/requestActions';
import * as stateActions from '../redux/stateActions';
import PeerView from './PeerView'; import PeerView from './PeerView';
import ScreenView from './ScreenView'; import ScreenView from './ScreenView';
@ -19,6 +20,9 @@ const Peer = (props) =>
onUnmuteMic, onUnmuteMic,
onDisableWebcam, onDisableWebcam,
onEnableWebcam, onEnableWebcam,
onDisableScreen,
onEnableScreen,
toggleConsumerFullscreen,
style style
} = props; } = props;
@ -57,35 +61,6 @@ const Peer = (props) =>
screen : screenConsumer screen : screenConsumer
})} })}
> >
<div className='controls'>
<div
className={classnames('button', 'mic', {
on : micEnabled,
off : !micEnabled,
disabled : peer.peerAudioInProgress
})}
onClick={(e) =>
{
e.stopPropagation();
micEnabled ? onMuteMic(peer.name) : onUnmuteMic(peer.name);
}}
/>
<div
className={classnames('button', 'webcam', {
on : videoVisible,
off : !videoVisible,
disabled : peer.peerVideoInProgress
})}
onClick={(e) =>
{
e.stopPropagation();
videoVisible ?
onDisableWebcam(peer.name) : onEnableWebcam(peer.name);
}}
/>
</div>
{videoVisible && !webcamConsumer.supported ? {videoVisible && !webcamConsumer.supported ?
<div className='incompatible-video'> <div className='incompatible-video'>
<p>incompatible video</p> <p>incompatible video</p>
@ -94,6 +69,43 @@ const Peer = (props) =>
} }
<div className={classnames('view-container', 'webcam')} style={style}> <div className={classnames('view-container', 'webcam')} style={style}>
<div className='controls'>
<div
className={classnames('button', 'mic', {
on : micEnabled,
off : !micEnabled,
disabled : peer.peerAudioInProgress
})}
onClick={(e) =>
{
e.stopPropagation();
micEnabled ? onMuteMic(peer.name) : onUnmuteMic(peer.name);
}}
/>
<div
className={classnames('button', 'webcam', {
on : videoVisible,
off : !videoVisible,
disabled : peer.peerVideoInProgress
})}
onClick={(e) =>
{
e.stopPropagation();
videoVisible ?
onDisableWebcam(peer.name) : onEnableWebcam(peer.name);
}}
/>
<div
className={classnames('button', 'fullscreen')}
onClick={(e) =>
{
e.stopPropagation();
toggleConsumerFullscreen(webcamConsumer);
}}
/>
</div>
<PeerView <PeerView
advancedMode={advancedMode} advancedMode={advancedMode}
peer={peer} peer={peer}
@ -108,6 +120,30 @@ const Peer = (props) =>
{screenConsumer ? {screenConsumer ?
<div className={classnames('view-container', 'screen')} style={style}> <div className={classnames('view-container', 'screen')} style={style}>
<div className='controls'>
<div
className={classnames('button', 'screen', {
on : screenVisible,
off : !screenVisible,
disabled : peer.peerScreenInProgress
})}
onClick={(e) =>
{
e.stopPropagation();
screenVisible ?
onDisableScreen(peer.name) : onEnableScreen(peer.name);
}}
/>
<div
className={classnames('button', 'fullscreen')}
onClick={(e) =>
{
e.stopPropagation();
toggleConsumerFullscreen(screenConsumer);
}}
/>
</div>
<ScreenView <ScreenView
advancedMode={advancedMode} advancedMode={advancedMode}
screenTrack={screenConsumer ? screenConsumer.track : null} screenTrack={screenConsumer ? screenConsumer.track : null}
@ -124,17 +160,20 @@ const Peer = (props) =>
Peer.propTypes = Peer.propTypes =
{ {
advancedMode : PropTypes.bool, advancedMode : PropTypes.bool,
peer : appPropTypes.Peer.isRequired, peer : appPropTypes.Peer.isRequired,
micConsumer : appPropTypes.Consumer, micConsumer : appPropTypes.Consumer,
webcamConsumer : appPropTypes.Consumer, webcamConsumer : appPropTypes.Consumer,
screenConsumer : appPropTypes.Consumer, screenConsumer : appPropTypes.Consumer,
onMuteMic : PropTypes.func.isRequired, onMuteMic : PropTypes.func.isRequired,
onUnmuteMic : PropTypes.func.isRequired, onUnmuteMic : PropTypes.func.isRequired,
onEnableWebcam : PropTypes.func.isRequired, onEnableWebcam : PropTypes.func.isRequired,
onDisableWebcam : PropTypes.func.isRequired, onDisableWebcam : PropTypes.func.isRequired,
streamDimensions : PropTypes.object, streamDimensions : PropTypes.object,
style : PropTypes.object style : PropTypes.object,
onEnableScreen : PropTypes.func.isRequired,
onDisableScreen : PropTypes.func.isRequired,
toggleConsumerFullscreen : PropTypes.func.isRequired
}; };
const mapStateToProps = (state, { name }) => const mapStateToProps = (state, { name }) =>
@ -176,6 +215,18 @@ const mapDispatchToProps = (dispatch) =>
onDisableWebcam : (peerName) => onDisableWebcam : (peerName) =>
{ {
dispatch(requestActions.pausePeerVideo(peerName)); dispatch(requestActions.pausePeerVideo(peerName));
},
onEnableScreen : (peerName) =>
{
dispatch(requestActions.resumePeerScreen(peerName));
},
onDisableScreen : (peerName) =>
{
dispatch(requestActions.pausePeerScreen(peerName));
},
toggleConsumerFullscreen : (consumer) =>
{
dispatch(stateActions.toggleConsumerFullscreen(consumer));
} }
}; };
}; };

View File

@ -0,0 +1,4 @@
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24">
<path d="M0 0h24v24H0z" fill="none"/>
<path d="M7 14H5v5h5v-2H7v-3zm-2-4h2V7h3V5H5v5zm12 7h-3v2h5v-5h-2v3zM14 5v2h3v3h2V5h-5z"/>
</svg>

After

Width:  |  Height:  |  Size: 228 B

View File

@ -0,0 +1,4 @@
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24">
<path d="M0 0h24v24H0z" fill="none"/>
<path d="M5 16h3v3h2v-5H5v2zm3-8H5v2h5V5H8v3zm6 11h2v-3h3v-2h-5v5zm2-11V5h-2v5h5V8h-3z"/>
</svg>

After

Width:  |  Height:  |  Size: 227 B

View File

@ -1,87 +1,123 @@
[data-component='Me'] { [data-component='Me'] {
flex: 100 100 auto;
position: relative; position: relative;
height: 100%; height: 15vmin;
width: 100%; width: 20vmin;
flex-direction: row;
display: flex;
> .controls { > .view-container {
position: absolute; position: relative;
z-index: 10 height: 100%;
top: 0; width: 100%;
left: 0;
right: 0;
display: flex;
flex-direction:; row;
justify-content: flex-end;
align-items: center;
padding: 0.4vmin;
> .button { &.webcam {
flex: 0 0 auto; order: 2;
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() { &.screen {
width: 24px; order: 1;
height: 24px; }
opacity: 0.85;
&:hover { > .controls {
opacity: 1; position: absolute;
} z-index: 10;
} right: 0;
top: 0;
display: flex;
flex-direction:; row;
justify-content: flex-start;
align-items: center;
padding: 0.4vmin;
+mobile() { > .button {
width: 22px; flex: 0 0 auto;
height: 22px; 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;
&.unsupported { +desktop() {
pointer-events: none; width: 24px;
} height: 24px;
opacity: 0.85;
&.disabled { &:hover {
pointer-events: none; opacity: 1;
opacity: 0.5; }
}
&.on {
background-color: rgba(#fff, 0.7);
}
&.mic {
&.on {
background-image: url('/resources/images/icon_mic_black_on.svg');
} }
&.off { +mobile() {
background-image: url('/resources/images/icon_mic_white_off.svg'); width: 22px;
background-color: rgba(#d42241, 0.7); height: 22px;
} }
&.unsupported { &.unsupported {
background-image: url('/resources/images/icon_mic_white_unsupported.svg'); pointer-events: none;
}
&.disabled {
pointer-events: none;
opacity: 0.5;
} }
}
&.webcam {
&.on { &.on {
background-image: url('/resources/images/icon_webcam_black_on.svg'); background-color: rgba(#fff, 0.7);
} }
&.off { &.mic {
background-image: url('/resources/images/icon_remote_webcam_white_off.svg'); &.on {
background-color: rgba(#d42241, 0.7); background-image: url('/resources/images/icon_mic_black_on.svg');
}
&.off {
background-image: url('/resources/images/icon_remote_mic_white_off.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/icon_mic_white_unsupported.svg');
}
} }
&.unsupported { &.webcam {
background-image: url('/resources/images/icon_webcam_white_unsupported.svg'); &.on {
background-image: url('/resources/images/icon_webcam_black_on.svg');
}
&.off {
background-image: url('/resources/images/icon_remote_webcam_white_off.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/icon_webcam_white_unsupported.svg');
}
}
&.screen {
&.on {
background-image: url('/resources/images/share-screen-black.svg');
}
&.off {
background-image: url('/resources/images/no-share-screen-white.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/no-share-screen-white.svg');
}
}
&.fullscreen {
background-image: url('/resources/images/icon_fullscreen_black.svg');
background-color: rgba(#fff, 0.7);
} }
} }
} }

View File

@ -13,89 +13,6 @@
&:not(.screen) { &:not(.screen) {
} }
> .controls {
position: absolute;
z-index: 10;
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: 24px;
height: 24px;
opacity: 0.85;
&:hover {
opacity: 1;
}
}
+mobile() {
width: 22px;
height: 22px;
}
&.unsupported {
pointer-events: none;
}
&.disabled {
pointer-events: none;
opacity: 0.5;
}
&.on {
background-color: rgba(#fff, 0.7);
}
&.mic {
&.on {
background-image: url('/resources/images/icon_mic_black_on.svg');
}
&.off {
background-image: url('/resources/images/icon_remote_mic_white_off.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/icon_mic_white_unsupported.svg');
}
}
&.webcam {
&.on {
background-image: url('/resources/images/icon_webcam_black_on.svg');
}
&.off {
background-image: url('/resources/images/icon_remote_webcam_white_off.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/icon_webcam_white_unsupported.svg');
}
}
}
}
+mobile() { +mobile() {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@ -104,6 +21,8 @@
} }
> .view-container { > .view-container {
position: relative;
&.webcam { &.webcam {
order: 2; order: 2;
} }
@ -111,6 +30,109 @@
&.screen { &.screen {
order: 1; order: 1;
} }
> .controls {
position: absolute;
z-index: 10;
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: 24px;
height: 24px;
opacity: 0.85;
&:hover {
opacity: 1;
}
}
+mobile() {
width: 22px;
height: 22px;
}
&.unsupported {
pointer-events: none;
}
&.disabled {
pointer-events: none;
opacity: 0.5;
}
&.on {
background-color: rgba(#fff, 0.7);
}
&.mic {
&.on {
background-image: url('/resources/images/icon_mic_black_on.svg');
}
&.off {
background-image: url('/resources/images/icon_remote_mic_white_off.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/icon_mic_white_unsupported.svg');
}
}
&.webcam {
&.on {
background-image: url('/resources/images/icon_webcam_black_on.svg');
}
&.off {
background-image: url('/resources/images/icon_remote_webcam_white_off.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/icon_webcam_white_unsupported.svg');
}
}
&.screen {
&.on {
background-image: url('/resources/images/share-screen-black.svg');
}
&.off {
background-image: url('/resources/images/no-share-screen-white.svg');
background-color: rgba(#d42241, 0.7);
}
&.unsupported {
background-image: url('/resources/images/no-share-screen-white.svg');
}
}
&.fullscreen {
background-image: url('/resources/images/icon_fullscreen_black.svg');
background-color: rgba(#fff, 0.7);
}
}
}
} }
.incompatible-video { .incompatible-video {

View File

@ -153,8 +153,6 @@
} }
+desktop() { +desktop() {
height: 200px;
width: 235px;
bottom: 20px; bottom: 20px;
left: 20px; left: 20px;
border: 1px solid rgba(#fff, 0.15); border: 1px solid rgba(#fff, 0.15);