2014-12-01 21:39:18 +00:00
2015-09-17 16:12:34 +00:00
( function ( $ ) { function findLine ( sdpLines , prefix , substr ) { return findLineInRange ( sdpLines , 0 , - 1 , prefix , substr ) ; }
2014-12-06 17:40:44 +00:00
function findLineInRange ( sdpLines , startLine , endLine , prefix , substr ) { var realEndLine = ( endLine != - 1 ) ? endLine : sdpLines . length ; for ( var i = startLine ; i < realEndLine ; ++ i ) { if ( sdpLines [ i ] . indexOf ( prefix ) === 0 ) { if ( ! substr || sdpLines [ i ] . toLowerCase ( ) . indexOf ( substr . toLowerCase ( ) ) !== - 1 ) { return i ; } } }
return null ; }
function getCodecPayloadType ( sdpLine ) { var pattern = new RegExp ( 'a=rtpmap:(\\d+) \\w+\\/\\d+' ) ; var result = sdpLine . match ( pattern ) ; return ( result && result . length == 2 ) ? result [ 1 ] : null ; }
function setDefaultCodec ( mLine , payload ) { var elements = mLine . split ( ' ' ) ; var newLine = [ ] ; var index = 0 ; for ( var i = 0 ; i < elements . length ; i ++ ) { if ( index === 3 ) { newLine [ index ++ ] = payload ; }
if ( elements [ i ] !== payload ) newLine [ index ++ ] = elements [ i ] ; }
return newLine . join ( ' ' ) ; }
2016-06-08 21:20:18 +00:00
$ . FSRTC = function ( options ) { this . options = $ . extend ( { useVideo : null , useStereo : false , userData : null , localVideo : null , screenShare : false , useCamera : "any" , iceServers : false , videoParams : { } , audioParams : { } , callbacks : { onICEComplete : function ( ) { } , onICE : function ( ) { } , onOfferSDP : function ( ) { } } , } , options ) ; this . audioEnabled = true ; this . videoEnabled = true ; this . mediaData = { SDP : null , profile : { } , candidateList : [ ] } ; this . constraints = { offerToReceiveAudio : this . options . useSpeak === "none" ? false : true , offerToReceiveVideo : this . options . useVideo ? true : false , } ; if ( self . options . useVideo ) { self . options . useVideo . style . display = 'none' ; }
setCompat ( ) ; checkCompat ( ) ; } ; $ . FSRTC . validRes = [ ] ; $ . FSRTC . prototype . useVideo = function ( obj , local ) { var self = this ; if ( obj ) { self . options . useVideo = obj ; self . options . localVideo = local ; self . constraints . offerToReceiveVideo = true ; } else { self . options . useVideo = null ; self . options . localVideo = null ; self . constraints . offerToReceiveVideo = false ; }
2015-02-20 23:19:09 +00:00
if ( self . options . useVideo ) { self . options . useVideo . style . display = 'none' ; } } ; $ . FSRTC . prototype . useStereo = function ( on ) { var self = this ; self . options . useStereo = on ; } ; $ . FSRTC . prototype . stereoHack = function ( sdp ) { var self = this ; if ( ! self . options . useStereo ) { return sdp ; }
2015-09-22 20:10:29 +00:00
var sdpLines = sdp . split ( '\r\n' ) ; var opusIndex = findLine ( sdpLines , 'a=rtpmap' , 'opus/48000' ) , opusPayload ; if ( ! opusIndex ) { return sdp ; } else { opusPayload = getCodecPayloadType ( sdpLines [ opusIndex ] ) ; }
2015-09-22 20:31:09 +00:00
var fmtpLineIndex = findLine ( sdpLines , 'a=fmtp:' + opusPayload . toString ( ) ) ; if ( fmtpLineIndex === null ) { sdpLines [ opusIndex ] = sdpLines [ opusIndex ] + '\r\na=fmtp:' + opusPayload . toString ( ) + " stereo=1; sprop-stereo=1" } else { sdpLines [ fmtpLineIndex ] = sdpLines [ fmtpLineIndex ] . concat ( '; stereo=1; sprop-stereo=1' ) ; }
2016-06-08 21:20:18 +00:00
sdp = sdpLines . join ( '\r\n' ) ; return sdp ; } ; function setCompat ( ) { }
function checkCompat ( ) { return true ; }
2015-03-24 17:49:06 +00:00
function onStreamError ( self , e ) { console . log ( 'There has been a problem retrieving the streams - did you allow access? Check Device Resolution' , e ) ; doCallback ( self , "onError" , e ) ; }
2015-01-24 09:13:44 +00:00
function onStreamSuccess ( self , stream ) { console . log ( "Stream Success" ) ; doCallback ( self , "onStream" , stream ) ; }
2014-12-06 17:40:44 +00:00
function onICE ( self , candidate ) { self . mediaData . candidate = candidate ; self . mediaData . candidateList . push ( self . mediaData . candidate ) ; doCallback ( self , "onICE" ) ; }
function doCallback ( self , func , arg ) { if ( func in self . options . callbacks ) { self . options . callbacks [ func ] ( self , arg ) ; } }
function onICEComplete ( self , candidate ) { console . log ( "ICE Complete" ) ; doCallback ( self , "onICEComplete" ) ; }
function onChannelError ( self , e ) { console . error ( "Channel Error" , e ) ; doCallback ( self , "onError" , e ) ; }
function onICESDP ( self , sdp ) { self . mediaData . SDP = self . stereoHack ( sdp . sdp ) ; console . log ( "ICE SDP" ) ; doCallback ( self , "onICESDP" ) ; }
function onAnswerSDP ( self , sdp ) { self . answer . SDP = self . stereoHack ( sdp . sdp ) ; console . log ( "ICE ANSWER SDP" ) ; doCallback ( self , "onAnswerSDP" , self . answer . SDP ) ; }
function onMessage ( self , msg ) { console . log ( "Message" ) ; doCallback ( self , "onICESDP" , msg ) ; }
2016-06-08 21:20:18 +00:00
FSRTCattachMediaStream = function ( element , stream ) { if ( element && element . id && attachMediaStream ) { attachMediaStream ( element , stream ) ; } else { if ( typeof element . srcObject !== 'undefined' ) { element . srcObject = stream ; } else if ( typeof element . src !== 'undefined' ) { element . src = URL . createObjectURL ( stream ) ; } else { console . error ( 'Error attaching stream to element.' ) ; } } }
2015-02-20 23:19:09 +00:00
function onRemoteStream ( self , stream ) { if ( self . options . useVideo ) { self . options . useVideo . style . display = 'block' ; }
2016-06-08 21:20:18 +00:00
var element = self . options . useAudio ; console . log ( "REMOTE STREAM" , stream , element ) ; FSRTCattachMediaStream ( element , stream ) ; self . options . useAudio . play ( ) ; self . remoteStream = stream ; }
2014-12-06 17:40:44 +00:00
function onOfferSDP ( self , sdp ) { self . mediaData . SDP = self . stereoHack ( sdp . sdp ) ; console . log ( "Offer SDP" ) ; doCallback ( self , "onOfferSDP" ) ; }
2015-09-22 16:21:15 +00:00
$ . FSRTC . prototype . answer = function ( sdp , onSuccess , onError ) { this . peer . addAnswerSDP ( { type : "answer" , sdp : sdp } , onSuccess , onError ) ; } ; $ . FSRTC . prototype . stopPeer = function ( ) { if ( self . peer ) { console . log ( "stopping peer" ) ; self . peer . stop ( ) ; } }
2016-06-08 21:20:18 +00:00
$ . FSRTC . prototype . stop = function ( ) { var self = this ; if ( self . options . useVideo ) { self . options . useVideo . style . display = 'none' ; self . options . useVideo [ 'src' ] = '' ; }
2017-01-26 17:43:45 +00:00
if ( self . localStream ) { if ( typeof self . localStream . stop == 'function' ) { self . localStream . stop ( ) ; } else { if ( self . localStream . active ) { var tracks = self . localStream . getTracks ( ) ; console . log ( tracks ) ; tracks . forEach ( function ( track , index ) { console . log ( track ) ; track . stop ( ) ; } ) } }
2015-09-26 01:00:03 +00:00
self . localStream = null ; }
2016-06-08 21:20:18 +00:00
if ( self . options . localVideo ) { self . options . localVideo . style . display = 'none' ; self . options . localVideo [ 'src' ] = '' ; }
2017-01-26 17:43:45 +00:00
if ( self . options . localVideoStream ) { if ( typeof self . options . localVideoStream . stop == 'function' ) { self . options . localVideoStream . stop ( ) ; } else { if ( self . options . localVideoStream . active ) { var tracks = self . options . localVideoStream . getTracks ( ) ; console . log ( tracks ) ; tracks . forEach ( function ( track , index ) { console . log ( track ) ; track . stop ( ) ; } ) } } }
2015-11-16 17:31:27 +00:00
if ( self . peer ) { console . log ( "stopping peer" ) ; self . peer . stop ( ) ; } } ; $ . FSRTC . prototype . getMute = function ( ) { var self = this ; return self . audioEnabled ; }
2015-03-24 18:39:51 +00:00
$ . FSRTC . prototype . setMute = function ( what ) { var self = this ; var audioTracks = self . localStream . getAudioTracks ( ) ; for ( var i = 0 , len = audioTracks . length ; i < len ; i ++ ) { switch ( what ) { case "on" : audioTracks [ i ] . enabled = true ; break ; case "off" : audioTracks [ i ] . enabled = false ; break ; case "toggle" : audioTracks [ i ] . enabled = ! audioTracks [ i ] . enabled ; default : break ; }
2015-11-16 17:31:27 +00:00
self . audioEnabled = audioTracks [ i ] . enabled ; }
return ! self . audioEnabled ; }
$ . FSRTC . prototype . getVideoMute = function ( ) { var self = this ; return self . videoEnabled ; }
$ . FSRTC . prototype . setVideoMute = function ( what ) { var self = this ; var videoTracks = self . localStream . getVideoTracks ( ) ; for ( var i = 0 , len = videoTracks . length ; i < len ; i ++ ) { switch ( what ) { case "on" : videoTracks [ i ] . enabled = true ; break ; case "off" : videoTracks [ i ] . enabled = false ; break ; case "toggle" : videoTracks [ i ] . enabled = ! videoTracks [ i ] . enabled ; default : break ; }
self . videoEnabled = videoTracks [ i ] . enabled ; }
return ! self . videoEnabled ; }
2017-01-26 17:43:45 +00:00
$ . FSRTC . prototype . createAnswer = function ( params ) { var self = this ; self . type = "answer" ; self . remoteSDP = params . sdp ; console . debug ( "inbound sdp: " , params . sdp ) ; function onSuccess ( stream ) { self . localStream = stream ; self . peer = FSRTCPeerConnection ( { type : self . type , attachStream : self . localStream , onICE : function ( candidate ) { return onICE ( self , candidate ) ; } , onICEComplete : function ( ) { return onICEComplete ( self ) ; } , onRemoteStream : function ( stream ) { return onRemoteStream ( self , stream ) ; } , onICESDP : function ( sdp ) { return onICESDP ( self , sdp ) ; } , onChannelError : function ( e ) { return onChannelError ( self , e ) ; } , constraints : self . constraints , iceServers : self . options . iceServers , offerSDP : { type : "offer" , sdp : self . remoteSDP } } ) ; onStreamSuccess ( self , stream ) ; }
2014-12-06 17:40:44 +00:00
function onError ( e ) { onStreamError ( self , e ) ; }
2016-06-08 21:20:18 +00:00
var mediaParams = getMediaParams ( self ) ; console . log ( "Audio constraints" , mediaParams . audio ) ; console . log ( "Video constraints" , mediaParams . video ) ; if ( self . options . useVideo && self . options . localVideo ) { getUserMedia ( { constraints : { audio : false , video : { } , } , localVideo : self . options . localVideo , onsuccess : function ( e ) { self . options . localVideoStream = e ; console . log ( "local video ready" ) ; } , onerror : function ( e ) { console . error ( "local video error!" ) ; } } ) ; }
2016-10-10 19:02:33 +00:00
getUserMedia ( { constraints : { audio : mediaParams . audio , video : mediaParams . video } , video : mediaParams . useVideo , onsuccess : onSuccess , onerror : onError } ) ; } ; function getMediaParams ( obj ) { var audio ; if ( obj . options . useMic && obj . options . useMic === "none" ) { console . log ( "Microphone Disabled" ) ; audio = false ; } else if ( obj . options . videoParams && obj . options . screenShare ) { console . error ( "SCREEN SHARE" , obj . options . videoParams ) ; audio = false ; } else { audio = { } ; if ( obj . options . audioParams ) { audio = obj . options . audioParams ; }
if ( obj . options . useMic !== "any" ) { audio . deviceId = { exact : obj . options . useMic } ; } }
2016-06-08 21:20:18 +00:00
if ( obj . options . useVideo && obj . options . localVideo ) { getUserMedia ( { constraints : { audio : false , video : obj . options . videoParams } , localVideo : obj . options . localVideo , onsuccess : function ( e ) { self . options . localVideoStream = e ; console . log ( "local video ready" ) ; } , onerror : function ( e ) { console . error ( "local video error!" ) ; } } ) ; }
2016-09-27 17:04:23 +00:00
var video = { } ; var bestFrameRate = obj . options . videoParams . vertoBestFrameRate ; var minFrameRate = obj . options . videoParams . minFrameRate || 15 ; delete obj . options . videoParams . vertoBestFrameRate ; if ( obj . options . screenShare ) { var opt = [ ] ; opt . push ( { sourceId : obj . options . useCamera } ) ; if ( bestFrameRate ) { opt . push ( { minFrameRate : bestFrameRate } ) ; opt . push ( { maxFrameRate : bestFrameRate } ) ; }
video = { mandatory : obj . options . videoParams , optional : opt } ; } else { video = { width : { min : obj . options . videoParams . minWidth , max : obj . options . videoParams . maxWidth } , height : { min : obj . options . videoParams . minHeight , max : obj . options . videoParams . maxHeight } } ; var useVideo = obj . options . useVideo ; if ( useVideo && obj . options . useCamera && obj . options . useCamera !== "none" ) { if ( obj . options . useCamera !== "any" ) { video . deviceId = obj . options . useCamera ; }
2016-06-08 21:20:18 +00:00
if ( bestFrameRate ) { video . frameRate = { ideal : bestFrameRate , min : minFrameRate , max : 30 } ; } } else { console . log ( "Camera Disabled" ) ; video = false ; useVideo = false ; } }
2015-02-19 23:08:36 +00:00
return { audio : audio , video : video , useVideo : useVideo } ; }
2015-03-04 21:13:35 +00:00
$ . FSRTC . prototype . call = function ( profile ) { checkCompat ( ) ; var self = this ; var screen = false ; self . type = "offer" ; if ( self . options . videoParams && self . options . screenShare ) { screen = true ; }
2016-06-08 21:20:18 +00:00
function onSuccess ( stream ) { self . localStream = stream ; if ( screen ) { self . constraints . offerToReceiveVideo = false ; }
self . peer = FSRTCPeerConnection ( { type : self . type , attachStream : self . localStream , onICE : function ( candidate ) { return onICE ( self , candidate ) ; } , onICEComplete : function ( ) { return onICEComplete ( self ) ; } , onRemoteStream : screen ? function ( stream ) { } : function ( stream ) { return onRemoteStream ( self , stream ) ; } , onOfferSDP : function ( sdp ) { return onOfferSDP ( self , sdp ) ; } , onICESDP : function ( sdp ) { return onICESDP ( self , sdp ) ; } , onChannelError : function ( e ) { return onChannelError ( self , e ) ; } , constraints : self . constraints , iceServers : self . options . iceServers , } ) ; onStreamSuccess ( self , stream ) ; }
2014-12-06 17:40:44 +00:00
function onError ( e ) { onStreamError ( self , e ) ; }
2016-06-08 21:20:18 +00:00
var mediaParams = getMediaParams ( self ) ; console . log ( "Audio constraints" , mediaParams . audio ) ; console . log ( "Video constraints" , mediaParams . video ) ; if ( mediaParams . audio || mediaParams . video ) { getUserMedia ( { constraints : { audio : mediaParams . audio , video : mediaParams . video } , video : mediaParams . useVideo , onsuccess : onSuccess , onerror : onError } ) ; } else { onSuccess ( null ) ; } } ; function FSRTCPeerConnection ( options ) { var gathering = false , done = false ; var config = { } ; var default _ice = { urls : [ 'stun:stun.l.google.com:19302' ] } ; if ( options . iceServers ) { if ( typeof ( options . iceServers ) === "boolean" ) { config . iceServers = [ default _ice ] ; } else { config . iceServers = options . iceServers ; } }
var peer = new window . RTCPeerConnection ( config ) ; openOffererChannel ( ) ; var x = 0 ; function ice _handler ( ) { done = true ; gathering = null ; if ( options . onICEComplete ) { options . onICEComplete ( ) ; }
if ( options . type == "offer" ) { options . onICESDP ( peer . localDescription ) ; } else { if ( ! x && options . onICESDP ) { options . onICESDP ( peer . localDescription ) ; } } }
2015-09-17 16:12:34 +00:00
peer . onicecandidate = function ( event ) { if ( done ) { return ; }
if ( ! gathering ) { gathering = setTimeout ( ice _handler , 1000 ) ; }
if ( event ) { if ( event . candidate ) { options . onICE ( event . candidate ) ; } } else { done = true ; if ( gathering ) { clearTimeout ( gathering ) ; gathering = null ; }
ice _handler ( ) ; } } ; if ( options . attachStream ) peer . addStream ( options . attachStream ) ; if ( options . attachStreams && options . attachStream . length ) { var streams = options . attachStreams ; for ( var i = 0 ; i < streams . length ; i ++ ) { peer . addStream ( streams [ i ] ) ; } }
2017-01-26 17:43:45 +00:00
peer . onaddstream = function ( event ) { var remoteMediaStream = event . stream ; remoteMediaStream . oninactive = function ( ) { if ( options . onRemoteStreamEnded ) options . onRemoteStreamEnded ( remoteMediaStream ) ; } ; if ( options . onRemoteStream ) options . onRemoteStream ( remoteMediaStream ) ; } ; function createOffer ( ) { if ( ! options . onOfferSDP ) return ; peer . createOffer ( function ( sessionDescription ) { sessionDescription . sdp = serializeSdp ( sessionDescription . sdp ) ; peer . setLocalDescription ( sessionDescription ) ; options . onOfferSDP ( sessionDescription ) ; } , onSdpError , options . constraints ) ; }
2016-06-08 21:20:18 +00:00
function createAnswer ( ) { if ( options . type != "answer" ) return ; peer . setRemoteDescription ( new window . RTCSessionDescription ( options . offerSDP ) , onSdpSuccess , onSdpError ) ; peer . createAnswer ( function ( sessionDescription ) { sessionDescription . sdp = serializeSdp ( sessionDescription . sdp ) ; peer . setLocalDescription ( sessionDescription ) ; if ( options . onAnswerSDP ) { options . onAnswerSDP ( sessionDescription ) ; } } , onSdpError ) ; }
if ( ( options . onChannelMessage ) || ! options . onChannelMessage ) { createOffer ( ) ; createAnswer ( ) ; }
2014-12-06 17:40:44 +00:00
function setBandwidth ( sdp ) { sdp = sdp . replace ( /b=AS([^\r\n]+\r\n)/g , '' ) ; sdp = sdp . replace ( /a=mid:data\r\n/g , 'a=mid:data\r\nb=AS:1638400\r\n' ) ; return sdp ; }
function getInteropSDP ( sdp ) { var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' . split ( '' ) , extractedChars = '' ; function getChars ( ) { extractedChars += chars [ parseInt ( Math . random ( ) * 40 ) ] || '' ; if ( extractedChars . length < 40 ) getChars ( ) ; return extractedChars ; }
if ( options . onAnswerSDP ) sdp = sdp . replace ( /(a=crypto:0 AES_CM_128_HMAC_SHA1_32)(.*?)(\r\n)/g , '' ) ; var inline = getChars ( ) + '\r\n' + ( extractedChars = '' ) ; sdp = sdp . indexOf ( 'a=crypto' ) == - 1 ? sdp . replace ( /c=IN/g , 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:' + inline + 'c=IN' ) : sdp ; return sdp ; }
function serializeSdp ( sdp ) { return sdp ; }
2016-06-08 21:20:18 +00:00
var channel ; function openOffererChannel ( ) { if ( ! options . onChannelMessage ) return ; _openOffererChannel ( ) ; return ; }
function _openOffererChannel ( ) { channel = peer . createDataChannel ( options . channel || 'RTCDataChannel' , { reliable : false } ) ; setChannelEvents ( ) ; }
2014-12-06 17:40:44 +00:00
function setChannelEvents ( ) { channel . onmessage = function ( event ) { if ( options . onChannelMessage ) options . onChannelMessage ( event ) ; } ; channel . onopen = function ( ) { if ( options . onChannelOpened ) options . onChannelOpened ( channel ) ; } ; channel . onclose = function ( event ) { if ( options . onChannelClosed ) options . onChannelClosed ( event ) ; console . warn ( 'WebRTC DataChannel closed' , event ) ; } ; channel . onerror = function ( event ) { if ( options . onChannelError ) options . onChannelError ( event ) ; console . error ( 'WebRTC DataChannel error' , event ) ; } ; }
2016-06-08 21:20:18 +00:00
function openAnswererChannel ( ) { peer . ondatachannel = function ( event ) { channel = event . channel ; channel . binaryType = 'blob' ; setChannelEvents ( ) ; } ; return ; }
2014-12-06 17:40:44 +00:00
function useless ( ) { log ( 'Error in fake:true' ) ; }
function onSdpSuccess ( ) { }
function onSdpError ( e ) { if ( options . onChannelError ) { options . onChannelError ( e ) ; }
console . error ( 'sdp error:' , e ) ; }
2016-06-08 21:20:18 +00:00
return { addAnswerSDP : function ( sdp , cbSuccess , cbError ) { peer . setRemoteDescription ( new window . RTCSessionDescription ( sdp ) , cbSuccess ? cbSuccess : onSdpSuccess , cbError ? cbError : onSdpError ) ; } , addICE : function ( candidate ) { peer . addIceCandidate ( new window . RTCIceCandidate ( { sdpMLineIndex : candidate . sdpMLineIndex , candidate : candidate . candidate } ) ) ; } , peer : peer , channel : channel , sendData : function ( message ) { if ( channel ) { channel . send ( message ) ; } } , stop : function ( ) { peer . close ( ) ; if ( options . attachStream ) { if ( typeof options . attachStream . stop == 'function' ) { options . attachStream . stop ( ) ; } else { options . attachStream . active = false ; } } } } ; }
var video _constraints = { } ; function getUserMedia ( options ) { var n = navigator , media ; n . getMedia = n . getUserMedia ; n . getMedia ( options . constraints || { audio : true , video : video _constraints } , streaming , options . onerror || function ( e ) { console . error ( e ) ; } ) ; function streaming ( stream ) { if ( options . localVideo ) { options . localVideo [ 'src' ] = window . URL . createObjectURL ( stream ) ; options . localVideo . style . display = 'block' ; }
2014-12-06 17:40:44 +00:00
if ( options . onsuccess ) { options . onsuccess ( stream ) ; }
media = stream ; }
2015-04-28 20:56:25 +00:00
return media ; }
2015-08-28 19:20:18 +00:00
$ . FSRTC . resSupported = function ( w , h ) { for ( var i in $ . FSRTC . validRes ) { if ( $ . FSRTC . validRes [ i ] [ 0 ] == w && $ . FSRTC . validRes [ i ] [ 1 ] == h ) { return true ; } }
2015-04-28 20:56:25 +00:00
return false ; }
2016-10-10 19:02:33 +00:00
$ . FSRTC . bestResSupported = function ( ) { var w = 0 , h = 0 ; for ( var i in $ . FSRTC . validRes ) { if ( $ . FSRTC . validRes [ i ] [ 0 ] >= w && $ . FSRTC . validRes [ i ] [ 1 ] >= h ) { w = $ . FSRTC . validRes [ i ] [ 0 ] ; h = $ . FSRTC . validRes [ i ] [ 1 ] ; } }
2015-04-28 20:56:25 +00:00
return [ w , h ] ; }
2016-06-08 21:20:18 +00:00
var resList = [ [ 160 , 120 ] , [ 320 , 180 ] , [ 320 , 240 ] , [ 640 , 360 ] , [ 640 , 480 ] , [ 1280 , 720 ] , [ 1920 , 1080 ] ] ; var resI = 0 ; var ttl = 0 ; var checkRes = function ( cam , func ) { if ( resI >= resList . length ) { var res = { 'validRes' : $ . FSRTC . validRes , 'bestResSupported' : $ . FSRTC . bestResSupported ( ) } ; localStorage . setItem ( "res_" + cam , $ . toJSON ( res ) ) ; if ( func ) return func ( res ) ; return ; }
var video = { }
if ( cam ) { video . deviceId = { exact : cam } ; }
2017-01-26 17:43:45 +00:00
w = resList [ resI ] [ 0 ] ; h = resList [ resI ] [ 1 ] ; resI ++ ; video = { width : w , height : h } ; getUserMedia ( { constraints : { audio : ttl ++ == 0 , video : video } , onsuccess : function ( e ) { e . getTracks ( ) . forEach ( function ( track ) { track . stop ( ) ; } ) ; console . info ( w + "x" + h + " supported." ) ; $ . FSRTC . validRes . push ( [ w , h ] ) ; checkRes ( cam , func ) ; } , onerror : function ( e ) { console . warn ( w + "x" + h + " not supported." ) ; checkRes ( cam , func ) ; } } ) ; }
2015-08-28 19:32:48 +00:00
$ . FSRTC . getValidRes = function ( cam , func ) { var used = [ ] ; var cached = localStorage . getItem ( "res_" + cam ) ; if ( cached ) { var cache = $ . parseJSON ( cached ) ; if ( cache ) { $ . FSRTC . validRes = cache . validRes ; console . log ( "CACHED RES FOR CAM " + cam , cache ) ; } else { console . error ( "INVALID CACHE" ) ; }
return func ? func ( cache ) : null ; }
2015-08-28 14:42:37 +00:00
$ . FSRTC . validRes = [ ] ; resI = 0 ; checkRes ( cam , func ) ; }
2015-09-30 06:56:50 +00:00
$ . FSRTC . checkPerms = function ( runtime , check _audio , check _video ) { getUserMedia ( { constraints : { audio : check _audio , video : check _video , } , onsuccess : function ( e ) { e . getTracks ( ) . forEach ( function ( track ) { track . stop ( ) ; } ) ; console . info ( "media perm init complete" ) ; if ( runtime ) { setTimeout ( runtime , 100 , true ) ; } } , onerror : function ( e ) { if ( check _video && check _audio ) { console . error ( "error, retesting with audio params only" ) ; return $ . FSRTC . checkPerms ( runtime , check _audio , false ) ; }
2015-11-16 17:31:27 +00:00
console . error ( "media perm init error" ) ; if ( runtime ) { runtime ( false ) } } } ) ; } } ) ( jQuery ) ; ( function ( $ ) { $ . JsonRpcClient = function ( options ) { var self = this ; this . options = $ . extend ( { ajaxUrl : null , socketUrl : null , onmessage : null , login : null , passwd : null , sessid : null , loginParams : null , userVariables : null , getSocket : function ( onmessage _cb ) { return self . _getSocket ( onmessage _cb ) ; } } , options ) ; self . ws _cnt = 0 ; this . wsOnMessage = function ( event ) { self . _wsOnMessage ( event ) ; } ; } ; $ . JsonRpcClient . prototype . _ws _socket = null ; $ . JsonRpcClient . prototype . _ws _callbacks = { } ; $ . JsonRpcClient . prototype . _current _id = 1 ; $ . JsonRpcClient . prototype . speedTest = function ( bytes , cb ) { var socket = this . options . getSocket ( this . wsOnMessage ) ; if ( socket !== null ) { this . speedCB = cb ; this . speedBytes = bytes ; socket . send ( "#SPU " + bytes ) ; var loops = bytes / 1024 ; var rem = bytes % 1024 ; var i ; var data = new Array ( 1024 ) . join ( "." ) ; for ( i = 0 ; i < loops ; i ++ ) { socket . send ( "#SPB " + data ) ; }
2015-10-03 18:34:09 +00:00
if ( rem ) { socket . send ( "#SPB " + data ) ; }
socket . send ( "#SPE" ) ; } } ; $ . JsonRpcClient . prototype . call = function ( method , params , success _cb , error _cb ) { if ( ! params ) { params = { } ; }
2014-12-06 17:40:44 +00:00
if ( this . options . sessid ) { params . sessid = this . options . sessid ; }
var request = { jsonrpc : '2.0' , method : method , params : params , id : this . _current _id ++ } ; if ( ! success _cb ) { success _cb = function ( e ) { console . log ( "Success: " , e ) ; } ; }
if ( ! error _cb ) { error _cb = function ( e ) { console . log ( "Error: " , e ) ; } ; }
var socket = this . options . getSocket ( this . wsOnMessage ) ; if ( socket !== null ) { this . _wsCall ( socket , request , success _cb , error _cb ) ; return ; }
if ( this . options . ajaxUrl === null ) { throw "$.JsonRpcClient.call used with no websocket and no http endpoint." ; }
$ . ajax ( { type : 'POST' , url : this . options . ajaxUrl , data : $ . toJSON ( request ) , dataType : 'json' , cache : false , success : function ( data ) { if ( 'error' in data ) error _cb ( data . error , this ) ; success _cb ( data . result , this ) ; } , error : function ( jqXHR , textStatus , errorThrown ) { try { var response = $ . parseJSON ( jqXHR . responseText ) ; if ( 'console' in window ) console . log ( response ) ; error _cb ( response . error , this ) ; } catch ( err ) { error _cb ( { error : jqXHR . responseText } , this ) ; } } } ) ; } ; $ . JsonRpcClient . prototype . notify = function ( method , params ) { if ( this . options . sessid ) { params . sessid = this . options . sessid ; }
var request = { jsonrpc : '2.0' , method : method , params : params } ; var socket = this . options . getSocket ( this . wsOnMessage ) ; if ( socket !== null ) { this . _wsCall ( socket , request ) ; return ; }
if ( this . options . ajaxUrl === null ) { throw "$.JsonRpcClient.notify used with no websocket and no http endpoint." ; }
$ . ajax ( { type : 'POST' , url : this . options . ajaxUrl , data : $ . toJSON ( request ) , dataType : 'json' , cache : false } ) ; } ; $ . JsonRpcClient . prototype . batch = function ( callback , all _done _cb , error _cb ) { var batch = new $ . JsonRpcClient . _batchObject ( this , all _done _cb , error _cb ) ; callback ( batch ) ; batch . _execute ( ) ; } ; $ . JsonRpcClient . prototype . socketReady = function ( ) { if ( this . _ws _socket === null || this . _ws _socket . readyState > 1 ) { return false ; }
2015-04-28 20:56:25 +00:00
return true ; } ; $ . JsonRpcClient . prototype . closeSocket = function ( ) { var self = this ; if ( self . socketReady ( ) ) { self . _ws _socket . onclose = function ( w ) { console . log ( "Closing Socket" ) ; } ; self . _ws _socket . close ( ) ; } } ; $ . JsonRpcClient . prototype . loginData = function ( params ) { var self = this ; self . options . login = params . login ; self . options . passwd = params . passwd ; self . options . loginParams = params . loginParams ; self . options . userVariables = params . userVariables ; } ; $ . JsonRpcClient . prototype . connectSocket = function ( onmessage _cb ) { var self = this ; if ( self . to ) { clearTimeout ( self . to ) ; }
2014-12-06 17:40:44 +00:00
if ( ! self . socketReady ( ) ) { self . authing = false ; if ( self . _ws _socket ) { delete self . _ws _socket ; }
self . _ws _socket = new WebSocket ( self . options . socketUrl ) ; if ( self . _ws _socket ) { self . _ws _socket . onmessage = onmessage _cb ; self . _ws _socket . onclose = function ( w ) { if ( ! self . ws _sleep ) { self . ws _sleep = 1000 ; }
if ( self . options . onWSClose ) { self . options . onWSClose ( self ) ; }
2015-01-07 07:43:49 +00:00
console . error ( "Websocket Lost " + self . ws _cnt + " sleep: " + self . ws _sleep + "msec" ) ; self . to = setTimeout ( function ( ) { console . log ( "Attempting Reconnection...." ) ; self . connectSocket ( onmessage _cb ) ; } , self . ws _sleep ) ; self . ws _cnt ++ ; if ( self . ws _sleep < 3000 && ( self . ws _cnt % 10 ) === 0 ) { self . ws _sleep += 1000 ; } } ; self . _ws _socket . onopen = function ( ) { if ( self . to ) { clearTimeout ( self . to ) ; }
2014-12-06 17:40:44 +00:00
self . ws _sleep = 1000 ; self . ws _cnt = 0 ; if ( self . options . onWSConnect ) { self . options . onWSConnect ( self ) ; }
2015-01-07 07:43:49 +00:00
var req ; while ( ( req = $ . JsonRpcClient . q . pop ( ) ) ) { self . _ws _socket . send ( req ) ; } } ; } }
2017-01-26 17:43:45 +00:00
return self . _ws _socket ? true : false ; } ; $ . JsonRpcClient . prototype . stopRetrying = function ( ) { if ( self . to )
clearTimeout ( self . to ) ; }
$ . JsonRpcClient . prototype . _getSocket = function ( onmessage _cb ) { if ( this . options . socketUrl === null || ! ( "WebSocket" in window ) ) return null ; this . connectSocket ( onmessage _cb ) ; return this . _ws _socket ; } ; $ . JsonRpcClient . q = [ ] ; $ . JsonRpcClient . prototype . _wsCall = function ( socket , request , success _cb , error _cb ) { var request _json = $ . toJSON ( request ) ; if ( socket . readyState < 1 ) { self = this ; $ . JsonRpcClient . q . push ( request _json ) ; } else { socket . send ( request _json ) ; }
2015-10-03 07:36:28 +00:00
if ( 'id' in request && typeof success _cb !== 'undefined' ) { this . _ws _callbacks [ request . id ] = { request : request _json , request _obj : request , success _cb : success _cb , error _cb : error _cb } ; } } ; $ . JsonRpcClient . prototype . _wsOnMessage = function ( event ) { var response ; if ( event . data [ 0 ] == "#" && event . data [ 1 ] == "S" && event . data [ 2 ] == "P" ) { if ( event . data [ 3 ] == "U" ) { this . up _dur = parseInt ( event . data . substring ( 4 ) ) ; } else if ( this . speedCB && event . data [ 3 ] == "D" ) { this . down _dur = parseInt ( event . data . substring ( 4 ) ) ; var up _kps = ( ( ( this . speedBytes * 8 ) / ( this . up _dur / 1000 ) ) / 1024 ) . toFixed ( 0 ) ; var down _kps = ( ( ( this . speedBytes * 8 ) / ( this . down _dur / 1000 ) ) / 1024 ) . toFixed ( 0 ) ; console . info ( "Speed Test: Up: " + up _kps + " Down: " + down _kps ) ; this . speedCB ( event , { upDur : this . up _dur , downDur : this . down _dur , upKPS : up _kps , downKPS : down _kps } ) ; this . speedCB = null ; }
return ; }
try { response = $ . parseJSON ( event . data ) ; if ( typeof response === 'object' && 'jsonrpc' in response && response . jsonrpc === '2.0' ) { if ( 'result' in response && this . _ws _callbacks [ response . id ] ) { var success _cb = this . _ws _callbacks [ response . id ] . success _cb ; delete this . _ws _callbacks [ response . id ] ; success _cb ( response . result , this ) ; return ; } else if ( 'error' in response && this . _ws _callbacks [ response . id ] ) { var error _cb = this . _ws _callbacks [ response . id ] . error _cb ; var orig _req = this . _ws _callbacks [ response . id ] . request ; if ( ! self . authing && response . error . code == - 32000 && self . options . login && self . options . passwd ) { self . authing = true ; this . call ( "login" , { login : self . options . login , passwd : self . options . passwd , loginParams : self . options . loginParams , userVariables : self . options . userVariables } , this . _ws _callbacks [ response . id ] . request _obj . method == "login" ? function ( e ) { self . authing = false ; console . log ( "logged in" ) ; delete self . _ws _callbacks [ response . id ] ; if ( self . options . onWSLogin ) { self . options . onWSLogin ( true , self ) ; } } : function ( e ) { self . authing = false ; console . log ( "logged in, resending request id: " + response . id ) ; var socket = self . options . getSocket ( self . wsOnMessage ) ; if ( socket !== null ) { socket . send ( orig _req ) ; }
2014-12-06 17:40:44 +00:00
if ( self . options . onWSLogin ) { self . options . onWSLogin ( true , self ) ; } } , function ( e ) { console . log ( "error logging in, request id:" , response . id ) ; delete self . _ws _callbacks [ response . id ] ; error _cb ( response . error , this ) ; if ( self . options . onWSLogin ) { self . options . onWSLogin ( false , self ) ; } } ) ; return ; }
delete this . _ws _callbacks [ response . id ] ; error _cb ( response . error , this ) ; return ; } } } catch ( err ) { console . log ( "ERROR: " + err ) ; return ; }
if ( typeof this . options . onmessage === 'function' ) { event . eventData = response ; if ( ! event . eventData ) { event . eventData = { } ; }
var reply = this . options . onmessage ( event ) ; if ( reply && typeof reply === "object" && event . eventData . id ) { var msg = { jsonrpc : "2.0" , id : event . eventData . id , result : reply } ; var socket = self . options . getSocket ( self . wsOnMessage ) ; if ( socket !== null ) { socket . send ( $ . toJSON ( msg ) ) ; } } } } ; $ . JsonRpcClient . _batchObject = function ( jsonrpcclient , all _done _cb , error _cb ) { this . _requests = [ ] ; this . jsonrpcclient = jsonrpcclient ; this . all _done _cb = all _done _cb ; this . error _cb = typeof error _cb === 'function' ? error _cb : function ( ) { } ; } ; $ . JsonRpcClient . _batchObject . prototype . call = function ( method , params , success _cb , error _cb ) { if ( ! params ) { params = { } ; }
if ( this . options . sessid ) { params . sessid = this . options . sessid ; }
if ( ! success _cb ) { success _cb = function ( e ) { console . log ( "Success: " , e ) ; } ; }
if ( ! error _cb ) { error _cb = function ( e ) { console . log ( "Error: " , e ) ; } ; }
this . _requests . push ( { request : { jsonrpc : '2.0' , method : method , params : params , id : this . jsonrpcclient . _current _id ++ } , success _cb : success _cb , error _cb : error _cb } ) ; } ; $ . JsonRpcClient . _batchObject . prototype . notify = function ( method , params ) { if ( this . options . sessid ) { params . sessid = this . options . sessid ; }
2015-01-07 07:43:49 +00:00
this . _requests . push ( { request : { jsonrpc : '2.0' , method : method , params : params } } ) ; } ; $ . JsonRpcClient . _batchObject . prototype . _execute = function ( ) { var self = this ; if ( this . _requests . length === 0 ) return ; var batch _request = [ ] ; var handlers = { } ; var i = 0 ; var call ; var success _cb ; var error _cb ; var socket = self . jsonrpcclient . options . getSocket ( self . jsonrpcclient . wsOnMessage ) ; if ( socket !== null ) { for ( i = 0 ; i < this . _requests . length ; i ++ ) { call = this . _requests [ i ] ; success _cb = ( 'success_cb' in call ) ? call . success _cb : undefined ; error _cb = ( 'error_cb' in call ) ? call . error _cb : undefined ; self . jsonrpcclient . _wsCall ( socket , call . request , success _cb , error _cb ) ; }
2014-12-06 17:40:44 +00:00
if ( typeof all _done _cb === 'function' ) all _done _cb ( result ) ; return ; }
2015-01-07 07:43:49 +00:00
for ( i = 0 ; i < this . _requests . length ; i ++ ) { call = this . _requests [ i ] ; batch _request . push ( call . request ) ; if ( 'id' in call . request ) { handlers [ call . request . id ] = { success _cb : call . success _cb , error _cb : call . error _cb } ; } }
success _cb = function ( data ) { self . _batchCb ( data , handlers , self . all _done _cb ) ; } ; if ( self . jsonrpcclient . options . ajaxUrl === null ) { throw "$.JsonRpcClient.batch used with no websocket and no http endpoint." ; }
2014-12-06 17:40:44 +00:00
$ . ajax ( { url : self . jsonrpcclient . options . ajaxUrl , data : $ . toJSON ( batch _request ) , dataType : 'json' , cache : false , type : 'POST' , error : function ( jqXHR , textStatus , errorThrown ) { self . error _cb ( jqXHR , textStatus , errorThrown ) ; } , success : success _cb } ) ; } ; $ . JsonRpcClient . _batchObject . prototype . _batchCb = function ( result , handlers , all _done _cb ) { for ( var i = 0 ; i < result . length ; i ++ ) { var response = result [ i ] ; if ( 'error' in response ) { if ( response . id === null || ! ( response . id in handlers ) ) { if ( 'console' in window ) console . log ( response ) ; } else { handlers [ response . id ] . error _cb ( response . error , this ) ; } } else { if ( ! ( response . id in handlers ) && 'console' in window ) { console . log ( response ) ; } else { handlers [ response . id ] . success _cb ( response . result , this ) ; } } }
2015-02-19 23:08:36 +00:00
if ( typeof all _done _cb === 'function' ) all _done _cb ( result ) ; } ; } ) ( jQuery ) ; ( function ( $ ) { var sources = [ ] ; var generateGUID = ( typeof ( window . crypto ) !== 'undefined' && typeof ( window . crypto . getRandomValues ) !== 'undefined' ) ? function ( ) { var buf = new Uint16Array ( 8 ) ; window . crypto . getRandomValues ( buf ) ; var S4 = function ( num ) { var ret = num . toString ( 16 ) ; while ( ret . length < 4 ) { ret = "0" + ret ; }
2015-09-30 08:30:24 +00:00
return ret ; } ; return ( S4 ( buf [ 0 ] ) + S4 ( buf [ 1 ] ) + "-" + S4 ( buf [ 2 ] ) + "-" + S4 ( buf [ 3 ] ) + "-" + S4 ( buf [ 4 ] ) + "-" + S4 ( buf [ 5 ] ) + S4 ( buf [ 6 ] ) + S4 ( buf [ 7 ] ) ) ; } : function ( ) { return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx' . replace ( /[xy]/g , function ( c ) { var r = Math . random ( ) * 16 | 0 , v = c == 'x' ? r : ( r & 0x3 | 0x8 ) ; return v . toString ( 16 ) ; } ) ; } ; $ . verto = function ( options , callbacks ) { var verto = this ; $ . verto . saved . push ( verto ) ; verto . options = $ . extend ( { login : null , passwd : null , socketUrl : null , tag : null , localTag : null , videoParams : { } , audioParams : { } , loginParams : { } , deviceParams : { onResCheck : null } , userVariables : { } , iceServers : false , ringSleep : 6000 , sessid : null } , options ) ; if ( verto . options . deviceParams . useCamera ) { $ . FSRTC . getValidRes ( verto . options . deviceParams . useCamera , verto . options . deviceParams . onResCheck ) ; }
2015-08-25 21:04:34 +00:00
if ( ! verto . options . deviceParams . useMic ) { verto . options . deviceParams . useMic = "any" ; }
if ( ! verto . options . deviceParams . useSpeak ) { verto . options . deviceParams . useSpeak = "any" ; }
2015-08-28 14:42:37 +00:00
if ( verto . options . sessid ) { verto . sessid = verto . options . sessid ; } else { verto . sessid = localStorage . getItem ( "verto_session_uuid" ) || generateGUID ( ) ; localStorage . setItem ( "verto_session_uuid" , verto . sessid ) ; }
2015-07-02 22:55:04 +00:00
verto . dialogs = { } ; verto . callbacks = callbacks || { } ; verto . eventSUBS = { } ; verto . rpcClient = new $ . JsonRpcClient ( { login : verto . options . login , passwd : verto . options . passwd , socketUrl : verto . options . socketUrl , loginParams : verto . options . loginParams , userVariables : verto . options . userVariables , sessid : verto . sessid , onmessage : function ( e ) { return verto . handleMessage ( e . eventData ) ; } , onWSConnect : function ( o ) { o . call ( 'login' , { } ) ; } , onWSLogin : function ( success ) { if ( verto . callbacks . onWSLogin ) { verto . callbacks . onWSLogin ( verto , success ) ; } } , onWSClose : function ( success ) { if ( verto . callbacks . onWSClose ) { verto . callbacks . onWSClose ( verto , success ) ; }
2016-05-12 22:18:13 +00:00
verto . purge ( ) ; } } ) ; var tag = verto . options . tag ; if ( typeof ( tag ) === "function" ) { tag = tag ( ) ; }
if ( verto . options . ringFile && verto . options . tag ) { verto . ringer = $ ( "#" + tag ) ; }
2015-08-28 14:45:52 +00:00
verto . rpcClient . call ( 'login' , { } ) ; } ; $ . verto . prototype . deviceParams = function ( obj ) { var verto = this ; for ( var i in obj ) { verto . options . deviceParams [ i ] = obj [ i ] ; }
2015-08-28 19:34:31 +00:00
if ( obj . useCamera ) { $ . FSRTC . getValidRes ( verto . options . deviceParams . useCamera , obj ? obj . onResCheck : undefined ) ; } } ; $ . verto . prototype . videoParams = function ( obj ) { var verto = this ; for ( var i in obj ) { verto . options . videoParams [ i ] = obj [ i ] ; } } ; $ . verto . prototype . iceServers = function ( obj ) { var verto = this ; verto . options . iceServers = obj ; } ; $ . verto . prototype . loginData = function ( params ) { var verto = this ; verto . options . login = params . login ; verto . options . passwd = params . passwd ; verto . rpcClient . loginData ( params ) ; } ; $ . verto . prototype . logout = function ( msg ) { var verto = this ; verto . rpcClient . closeSocket ( ) ; if ( verto . callbacks . onWSClose ) { verto . callbacks . onWSClose ( verto , false ) ; }
2015-01-06 03:20:31 +00:00
verto . purge ( ) ; } ; $ . verto . prototype . login = function ( msg ) { var verto = this ; verto . logout ( ) ; verto . rpcClient . call ( 'login' , { } ) ; } ; $ . verto . prototype . message = function ( msg ) { var verto = this ; var err = 0 ; if ( ! msg . to ) { console . error ( "Missing To" ) ; err ++ ; }
2014-12-06 17:40:44 +00:00
if ( ! msg . body ) { console . error ( "Missing Body" ) ; err ++ ; }
if ( err ) { return false ; }
verto . sendMethod ( "verto.info" , { msg : msg } ) ; return true ; } ; $ . verto . prototype . processReply = function ( method , success , e ) { var verto = this ; var i ; switch ( method ) { case "verto.subscribe" : for ( i in e . unauthorizedChannels ) { drop _bad ( verto , e . unauthorizedChannels [ i ] ) ; }
for ( i in e . subscribedChannels ) { mark _ready ( verto , e . subscribedChannels [ i ] ) ; }
break ; case "verto.unsubscribe" : break ; } } ; $ . verto . prototype . sendMethod = function ( method , params ) { var verto = this ; verto . rpcClient . call ( method , params , function ( e ) { verto . processReply ( method , true , e ) ; } , function ( e ) { verto . processReply ( method , false , e ) ; } ) ; } ; function do _sub ( verto , channel , obj ) { }
function drop _bad ( verto , channel ) { console . error ( "drop unauthorized channel: " + channel ) ; delete verto . eventSUBS [ channel ] ; }
function mark _ready ( verto , channel ) { for ( var j in verto . eventSUBS [ channel ] ) { verto . eventSUBS [ channel ] [ j ] . ready = true ; console . log ( "subscribed to channel: " + channel ) ; if ( verto . eventSUBS [ channel ] [ j ] . readyHandler ) { verto . eventSUBS [ channel ] [ j ] . readyHandler ( verto , channel ) ; } } }
var SERNO = 1 ; function do _subscribe ( verto , channel , subChannels , sparams ) { var params = sparams || { } ; var local = params . local ; var obj = { eventChannel : channel , userData : params . userData , handler : params . handler , ready : false , readyHandler : params . readyHandler , serno : SERNO ++ } ; var isnew = false ; if ( ! verto . eventSUBS [ channel ] ) { verto . eventSUBS [ channel ] = [ ] ; subChannels . push ( channel ) ; isnew = true ; }
verto . eventSUBS [ channel ] . push ( obj ) ; if ( local ) { obj . ready = true ; obj . local = true ; }
if ( ! isnew && verto . eventSUBS [ channel ] [ 0 ] . ready ) { obj . ready = true ; if ( obj . readyHandler ) { obj . readyHandler ( verto , channel ) ; } }
return { serno : obj . serno , eventChannel : channel } ; }
$ . verto . prototype . subscribe = function ( channel , sparams ) { var verto = this ; var r = [ ] ; var subChannels = [ ] ; var params = sparams || { } ; if ( typeof ( channel ) === "string" ) { r . push ( do _subscribe ( verto , channel , subChannels , params ) ) ; } else { for ( var i in channel ) { r . push ( do _subscribe ( verto , channel , subChannels , params ) ) ; } }
if ( subChannels . length ) { verto . sendMethod ( "verto.subscribe" , { eventChannel : subChannels . length == 1 ? subChannels [ 0 ] : subChannels , subParams : params . subParams } ) ; }
return r ; } ; $ . verto . prototype . unsubscribe = function ( handle ) { var verto = this ; var i ; if ( ! handle ) { for ( i in verto . eventSUBS ) { if ( verto . eventSUBS [ i ] ) { verto . unsubscribe ( verto . eventSUBS [ i ] ) ; } } } else { var unsubChannels = { } ; var sendChannels = [ ] ; var channel ; if ( typeof ( handle ) == "string" ) { delete verto . eventSUBS [ handle ] ; unsubChannels [ handle ] ++ ; } else { for ( i in handle ) { if ( typeof ( handle [ i ] ) == "string" ) { channel = handle [ i ] ; delete verto . eventSUBS [ channel ] ; unsubChannels [ channel ] ++ ; } else { var repl = [ ] ; channel = handle [ i ] . eventChannel ; for ( var j in verto . eventSUBS [ channel ] ) { if ( verto . eventSUBS [ channel ] [ j ] . serno == handle [ i ] . serno ) { } else { repl . push ( verto . eventSUBS [ channel ] [ j ] ) ; } }
verto . eventSUBS [ channel ] = repl ; if ( verto . eventSUBS [ channel ] . length === 0 ) { delete verto . eventSUBS [ channel ] ; unsubChannels [ channel ] ++ ; } } } }
for ( var u in unsubChannels ) { console . log ( "Sending Unsubscribe for: " , u ) ; sendChannels . push ( u ) ; }
if ( sendChannels . length ) { verto . sendMethod ( "verto.unsubscribe" , { eventChannel : sendChannels . length == 1 ? sendChannels [ 0 ] : sendChannels } ) ; } } } ; $ . verto . prototype . broadcast = function ( channel , params ) { var verto = this ; var msg = { eventChannel : channel , data : { } } ; for ( var i in params ) { msg . data [ i ] = params [ i ] ; }
verto . sendMethod ( "verto.broadcast" , msg ) ; } ; $ . verto . prototype . purge = function ( callID ) { var verto = this ; var x = 0 ; var i ; for ( i in verto . dialogs ) { if ( ! x ) { console . log ( "purging dialogs" ) ; }
x ++ ; verto . dialogs [ i ] . setState ( $ . verto . enum . state . purge ) ; }
for ( i in verto . eventSUBS ) { if ( verto . eventSUBS [ i ] ) { console . log ( "purging subscription: " + i ) ; delete verto . eventSUBS [ i ] ; } } } ; $ . verto . prototype . hangup = function ( callID ) { var verto = this ; if ( callID ) { var dialog = verto . dialogs [ callID ] ; if ( dialog ) { dialog . hangup ( ) ; } } else { for ( var i in verto . dialogs ) { verto . dialogs [ i ] . hangup ( ) ; } } } ; $ . verto . prototype . newCall = function ( args , callbacks ) { var verto = this ; if ( ! verto . rpcClient . socketReady ( ) ) { console . error ( "Not Connected..." ) ; return ; }
var dialog = new $ . verto . dialog ( $ . verto . enum . direction . outbound , this , args ) ; dialog . invite ( ) ; if ( callbacks ) { dialog . callbacks = callbacks ; }
return dialog ; } ; $ . verto . prototype . handleMessage = function ( data ) { var verto = this ; if ( ! ( data && data . method ) ) { console . error ( "Invalid Data" , data ) ; return ; }
2015-03-24 00:56:19 +00:00
if ( data . params . callID ) { var dialog = verto . dialogs [ data . params . callID ] ; if ( data . method === "verto.attach" && dialog ) { delete dialog . verto . dialogs [ dialog . callID ] ; dialog . rtc . stop ( ) ; dialog = null ; }
if ( dialog ) { switch ( data . method ) { case 'verto.bye' : dialog . hangup ( data . params ) ; break ; case 'verto.answer' : dialog . handleAnswer ( data . params ) ; break ; case 'verto.media' : dialog . handleMedia ( data . params ) ; break ; case 'verto.display' : dialog . handleDisplay ( data . params ) ; break ; case 'verto.info' : dialog . handleInfo ( data . params ) ; break ; default : console . debug ( "INVALID METHOD OR NON-EXISTANT CALL REFERENCE IGNORED" , dialog , data . method ) ; break ; } } else { switch ( data . method ) { case 'verto.attach' : data . params . attach = true ; if ( data . params . sdp && data . params . sdp . indexOf ( "m=video" ) > 0 ) { data . params . useVideo = true ; }
2014-12-06 17:40:44 +00:00
if ( data . params . sdp && data . params . sdp . indexOf ( "stereo=1" ) > 0 ) { data . params . useStereo = true ; }
dialog = new $ . verto . dialog ( $ . verto . enum . direction . inbound , verto , data . params ) ; dialog . setState ( $ . verto . enum . state . recovering ) ; break ; case 'verto.invite' : if ( data . params . sdp && data . params . sdp . indexOf ( "m=video" ) > 0 ) { data . params . wantVideo = true ; }
if ( data . params . sdp && data . params . sdp . indexOf ( "stereo=1" ) > 0 ) { data . params . useStereo = true ; }
dialog = new $ . verto . dialog ( $ . verto . enum . direction . inbound , verto , data . params ) ; break ; default : console . debug ( "INVALID METHOD OR NON-EXISTANT CALL REFERENCE IGNORED" ) ; break ; } }
2015-01-06 03:20:31 +00:00
return { method : data . method } ; } else { switch ( data . method ) { case 'verto.punt' : verto . purge ( ) ; verto . logout ( ) ; break ; case 'verto.event' : var list = null ; var key = null ; if ( data . params ) { key = data . params . eventChannel ; }
2014-12-06 17:40:44 +00:00
if ( key ) { list = verto . eventSUBS [ key ] ; if ( ! list ) { list = verto . eventSUBS [ key . split ( "." ) [ 0 ] ] ; } }
if ( ! list && key && key === verto . sessid ) { if ( verto . callbacks . onMessage ) { verto . callbacks . onMessage ( verto , null , $ . verto . enum . message . pvtEvent , data . params ) ; } } else if ( ! list && key && verto . dialogs [ key ] ) { verto . dialogs [ key ] . sendMessage ( $ . verto . enum . message . pvtEvent , data . params ) ; } else if ( ! list ) { if ( ! key ) { key = "UNDEFINED" ; }
console . error ( "UNSUBBED or invalid EVENT " + key + " IGNORED" ) ; } else { for ( var i in list ) { var sub = list [ i ] ; if ( ! sub || ! sub . ready ) { console . error ( "invalid EVENT for " + key + " IGNORED" ) ; } else if ( sub . handler ) { sub . handler ( verto , data . params , sub . userData ) ; } else if ( verto . callbacks . onEvent ) { verto . callbacks . onEvent ( verto , data . params , sub . userData ) ; } else { console . log ( "EVENT:" , data . params ) ; } } }
break ; case "verto.info" : if ( verto . callbacks . onMessage ) { verto . callbacks . onMessage ( verto , null , $ . verto . enum . message . info , data . params . msg ) ; }
console . debug ( "MESSAGE from: " + data . params . msg . from , data . params . msg . body ) ; break ; default : console . error ( "INVALID METHOD OR NON-EXISTANT CALL REFERENCE IGNORED" , data . method ) ; break ; } } } ; var del _array = function ( array , name ) { var r = [ ] ; var len = array . length ; for ( var i = 0 ; i < len ; i ++ ) { if ( array [ i ] != name ) { r . push ( array [ i ] ) ; } }
return r ; } ; var hashArray = function ( ) { var vha = this ; var hash = { } ; var array = [ ] ; vha . reorder = function ( a ) { array = a ; var h = hash ; hash = { } ; var len = array . length ; for ( var i = 0 ; i < len ; i ++ ) { var key = array [ i ] ; if ( h [ key ] ) { hash [ key ] = h [ key ] ; delete h [ key ] ; } }
h = undefined ; } ; vha . clear = function ( ) { hash = undefined ; array = undefined ; hash = { } ; array = [ ] ; } ; vha . add = function ( name , val , insertAt ) { var redraw = false ; if ( ! hash [ name ] ) { if ( insertAt === undefined || insertAt < 0 || insertAt >= array . length ) { array . push ( name ) ; } else { var x = 0 ; var n = [ ] ; var len = array . length ; for ( var i = 0 ; i < len ; i ++ ) { if ( x ++ == insertAt ) { n . push ( name ) ; }
n . push ( array [ i ] ) ; }
array = undefined ; array = n ; n = undefined ; redraw = true ; } }
hash [ name ] = val ; return redraw ; } ; vha . del = function ( name ) { var r = false ; if ( hash [ name ] ) { array = del _array ( array , name ) ; delete hash [ name ] ; r = true ; } else { console . error ( "can't del nonexistant key " + name ) ; }
return r ; } ; vha . get = function ( name ) { return hash [ name ] ; } ; vha . order = function ( ) { return array ; } ; vha . hash = function ( ) { return hash ; } ; vha . indexOf = function ( name ) { var len = array . length ; for ( var i = 0 ; i < len ; i ++ ) { if ( array [ i ] == name ) { return i ; } } } ; vha . arrayLen = function ( ) { return array . length ; } ; vha . asArray = function ( ) { var r = [ ] ; var len = array . length ; for ( var i = 0 ; i < len ; i ++ ) { var key = array [ i ] ; r . push ( hash [ key ] ) ; }
return r ; } ; vha . each = function ( cb ) { var len = array . length ; for ( var i = 0 ; i < len ; i ++ ) { cb ( array [ i ] , hash [ array [ i ] ] ) ; } } ; vha . dump = function ( html ) { var str = "" ; vha . each ( function ( name , val ) { str += "name: " + name + " val: " + JSON . stringify ( val ) + ( html ? "<br>" : "\n" ) ; } ) ; return str ; } ; } ; $ . verto . liveArray = function ( verto , context , name , config ) { var la = this ; var lastSerno = 0 ; var binding = null ; var user _obj = config . userObj ; var local = false ; hashArray . call ( la ) ; la . _add = la . add ; la . _del = la . del ; la . _reorder = la . reorder ; la . _clear = la . clear ; la . context = context ; la . name = name ; la . user _obj = user _obj ; la . verto = verto ; la . broadcast = function ( channel , obj ) { verto . broadcast ( channel , obj ) ; } ; la . errs = 0 ; la . clear = function ( ) { la . _clear ( ) ; lastSerno = 0 ; if ( la . onChange ) { la . onChange ( la , { action : "clear" } ) ; } } ; la . checkSerno = function ( serno ) { if ( serno < 0 ) { return true ; }
if ( lastSerno > 0 && serno != ( lastSerno + 1 ) ) { if ( la . onErr ) { la . onErr ( la , { lastSerno : lastSerno , serno : serno } ) ; }
la . errs ++ ; console . debug ( la . errs ) ; if ( la . errs < 3 ) { la . bootstrap ( la . user _obj ) ; }
return false ; } else { lastSerno = serno ; return true ; } } ; la . reorder = function ( serno , a ) { if ( la . checkSerno ( serno ) ) { la . _reorder ( a ) ; if ( la . onChange ) { la . onChange ( la , { serno : serno , action : "reorder" } ) ; } } } ; la . init = function ( serno , val , key , index ) { if ( key === null || key === undefined ) { key = serno ; }
if ( la . checkSerno ( serno ) ) { if ( la . onChange ) { la . onChange ( la , { serno : serno , action : "init" , index : index , key : key , data : val } ) ; } } } ; la . bootObj = function ( serno , val ) { if ( la . checkSerno ( serno ) ) { for ( var i in val ) { la . _add ( val [ i ] [ 0 ] , val [ i ] [ 1 ] ) ; }
if ( la . onChange ) { la . onChange ( la , { serno : serno , action : "bootObj" , data : val , redraw : true } ) ; } } } ; la . add = function ( serno , val , key , index ) { if ( key === null || key === undefined ) { key = serno ; }
if ( la . checkSerno ( serno ) ) { var redraw = la . _add ( key , val , index ) ; if ( la . onChange ) { la . onChange ( la , { serno : serno , action : "add" , index : index , key : key , data : val , redraw : redraw } ) ; } } } ; la . modify = function ( serno , val , key , index ) { if ( key === null || key === undefined ) { key = serno ; }
if ( la . checkSerno ( serno ) ) { la . _add ( key , val , index ) ; if ( la . onChange ) { la . onChange ( la , { serno : serno , action : "modify" , key : key , data : val , index : index } ) ; } } } ; la . del = function ( serno , key , index ) { if ( key === null || key === undefined ) { key = serno ; }
if ( la . checkSerno ( serno ) ) { if ( index === null || index < 0 || index === undefined ) { index = la . indexOf ( key ) ; }
var ok = la . _del ( key ) ; if ( ok && la . onChange ) { la . onChange ( la , { serno : serno , action : "del" , key : key , index : index } ) ; } } } ; var eventHandler = function ( v , e , la ) { var packet = e . data ; if ( packet . name != la . name ) { return ; }
switch ( packet . action ) { case "init" : la . init ( packet . wireSerno , packet . data , packet . hashKey , packet . arrIndex ) ; break ; case "bootObj" : la . bootObj ( packet . wireSerno , packet . data ) ; break ; case "add" : la . add ( packet . wireSerno , packet . data , packet . hashKey , packet . arrIndex ) ; break ; case "modify" : if ( ! ( packet . arrIndex || packet . hashKey ) ) { console . error ( "Invalid Packet" , packet ) ; } else { la . modify ( packet . wireSerno , packet . data , packet . hashKey , packet . arrIndex ) ; }
break ; case "del" : if ( ! ( packet . arrIndex || packet . hashKey ) ) { console . error ( "Invalid Packet" , packet ) ; } else { la . del ( packet . wireSerno , packet . hashKey , packet . arrIndex ) ; }
break ; case "clear" : la . clear ( ) ; break ; case "reorder" : la . reorder ( packet . wireSerno , packet . order ) ; break ; default : if ( la . checkSerno ( packet . wireSerno ) ) { if ( la . onChange ) { la . onChange ( la , { serno : packet . wireSerno , action : packet . action , data : packet . data } ) ; } }
break ; } } ; if ( la . context ) { binding = la . verto . subscribe ( la . context , { handler : eventHandler , userData : la , subParams : config . subParams } ) ; }
la . destroy = function ( ) { la . _clear ( ) ; la . verto . unsubscribe ( binding ) ; } ; la . sendCommand = function ( cmd , obj ) { var self = la ; self . broadcast ( self . context , { liveArray : { command : cmd , context : self . context , name : self . name , obj : obj } } ) ; } ; la . bootstrap = function ( obj ) { var self = la ; la . sendCommand ( "bootstrap" , obj ) ; } ; la . changepage = function ( obj ) { var self = la ; self . clear ( ) ; self . broadcast ( self . context , { liveArray : { command : "changepage" , context : la . context , name : la . name , obj : obj } } ) ; } ; la . heartbeat = function ( obj ) { var self = la ; var callback = function ( ) { self . heartbeat . call ( self , obj ) ; } ; self . broadcast ( self . context , { liveArray : { command : "heartbeat" , context : self . context , name : self . name , obj : obj } } ) ; self . hb _pid = setTimeout ( callback , 30000 ) ; } ; la . bootstrap ( la . user _obj ) ; } ; $ . verto . liveTable = function ( verto , context , name , jq , config ) { var dt ; var la = new $ . verto . liveArray ( verto , context , name , { subParams : config . subParams } ) ; var lt = this ; lt . liveArray = la ; lt . dataTable = dt ; lt . verto = verto ; lt . destroy = function ( ) { if ( dt ) { dt . fnDestroy ( ) ; }
if ( la ) { la . destroy ( ) ; }
2015-11-16 23:02:07 +00:00
dt = null ; la = null ; } ; la . onErr = function ( obj , args ) { console . error ( "Error: " , obj , args ) ; } ; function genRow ( data ) { if ( typeof ( data [ 4 ] ) === "string" && data [ 4 ] . indexOf ( "{" ) > - 1 ) { var tmp = $ . parseJSON ( data [ 4 ] ) ; data [ 4 ] = tmp . oldStatus ; data [ 5 ] = null ; }
return data ; }
function genArray ( obj ) { var data = obj . asArray ( ) ; for ( var i in data ) { data [ i ] = genRow ( data [ i ] ) ; }
return data ; }
la . onChange = function ( obj , args ) { var index = 0 ; var iserr = 0 ; if ( ! dt ) { if ( ! config . aoColumns ) { if ( args . action != "init" ) { return ; }
2014-12-06 17:40:44 +00:00
config . aoColumns = [ ] ; for ( var i in args . data ) { config . aoColumns . push ( { "sTitle" : args . data [ i ] } ) ; } }
dt = jq . dataTable ( config ) ; }
if ( dt && ( args . action == "del" || args . action == "modify" ) ) { index = args . index ; if ( index === undefined && args . key ) { index = la . indexOf ( args . key ) ; }
if ( index === undefined ) { console . error ( "INVALID PACKET Missing INDEX\n" , args ) ; return ; } }
if ( config . onChange ) { config . onChange ( obj , args ) ; }
try { switch ( args . action ) { case "bootObj" : if ( ! args . data ) { console . error ( "missing data" ) ; return ; }
2015-11-16 23:02:07 +00:00
dt . fnClearTable ( ) ; dt . fnAddData ( genArray ( obj ) ) ; dt . fnAdjustColumnSizing ( ) ; break ; case "add" : if ( ! args . data ) { console . error ( "missing data" ) ; return ; }
if ( args . redraw > - 1 ) { dt . fnClearTable ( ) ; dt . fnAddData ( genArray ( obj ) ) ; } else { dt . fnAddData ( genRow ( args . data ) ) ; }
2014-12-06 17:40:44 +00:00
dt . fnAdjustColumnSizing ( ) ; break ; case "modify" : if ( ! args . data ) { return ; }
2015-11-16 23:02:07 +00:00
dt . fnUpdate ( genRow ( args . data ) , index ) ; dt . fnAdjustColumnSizing ( ) ; break ; case "del" : dt . fnDeleteRow ( index ) ; dt . fnAdjustColumnSizing ( ) ; break ; case "clear" : dt . fnClearTable ( ) ; break ; case "reorder" : dt . fnClearTable ( ) ; dt . fnAddData ( genArray ( obj ) ) ; break ; case "hide" : jq . hide ( ) ; break ; case "show" : jq . show ( ) ; break ; } } catch ( err ) { console . error ( "ERROR: " + err ) ; iserr ++ ; }
2015-08-28 22:47:05 +00:00
if ( iserr ) { obj . errs ++ ; if ( obj . errs < 3 ) { obj . bootstrap ( obj . user _obj ) ; } } else { obj . errs = 0 ; } } ; la . onChange ( la , { action : "init" } ) ; } ; var CONFMAN _SERNO = 1 ; $ . verto . conf = function ( verto , params ) { var conf = this ; conf . params = $ . extend ( { dialog : null , hasVid : false , laData : null , onBroadcast : null , onLaChange : null , onLaRow : null } , params ) ; conf . verto = verto ; conf . serno = CONFMAN _SERNO ++ ; createMainModeratorMethods ( ) ; verto . subscribe ( conf . params . laData . modChannel , { handler : function ( v , e ) { if ( conf . params . onBroadcast ) { conf . params . onBroadcast ( verto , conf , e . data ) ; } } } ) ; verto . subscribe ( conf . params . laData . chatChannel , { handler : function ( v , e ) { if ( typeof ( conf . params . chatCallback ) === "function" ) { conf . params . chatCallback ( v , e ) ; } } } ) ; } ; $ . verto . conf . prototype . modCommand = function ( cmd , id , value ) { var conf = this ; conf . verto . rpcClient . call ( "verto.broadcast" , { "eventChannel" : conf . params . laData . modChannel , "data" : { "application" : "conf-control" , "command" : cmd , "id" : id , "value" : value } } ) ; } ; $ . verto . conf . prototype . destroy = function ( ) { var conf = this ; conf . destroyed = true ; conf . params . onBroadcast ( conf . verto , conf , 'destroy' ) ; if ( conf . params . laData . modChannel ) { conf . verto . unsubscribe ( conf . params . laData . modChannel ) ; }
2016-10-10 19:02:33 +00:00
if ( conf . params . laData . chatChannel ) { conf . verto . unsubscribe ( conf . params . laData . chatChannel ) ; } } ; function createMainModeratorMethods ( ) { $ . verto . conf . prototype . listVideoLayouts = function ( ) { this . modCommand ( "list-videoLayouts" , null , null ) ; } ; $ . verto . conf . prototype . play = function ( file ) { this . modCommand ( "play" , null , file ) ; } ; $ . verto . conf . prototype . stop = function ( ) { this . modCommand ( "stop" , null , "all" ) ; } ; $ . verto . conf . prototype . deaf = function ( memberID ) { this . modCommand ( "deaf" , parseInt ( memberID ) ) ; } ; $ . verto . conf . prototype . undeaf = function ( memberID ) { this . modCommand ( "undeaf" , parseInt ( memberID ) ) ; } ; $ . verto . conf . prototype . record = function ( file ) { this . modCommand ( "recording" , null , [ "start" , file ] ) ; } ; $ . verto . conf . prototype . stopRecord = function ( ) { this . modCommand ( "recording" , null , [ "stop" , "all" ] ) ; } ; $ . verto . conf . prototype . snapshot = function ( file ) { if ( ! this . params . hasVid ) { throw 'Conference has no video' ; }
2016-03-18 23:30:49 +00:00
this . modCommand ( "vid-write-png" , null , file ) ; } ; $ . verto . conf . prototype . setVideoLayout = function ( layout , canvasID ) { if ( ! this . params . hasVid ) { throw 'Conference has no video' ; }
if ( canvasID ) { this . modCommand ( "vid-layout" , null , [ layout , canvasID ] ) ; } else { this . modCommand ( "vid-layout" , null , layout ) ; } } ; $ . verto . conf . prototype . kick = function ( memberID ) { this . modCommand ( "kick" , parseInt ( memberID ) ) ; } ; $ . verto . conf . prototype . muteMic = function ( memberID ) { this . modCommand ( "tmute" , parseInt ( memberID ) ) ; } ; $ . verto . conf . prototype . muteVideo = function ( memberID ) { if ( ! this . params . hasVid ) { throw 'Conference has no video' ; }
2015-08-20 22:14:03 +00:00
this . modCommand ( "tvmute" , parseInt ( memberID ) ) ; } ; $ . verto . conf . prototype . presenter = function ( memberID ) { if ( ! this . params . hasVid ) { throw 'Conference has no video' ; }
this . modCommand ( "vid-res-id" , parseInt ( memberID ) , "presenter" ) ; } ; $ . verto . conf . prototype . videoFloor = function ( memberID ) { if ( ! this . params . hasVid ) { throw 'Conference has no video' ; }
this . modCommand ( "vid-floor" , parseInt ( memberID ) , "force" ) ; } ; $ . verto . conf . prototype . banner = function ( memberID , text ) { if ( ! this . params . hasVid ) { throw 'Conference has no video' ; }
2015-12-04 19:55:43 +00:00
this . modCommand ( "vid-banner" , parseInt ( memberID ) , escape ( text ) ) ; } ; $ . verto . conf . prototype . volumeDown = function ( memberID ) { this . modCommand ( "volume_out" , parseInt ( memberID ) , "down" ) ; } ; $ . verto . conf . prototype . volumeUp = function ( memberID ) { this . modCommand ( "volume_out" , parseInt ( memberID ) , "up" ) ; } ; $ . verto . conf . prototype . gainDown = function ( memberID ) { this . modCommand ( "volume_in" , parseInt ( memberID ) , "down" ) ; } ; $ . verto . conf . prototype . gainUp = function ( memberID ) { this . modCommand ( "volume_in" , parseInt ( memberID ) , "up" ) ; } ; $ . verto . conf . prototype . transfer = function ( memberID , exten ) { this . modCommand ( "transfer" , parseInt ( memberID ) , exten ) ; } ; $ . verto . conf . prototype . sendChat = function ( message , type ) { var conf = this ; conf . verto . rpcClient . call ( "verto.broadcast" , { "eventChannel" : conf . params . laData . chatChannel , "data" : { "action" : "send" , "message" : message , "type" : type } } ) ; } ; }
2015-08-20 22:14:03 +00:00
$ . verto . modfuncs = { } ; $ . verto . confMan = function ( verto , params ) { var confMan = this ; confMan . params = $ . extend ( { tableID : null , statusID : null , mainModID : null , dialog : null , hasVid : false , laData : null , onBroadcast : null , onLaChange : null , onLaRow : null } , params ) ; confMan . verto = verto ; confMan . serno = CONFMAN _SERNO ++ ; confMan . canvasCount = confMan . params . laData . canvasCount ; function genMainMod ( jq ) { var play _id = "play_" + confMan . serno ; var stop _id = "stop_" + confMan . serno ; var recording _id = "recording_" + confMan . serno ; var snapshot _id = "snapshot_" + confMan . serno ; var rec _stop _id = "recording_stop" + confMan . serno ; var div _id = "confman_" + confMan . serno ; var html = "<div id='" + div _id + "'><br>" + "<button class='ctlbtn' id='" + play _id + "'>Play</button>" + "<button class='ctlbtn' id='" + stop _id + "'>Stop</button>" + "<button class='ctlbtn' id='" + recording _id + "'>Record</button>" + "<button class='ctlbtn' id='" + rec _stop _id + "'>Record Stop</button>" +
2016-03-18 23:21:25 +00:00
( confMan . params . hasVid ? "<button class='ctlbtn' id='" + snapshot _id + "'>PNG Snapshot</button>" : "" ) + "<br><br></div>" ; jq . html ( html ) ; $ . verto . modfuncs . change _video _layout = function ( id , canvas _id ) { var val = $ ( "#" + id + " option:selected" ) . text ( ) ; if ( val !== "none" ) { confMan . modCommand ( "vid-layout" , null , [ val , canvas _id ] ) ; } } ; if ( confMan . params . hasVid ) { for ( var j = 0 ; j < confMan . canvasCount ; j ++ ) { var vlayout _id = "confman_vid_layout_" + j + "_" + confMan . serno ; var vlselect _id = "confman_vl_select_" + j + "_" + confMan . serno ; var vlhtml = "<div id='" + vlayout _id + "'><br>" + "<b>Video Layout Canvas " + ( j + 1 ) + "</b> <select onChange='$.verto.modfuncs.change_video_layout(\"" + vlayout _id + "\", \"" + ( j + 1 ) + "\")' id='" + vlselect _id + "'></select> " + "<br><br></div>" ; jq . append ( vlhtml ) ; }
2015-07-02 22:55:04 +00:00
$ ( "#" + snapshot _id ) . click ( function ( ) { var file = prompt ( "Please enter file name" , "" ) ; if ( file ) { confMan . modCommand ( "vid-write-png" , null , file ) ; } } ) ; }
2015-02-21 00:04:00 +00:00
$ ( "#" + play _id ) . click ( function ( ) { var file = prompt ( "Please enter file name" , "" ) ; if ( file ) { confMan . modCommand ( "play" , null , file ) ; } } ) ; $ ( "#" + stop _id ) . click ( function ( ) { confMan . modCommand ( "stop" , null , "all" ) ; } ) ; $ ( "#" + recording _id ) . click ( function ( ) { var file = prompt ( "Please enter file name" , "" ) ; if ( file ) { confMan . modCommand ( "recording" , null , [ "start" , file ] ) ; } } ) ; $ ( "#" + rec _stop _id ) . click ( function ( ) { confMan . modCommand ( "recording" , null , [ "stop" , "all" ] ) ; } ) ; }
2015-12-04 19:55:43 +00:00
function genControls ( jq , rowid ) { var x = parseInt ( rowid ) ; var kick _id = "kick_" + x ; var canvas _in _next _id = "canvas_in_next_" + x ; var canvas _in _prev _id = "canvas_in_prev_" + x ; var canvas _out _next _id = "canvas_out_next_" + x ; var canvas _out _prev _id = "canvas_out_prev_" + x ; var canvas _in _set _id = "canvas_in_set_" + x ; var canvas _out _set _id = "canvas_out_set_" + x ; var layer _set _id = "layer_set_" + x ; var layer _next _id = "layer_next_" + x ; var layer _prev _id = "layer_prev_" + x ; var tmute _id = "tmute_" + x ; var tvmute _id = "tvmute_" + x ; var vbanner _id = "vbanner_" + x ; var tvpresenter _id = "tvpresenter_" + x ; var tvfloor _id = "tvfloor_" + x ; var box _id = "box_" + x ; var gainup _id = "gain_in_up" + x ; var gaindn _id = "gain_in_dn" + x ; var volup _id = "vol_in_up" + x ; var voldn _id = "vol_in_dn" + x ; var transfer _id = "transfer" + x ; var html = "<div id='" + box _id + "'>" ; html += "<b>General Controls</b><hr noshade>" ; html += "<button class='ctlbtn' id='" + kick _id + "'>Kick</button>" + "<button class='ctlbtn' id='" + tmute _id + "'>Mute</button>" + "<button class='ctlbtn' id='" + gainup _id + "'>Gain -</button>" + "<button class='ctlbtn' id='" + gaindn _id + "'>Gain +</button>" + "<button class='ctlbtn' id='" + voldn _id + "'>Vol -</button>" + "<button class='ctlbtn' id='" + volup _id + "'>Vol +</button>" + "<button class='ctlbtn' id='" + transfer _id + "'>Transfer</button>" ; if ( confMan . params . hasVid ) { html += "<br><br><b>Video Controls</b><hr noshade>" ; html += "<button class='ctlbtn' id='" + tvmute _id + "'>VMute</button>" + "<button class='ctlbtn' id='" + tvpresenter _id + "'>Presenter</button>" + "<button class='ctlbtn' id='" + tvfloor _id + "'>Vid Floor</button>" + "<button class='ctlbtn' id='" + vbanner _id + "'>Banner</button>" ; if ( confMan . canvasCount > 1 ) { html += "<br><br><b>Canvas Controls</b><hr noshade>" + "<button class='ctlbtn' id='" + canvas _in _set _id + "'>Set Input Canvas</button>" + "<button class='ctlbtn' id='" + canvas _in _prev _id + "'>Prev Input Canvas</button>" + "<button class='ctlbtn' id='" + canvas _in _next _id + "'>Next Input Canvas</button>" + "<br>" + "<button class='ctlbtn' id='" + canvas _out _set _id + "'>Set Watching Canvas</button>" + "<button class='ctlbtn' id='" + canvas _out _prev _id + "'>Prev Watching Canvas</button>" + "<button class='ctlbtn' id='" + canvas _out _next _id + "'>Next Watching Canvas</button>" ; }
2015-07-02 22:55:04 +00:00
html += "<br>" + "<button class='ctlbtn' id='" + layer _set _id + "'>Set Layer</button>" + "<button class='ctlbtn' id='" + layer _prev _id + "'>Prev Layer</button>" + "<button class='ctlbtn' id='" + layer _next _id + "'>Next Layer</button>" + "</div>" ; }
jq . html ( html ) ; if ( ! jq . data ( "mouse" ) ) { $ ( "#" + box _id ) . hide ( ) ; }
jq . mouseover ( function ( e ) { jq . data ( { "mouse" : true } ) ; $ ( "#" + box _id ) . show ( ) ; } ) ; jq . mouseout ( function ( e ) { jq . data ( { "mouse" : false } ) ; $ ( "#" + box _id ) . hide ( ) ; } ) ; $ ( "#" + transfer _id ) . click ( function ( ) { var xten = prompt ( "Enter Extension" ) ; if ( xten ) { confMan . modCommand ( "transfer" , x , xten ) ; } } ) ; $ ( "#" + kick _id ) . click ( function ( ) { confMan . modCommand ( "kick" , x ) ; } ) ; $ ( "#" + layer _set _id ) . click ( function ( ) { var cid = prompt ( "Please enter layer ID" , "" ) ; if ( cid ) { confMan . modCommand ( "vid-layer" , x , cid ) ; } } ) ; $ ( "#" + layer _next _id ) . click ( function ( ) { confMan . modCommand ( "vid-layer" , x , "next" ) ; } ) ; $ ( "#" + layer _prev _id ) . click ( function ( ) { confMan . modCommand ( "vid-layer" , x , "prev" ) ; } ) ; $ ( "#" + canvas _in _set _id ) . click ( function ( ) { var cid = prompt ( "Please enter canvas ID" , "" ) ; if ( cid ) { confMan . modCommand ( "vid-canvas" , x , cid ) ; } } ) ; $ ( "#" + canvas _out _set _id ) . click ( function ( ) { var cid = prompt ( "Please enter canvas ID" , "" ) ; if ( cid ) { confMan . modCommand ( "vid-watching-canvas" , x , cid ) ; } } ) ; $ ( "#" + canvas _in _next _id ) . click ( function ( ) { confMan . modCommand ( "vid-canvas" , x , "next" ) ; } ) ; $ ( "#" + canvas _in _prev _id ) . click ( function ( ) { confMan . modCommand ( "vid-canvas" , x , "prev" ) ; } ) ; $ ( "#" + canvas _out _next _id ) . click ( function ( ) { confMan . modCommand ( "vid-watching-canvas" , x , "next" ) ; } ) ; $ ( "#" + canvas _out _prev _id ) . click ( function ( ) { confMan . modCommand ( "vid-watching-canvas" , x , "prev" ) ; } ) ; $ ( "#" + tmute _id ) . click ( function ( ) { confMan . modCommand ( "tmute" , x ) ; } ) ; if ( confMan . params . hasVid ) { $ ( "#" + tvmute _id ) . click ( function ( ) { confMan . modCommand ( "tvmute" , x ) ; } ) ; $ ( "#" + tvpresenter _id ) . click ( function ( ) { confMan . modCommand ( "vid-res-id" , x , "presenter" ) ; } ) ; $ ( "#" + tvfloor _id ) . click ( function ( ) { confMan . modCommand ( "vid-floor" , x , "force" ) ; } ) ; $ ( "#" + vbanner _id ) . click ( function ( ) { var text = prompt ( "Please enter text" , "" ) ; if ( text ) { confMan . modCommand ( "vid-banner" , x , escape ( text ) ) ; } } ) ; }
2015-12-04 19:55:43 +00:00
$ ( "#" + gainup _id ) . click ( function ( ) { confMan . modCommand ( "volume_in" , x , "up" ) ; } ) ; $ ( "#" + gaindn _id ) . click ( function ( ) { confMan . modCommand ( "volume_in" , x , "down" ) ; } ) ; $ ( "#" + volup _id ) . click ( function ( ) { confMan . modCommand ( "volume_out" , x , "up" ) ; } ) ; $ ( "#" + voldn _id ) . click ( function ( ) { confMan . modCommand ( "volume_out" , x , "down" ) ; } ) ; return html ; }
2015-07-24 00:09:24 +00:00
var atitle = "" ; var awidth = 0 ; verto . subscribe ( confMan . params . laData . chatChannel , { handler : function ( v , e ) { if ( typeof ( confMan . params . chatCallback ) === "function" ) { confMan . params . chatCallback ( v , e ) ; } } } ) ; if ( confMan . params . laData . role === "moderator" ) { atitle = "Action" ; awidth = 600 ; if ( confMan . params . mainModID ) { genMainMod ( $ ( confMan . params . mainModID ) ) ; $ ( confMan . params . displayID ) . html ( "Moderator Controls Ready<br><br>" ) ; } else { $ ( confMan . params . mainModID ) . html ( "" ) ; }
2015-04-09 19:29:23 +00:00
verto . subscribe ( confMan . params . laData . modChannel , { handler : function ( v , e ) { if ( confMan . params . onBroadcast ) { confMan . params . onBroadcast ( verto , confMan , e . data ) ; }
2015-11-19 17:54:29 +00:00
if ( e . data [ "conf-command" ] === "list-videoLayouts" ) { for ( var j = 0 ; j < confMan . canvasCount ; j ++ ) { var vlselect _id = "#confman_vl_select_" + j + "_" + confMan . serno ; var vlayout _id = "#confman_vid_layout_" + j + "_" + confMan . serno ; var x = 0 ; var options ; $ ( vlselect _id ) . selectmenu ( { } ) ; $ ( vlselect _id ) . selectmenu ( "enable" ) ; $ ( vlselect _id ) . empty ( ) ; $ ( vlselect _id ) . append ( new Option ( "Choose a Layout" , "none" ) ) ; if ( e . data . responseData ) { var rdata = [ ] ; for ( var i in e . data . responseData ) { rdata . push ( e . data . responseData [ i ] . name ) ; }
options = rdata . sort ( function ( a , b ) { var ga = a . substring ( 0 , 6 ) == "group:" ? true : false ; var gb = b . substring ( 0 , 6 ) == "group:" ? true : false ; if ( ( ga || gb ) && ga != gb ) { return ga ? - 1 : 1 ; }
return ( ( a == b ) ? 0 : ( ( a > b ) ? 1 : - 1 ) ) ; } ) ; for ( var i in options ) { $ ( vlselect _id ) . append ( new Option ( options [ i ] , options [ i ] ) ) ; x ++ ; } }
2015-07-02 22:55:04 +00:00
if ( x ) { $ ( vlselect _id ) . selectmenu ( 'refresh' , true ) ; } else { $ ( vlayout _id ) . hide ( ) ; } } } else { if ( ! confMan . destroyed && confMan . params . displayID ) { $ ( confMan . params . displayID ) . html ( e . data . response + "<br><br>" ) ; if ( confMan . lastTimeout ) { clearTimeout ( confMan . lastTimeout ) ; confMan . lastTimeout = 0 ; }
2015-02-20 23:19:09 +00:00
confMan . lastTimeout = setTimeout ( function ( ) { $ ( confMan . params . displayID ) . html ( confMan . destroyed ? "" : "Moderator Controls Ready<br><br>" ) ; } , 4000 ) ; } } } } ) ; if ( confMan . params . hasVid ) { confMan . modCommand ( "list-videoLayouts" , null , null ) ; } }
2014-12-06 17:40:44 +00:00
var row _callback = null ; if ( confMan . params . laData . role === "moderator" ) { row _callback = function ( nRow , aData , iDisplayIndex , iDisplayIndexFull ) { if ( ! aData [ 5 ] ) { var $row = $ ( 'td:eq(5)' , nRow ) ; genControls ( $row , aData ) ; if ( confMan . params . onLaRow ) { confMan . params . onLaRow ( verto , confMan , $row , aData ) ; } } } ; }
2015-07-24 00:09:24 +00:00
confMan . lt = new $ . verto . liveTable ( verto , confMan . params . laData . laChannel , confMan . params . laData . laName , $ ( confMan . params . tableID ) , { subParams : { callID : confMan . params . dialog ? confMan . params . dialog . callID : null } , "onChange" : function ( obj , args ) { $ ( confMan . params . statusID ) . text ( "Conference Members: " + " (" + obj . arrayLen ( ) + " Total)" ) ; if ( confMan . params . onLaChange ) { confMan . params . onLaChange ( verto , confMan , $ . verto . enum . confEvent . laChange , obj , args ) ; } } , "aaData" : [ ] , "aoColumns" : [ { "sTitle" : "ID" , "sWidth" : "50" } , { "sTitle" : "Number" , "sWidth" : "250" } , { "sTitle" : "Name" , "sWidth" : "250" } , { "sTitle" : "Codec" , "sWidth" : "100" } , { "sTitle" : "Status" , "sWidth" : confMan . params . hasVid ? "200px" : "150px" } , { "sTitle" : atitle , "sWidth" : awidth , } ] , "bAutoWidth" : true , "bDestroy" : true , "bSort" : false , "bInfo" : false , "bFilter" : false , "bLengthChange" : false , "bPaginate" : false , "iDisplayLength" : 1400 , "oLanguage" : { "sEmptyTable" : "The Conference is Empty....." } , "fnRowCallback" : row _callback } ) ; } ; $ . verto . confMan . prototype . modCommand = function ( cmd , id , value ) { var confMan = this ; confMan . verto . rpcClient . call ( "verto.broadcast" , { "eventChannel" : confMan . params . laData . modChannel , "data" : { "application" : "conf-control" , "command" : cmd , "id" : id , "value" : value } } ) ; } ; $ . verto . confMan . prototype . sendChat = function ( message , type ) { var confMan = this ; confMan . verto . rpcClient . call ( "verto.broadcast" , { "eventChannel" : confMan . params . laData . chatChannel , "data" : { "action" : "send" , "message" : message , "type" : type } } ) ; } ; $ . verto . confMan . prototype . destroy = function ( ) { var confMan = this ; confMan . destroyed = true ; if ( confMan . lt ) { confMan . lt . destroy ( ) ; }
if ( confMan . params . laData . chatChannel ) { confMan . verto . unsubscribe ( confMan . params . laData . chatChannel ) ; }
2014-12-06 17:40:44 +00:00
if ( confMan . params . laData . modChannel ) { confMan . verto . unsubscribe ( confMan . params . laData . modChannel ) ; }
2017-01-26 17:43:45 +00:00
if ( confMan . params . mainModID ) { $ ( confMan . params . mainModID ) . html ( "" ) ; } } ; $ . verto . dialog = function ( direction , verto , params ) { var dialog = this ; dialog . params = $ . extend ( { useVideo : verto . options . useVideo , useStereo : verto . options . useStereo , screenShare : false , useCamera : verto . options . deviceParams . useCamera , useMic : verto . options . deviceParams . useMic , useSpeak : verto . options . deviceParams . useSpeak , tag : verto . options . tag , localTag : verto . options . localTag , login : verto . options . login , videoParams : verto . options . videoParams } , params ) ; dialog . verto = verto ; dialog . direction = direction ; dialog . lastState = null ; dialog . state = dialog . lastState = $ . verto . enum . state . new ; dialog . callbacks = verto . callbacks ; dialog . answered = false ; dialog . attach = params . attach || false ; dialog . screenShare = params . screenShare || false ; dialog . useCamera = dialog . params . useCamera ; dialog . useMic = dialog . params . useMic ; dialog . useSpeak = dialog . params . useSpeak ; if ( dialog . params . callID ) { dialog . callID = dialog . params . callID ; } else { dialog . callID = dialog . params . callID = generateGUID ( ) ; }
if ( typeof ( dialog . params . tag ) === "function" ) { dialog . params . tag = dialog . params . tag ( ) ; }
2014-12-06 17:40:44 +00:00
if ( dialog . params . tag ) { dialog . audioStream = document . getElementById ( dialog . params . tag ) ; if ( dialog . params . useVideo ) { dialog . videoStream = dialog . audioStream ; } }
2015-01-24 09:13:44 +00:00
if ( dialog . params . localTag ) { dialog . localVideo = document . getElementById ( dialog . params . localTag ) ; }
2014-12-06 17:40:44 +00:00
dialog . verto . dialogs [ dialog . callID ] = dialog ; var RTCcallbacks = { } ; if ( dialog . direction == $ . verto . enum . direction . inbound ) { if ( dialog . params . display _direction === "outbound" ) { dialog . params . remote _caller _id _name = dialog . params . caller _id _name ; dialog . params . remote _caller _id _number = dialog . params . caller _id _number ; } else { dialog . params . remote _caller _id _name = dialog . params . callee _id _name ; dialog . params . remote _caller _id _number = dialog . params . callee _id _number ; }
if ( ! dialog . params . remote _caller _id _name ) { dialog . params . remote _caller _id _name = "Nobody" ; }
if ( ! dialog . params . remote _caller _id _number ) { dialog . params . remote _caller _id _number = "UNKNOWN" ; }
RTCcallbacks . onMessage = function ( rtc , msg ) { console . debug ( msg ) ; } ; RTCcallbacks . onAnswerSDP = function ( rtc , sdp ) { console . error ( "answer sdp" , sdp ) ; } ; } else { dialog . params . remote _caller _id _name = "Outbound Call" ; dialog . params . remote _caller _id _number = dialog . params . destination _number ; }
2015-08-28 14:42:37 +00:00
RTCcallbacks . onICESDP = function ( rtc ) { console . log ( "RECV " + rtc . type + " SDP" , rtc . mediaData . SDP ) ; if ( dialog . state == $ . verto . enum . state . requesting || dialog . state == $ . verto . enum . state . answering || dialog . state == $ . verto . enum . state . active ) { location . reload ( ) ; return ; }
2017-01-26 17:43:45 +00:00
if ( rtc . type == "offer" ) { if ( dialog . state == $ . verto . enum . state . active ) { dialog . setState ( $ . verto . enum . state . requesting ) ; dialog . sendMethod ( "verto.attach" , { sdp : rtc . mediaData . SDP } ) ; } else { dialog . setState ( $ . verto . enum . state . requesting ) ; dialog . sendMethod ( "verto.invite" , { sdp : rtc . mediaData . SDP } ) ; } } else { dialog . setState ( $ . verto . enum . state . answering ) ; dialog . sendMethod ( dialog . attach ? "verto.attach" : "verto.answer" , { sdp : dialog . rtc . mediaData . SDP } ) ; } } ; RTCcallbacks . onICE = function ( rtc ) { if ( rtc . type == "offer" ) { console . log ( "offer" , rtc . mediaData . candidate ) ; return ; } } ; RTCcallbacks . onStream = function ( rtc , stream ) { if ( dialog . verto . options . permissionCallback && typeof dialog . verto . options . permissionCallback . onGranted === 'function' ) { dialog . verto . options . permissionCallback . onGranted ( stream ) ; }
2016-06-08 21:20:18 +00:00
console . log ( "stream started" ) ; } ; RTCcallbacks . onError = function ( e ) { if ( dialog . verto . options . permissionCallback && typeof dialog . verto . options . permissionCallback . onDenied === 'function' ) { dialog . verto . options . permissionCallback . onDenied ( ) ; }
console . error ( "ERROR:" , e ) ; dialog . hangup ( { cause : "Device or Permission Error" } ) ; } ; dialog . rtc = new $ . FSRTC ( { callbacks : RTCcallbacks , localVideo : dialog . screenShare ? null : dialog . localVideo , useVideo : dialog . params . useVideo ? dialog . videoStream : null , useAudio : dialog . audioStream , useStereo : dialog . params . useStereo , videoParams : dialog . params . videoParams , audioParams : verto . options . audioParams , iceServers : verto . options . iceServers , screenShare : dialog . screenShare , useCamera : dialog . useCamera , useMic : dialog . useMic , useSpeak : dialog . useSpeak } ) ; dialog . rtc . verto = dialog . verto ; if ( dialog . direction == $ . verto . enum . direction . inbound ) { if ( dialog . attach ) { dialog . answer ( ) ; } else { dialog . ring ( ) ; } } } ; $ . verto . dialog . prototype . invite = function ( ) { var dialog = this ; dialog . rtc . call ( ) ; } ; $ . verto . dialog . prototype . sendMethod = function ( method , obj ) { var dialog = this ; obj . dialogParams = { } ; for ( var i in dialog . params ) { if ( i == "sdp" && method != "verto.invite" && method != "verto.attach" ) { continue ; }
2016-07-11 03:25:14 +00:00
if ( ( obj . noDialogParams && i != "callID" ) ) { continue ; }
2014-12-06 17:40:44 +00:00
obj . dialogParams [ i ] = dialog . params [ i ] ; }
2016-07-11 03:25:14 +00:00
delete obj . noDialogParams ; dialog . verto . rpcClient . call ( method , obj , function ( e ) { dialog . processReply ( method , true , e ) ; } , function ( e ) { dialog . processReply ( method , false , e ) ; } ) ; } ; function checkStateChange ( oldS , newS ) { if ( newS == $ . verto . enum . state . purge || $ . verto . enum . states [ oldS . name ] [ newS . name ] ) { return true ; }
2014-12-06 17:40:44 +00:00
return false ; }
2015-11-18 20:43:29 +00:00
function find _name ( id ) { for ( var i in $ . verto . audioOutDevices ) { var source = $ . verto . audioOutDevices [ i ] ; if ( source . id === id ) { return ( source . label ) ; } }
return id ; }
2015-11-19 17:54:29 +00:00
$ . verto . dialog . prototype . setAudioPlaybackDevice = function ( sinkId , callback , arg ) { var dialog = this ; var element = dialog . audioStream ; if ( typeof element . sinkId !== 'undefined' ) { var devname = find _name ( sinkId ) ; console . info ( "Dialog: " + dialog . callID + " Setting speaker:" , element , devname ) ; element . setSinkId ( sinkId ) . then ( function ( ) { console . log ( "Dialog: " + dialog . callID + ' Success, audio output device attached: ' + sinkId ) ; if ( callback ) { callback ( true , devname , arg ) ; } } ) . catch ( function ( error ) { var errorMessage = error ; if ( error . name === 'SecurityError' ) { errorMessage = "Dialog: " + dialog . callID + ' You need to use HTTPS for selecting audio output ' + 'device: ' + error ; }
if ( callback ) { callback ( false , null , arg ) ; }
console . error ( errorMessage ) ; } ) ; } else { console . warn ( "Dialog: " + dialog . callID + ' Browser does not support output device selection.' ) ; if ( callback ) { callback ( false , null , arg ) ; } } }
2014-12-06 17:40:44 +00:00
$ . verto . dialog . prototype . setState = function ( state ) { var dialog = this ; if ( dialog . state == $ . verto . enum . state . ringing ) { dialog . stopRinging ( ) ; }
if ( dialog . state == state || ! checkStateChange ( dialog . state , state ) ) { console . error ( "Dialog " + dialog . callID + ": INVALID state change from " + dialog . state . name + " to " + state . name ) ; dialog . hangup ( ) ; return false ; }
2015-08-28 22:12:06 +00:00
console . log ( "Dialog " + dialog . callID + ": state change from " + dialog . state . name + " to " + state . name ) ; dialog . lastState = dialog . state ; dialog . state = state ; if ( ! dialog . causeCode ) { dialog . causeCode = 16 ; }
2014-12-06 17:40:44 +00:00
if ( ! dialog . cause ) { dialog . cause = "NORMAL CLEARING" ; }
if ( dialog . callbacks . onDialogState ) { dialog . callbacks . onDialogState ( this ) ; }
2016-03-18 23:21:25 +00:00
switch ( dialog . state ) { case $ . verto . enum . state . early : case $ . verto . enum . state . active : var speaker = dialog . useSpeak ; console . info ( "Using Speaker: " , speaker ) ; if ( speaker && speaker !== "any" && speaker !== "none" ) { setTimeout ( function ( ) { dialog . setAudioPlaybackDevice ( speaker ) ; } , 500 ) ; }
2015-07-21 15:59:17 +00:00
break ; case $ . verto . enum . state . trying : setTimeout ( function ( ) { if ( dialog . state == $ . verto . enum . state . trying ) { dialog . setState ( $ . verto . enum . state . hangup ) ; } } , 30000 ) ; break ; case $ . verto . enum . state . purge : dialog . setState ( $ . verto . enum . state . destroy ) ; break ; case $ . verto . enum . state . hangup : if ( dialog . lastState . val > $ . verto . enum . state . requesting . val && dialog . lastState . val < $ . verto . enum . state . hangup . val ) { dialog . sendMethod ( "verto.bye" , { } ) ; }
2016-05-12 22:18:13 +00:00
dialog . setState ( $ . verto . enum . state . destroy ) ; break ; case $ . verto . enum . state . destroy : if ( typeof ( dialog . verto . options . tag ) === "function" ) { $ ( '#' + dialog . params . tag ) . remove ( ) ; }
delete dialog . verto . dialogs [ dialog . callID ] ; if ( dialog . params . screenShare ) { dialog . rtc . stopPeer ( ) ; } else { dialog . rtc . stop ( ) ; }
2015-02-17 03:21:10 +00:00
break ; }
2014-12-06 17:40:44 +00:00
return true ; } ; $ . verto . dialog . prototype . processReply = function ( method , success , e ) { var dialog = this ; switch ( method ) { case "verto.answer" : case "verto.attach" : if ( success ) { dialog . setState ( $ . verto . enum . state . active ) ; } else { dialog . hangup ( ) ; }
break ; case "verto.invite" : if ( success ) { dialog . setState ( $ . verto . enum . state . trying ) ; } else { dialog . setState ( $ . verto . enum . state . destroy ) ; }
break ; case "verto.bye" : dialog . hangup ( ) ; break ; case "verto.modify" : if ( e . holdState ) { if ( e . holdState == "held" ) { if ( dialog . state != $ . verto . enum . state . held ) { dialog . setState ( $ . verto . enum . state . held ) ; } } else if ( e . holdState == "active" ) { if ( dialog . state != $ . verto . enum . state . active ) { dialog . setState ( $ . verto . enum . state . active ) ; } } }
if ( success ) { }
break ; default : break ; } } ; $ . verto . dialog . prototype . hangup = function ( params ) { var dialog = this ; if ( params ) { if ( params . causeCode ) { dialog . causeCode = params . causeCode ; }
if ( params . cause ) { dialog . cause = params . cause ; } }
2015-03-24 17:49:06 +00:00
if ( dialog . state . val >= $ . verto . enum . state . new . val && dialog . state . val < $ . verto . enum . state . hangup . val ) { dialog . setState ( $ . verto . enum . state . hangup ) ; } else if ( dialog . state . val < $ . verto . enum . state . destroy ) { dialog . setState ( $ . verto . enum . state . destroy ) ; } } ; $ . verto . dialog . prototype . stopRinging = function ( ) { var dialog = this ; if ( dialog . verto . ringer ) { dialog . verto . ringer . stop ( ) ; } } ; $ . verto . dialog . prototype . indicateRing = function ( ) { var dialog = this ; if ( dialog . verto . ringer ) { dialog . verto . ringer . attr ( "src" , dialog . verto . options . ringFile ) [ 0 ] . play ( ) ; setTimeout ( function ( ) { dialog . stopRinging ( ) ; if ( dialog . state == $ . verto . enum . state . ringing ) { dialog . indicateRing ( ) ; } } , dialog . verto . options . ringSleep ) ; } } ; $ . verto . dialog . prototype . ring = function ( ) { var dialog = this ; dialog . setState ( $ . verto . enum . state . ringing ) ; dialog . indicateRing ( ) ; } ; $ . verto . dialog . prototype . useVideo = function ( on ) { var dialog = this ; dialog . params . useVideo = on ; if ( on ) { dialog . videoStream = dialog . audioStream ; } else { dialog . videoStream = null ; }
2016-07-11 03:25:14 +00:00
dialog . rtc . useVideo ( dialog . videoStream , dialog . localVideo ) ; } ; $ . verto . dialog . prototype . setMute = function ( what ) { var dialog = this ; return dialog . rtc . setMute ( what ) ; } ; $ . verto . dialog . prototype . getMute = function ( ) { var dialog = this ; return dialog . rtc . getMute ( ) ; } ; $ . verto . dialog . prototype . setVideoMute = function ( what ) { var dialog = this ; return dialog . rtc . setVideoMute ( what ) ; } ; $ . verto . dialog . prototype . getVideoMute = function ( ) { var dialog = this ; return dialog . rtc . getVideoMute ( ) ; } ; $ . verto . dialog . prototype . useStereo = function ( on ) { var dialog = this ; dialog . params . useStereo = on ; dialog . rtc . useStereo ( on ) ; } ; $ . verto . dialog . prototype . dtmf = function ( digits ) { var dialog = this ; if ( digits ) { dialog . sendMethod ( "verto.info" , { dtmf : digits } ) ; } } ; $ . verto . dialog . prototype . rtt = function ( obj ) { var dialog = this ; var pobj = { } ; if ( ! obj ) { return false ; }
pobj . code = obj . code ; pobj . chars = obj . chars ; if ( pobj . chars || pobj . code ) { dialog . sendMethod ( "verto.info" , { txt : obj , noDialogParams : true } ) ; } } ; $ . verto . dialog . prototype . transfer = function ( dest , params ) { var dialog = this ; if ( dest ) { dialog . sendMethod ( "verto.modify" , { action : "transfer" , destination : dest , params : params } ) ; } } ; $ . verto . dialog . prototype . hold = function ( params ) { var dialog = this ; dialog . sendMethod ( "verto.modify" , { action : "hold" , params : params } ) ; } ; $ . verto . dialog . prototype . unhold = function ( params ) { var dialog = this ; dialog . sendMethod ( "verto.modify" , { action : "unhold" , params : params } ) ; } ; $ . verto . dialog . prototype . toggleHold = function ( params ) { var dialog = this ; dialog . sendMethod ( "verto.modify" , { action : "toggleHold" , params : params } ) ; } ; $ . verto . dialog . prototype . message = function ( msg ) { var dialog = this ; var err = 0 ; msg . from = dialog . params . login ; if ( ! msg . to ) { console . error ( "Missing To" ) ; err ++ ; }
2014-12-06 17:40:44 +00:00
if ( ! msg . body ) { console . error ( "Missing Body" ) ; err ++ ; }
if ( err ) { return false ; }
2015-02-19 23:08:36 +00:00
dialog . sendMethod ( "verto.info" , { msg : msg } ) ; return true ; } ; $ . verto . dialog . prototype . answer = function ( params ) { var dialog = this ; if ( ! dialog . answered ) { if ( ! params ) { params = { } ; }
2015-08-25 21:04:34 +00:00
params . sdp = dialog . params . sdp ; if ( params ) { if ( params . useVideo ) { dialog . useVideo ( true ) ; }
2015-08-28 14:42:37 +00:00
dialog . params . callee _id _name = params . callee _id _name ; dialog . params . callee _id _number = params . callee _id _number ; if ( params . useCamera ) { dialog . useCamera = params . useCamera ; }
if ( params . useMic ) { dialog . useMic = params . useMic ; }
if ( params . useSpeak ) { dialog . useSpeak = params . useSpeak ; } }
2015-02-19 23:08:36 +00:00
dialog . rtc . createAnswer ( params ) ; dialog . answered = true ; } } ; $ . verto . dialog . prototype . handleAnswer = function ( params ) { var dialog = this ; dialog . gotAnswer = true ; if ( dialog . state . val >= $ . verto . enum . state . active . val ) { return ; }
2016-07-11 03:25:14 +00:00
if ( dialog . state . val >= $ . verto . enum . state . early . val ) { dialog . setState ( $ . verto . enum . state . active ) ; } else { if ( dialog . gotEarly ) { console . log ( "Dialog " + dialog . callID + " Got answer while still establishing early media, delaying..." ) ; } else { console . log ( "Dialog " + dialog . callID + " Answering Channel" ) ; dialog . rtc . answer ( params . sdp , function ( ) { dialog . setState ( $ . verto . enum . state . active ) ; } , function ( e ) { console . error ( e ) ; dialog . hangup ( ) ; } ) ; console . log ( "Dialog " + dialog . callID + "ANSWER SDP" , params . sdp ) ; } } } ; $ . verto . dialog . prototype . cidString = function ( enc ) { var dialog = this ; var party = dialog . params . remote _caller _id _name + ( enc ? " <" : " <" ) + dialog . params . remote _caller _id _number + ( enc ? ">" : ">" ) ; return party ; } ; $ . verto . dialog . prototype . sendMessage = function ( msg , params ) { var dialog = this ; if ( dialog . callbacks . onMessage ) { dialog . callbacks . onMessage ( dialog . verto , dialog , msg , params ) ; } } ; $ . verto . dialog . prototype . handleInfo = function ( params ) { var dialog = this ; dialog . sendMessage ( $ . verto . enum . message . info , params ) ; } ; $ . verto . dialog . prototype . handleDisplay = function ( params ) { var dialog = this ; if ( params . display _name ) { dialog . params . remote _caller _id _name = params . display _name ; }
2014-12-06 17:40:44 +00:00
if ( params . display _number ) { dialog . params . remote _caller _id _number = params . display _number ; }
dialog . sendMessage ( $ . verto . enum . message . display , { } ) ; } ; $ . verto . dialog . prototype . handleMedia = function ( params ) { var dialog = this ; if ( dialog . state . val >= $ . verto . enum . state . early . val ) { return ; }
2015-08-28 14:42:37 +00:00
dialog . gotEarly = true ; dialog . rtc . answer ( params . sdp , function ( ) { console . log ( "Dialog " + dialog . callID + "Establishing early media" ) ; dialog . setState ( $ . verto . enum . state . early ) ; if ( dialog . gotAnswer ) { console . log ( "Dialog " + dialog . callID + "Answering Channel" ) ; dialog . setState ( $ . verto . enum . state . active ) ; } } , function ( e ) { console . error ( e ) ; dialog . hangup ( ) ; } ) ; console . log ( "Dialog " + dialog . callID + "EARLY SDP" , params . sdp ) ; } ; $ . verto . ENUM = function ( s ) { var i = 0 , o = { } ; s . split ( " " ) . map ( function ( x ) { o [ x ] = { name : x , val : i ++ } ; } ) ; return Object . freeze ( o ) ; } ; $ . verto . enum = { } ; $ . verto . enum . states = Object . freeze ( { new : { requesting : 1 , recovering : 1 , ringing : 1 , destroy : 1 , answering : 1 , hangup : 1 } , requesting : { trying : 1 , hangup : 1 , active : 1 } , recovering : { answering : 1 , hangup : 1 } , trying : { active : 1 , early : 1 , hangup : 1 } , ringing : { answering : 1 , hangup : 1 } , answering : { active : 1 , hangup : 1 } , active : { answering : 1 , requesting : 1 , hangup : 1 , held : 1 } , held : { hangup : 1 , active : 1 } , early : { hangup : 1 , active : 1 } , hangup : { destroy : 1 } , destroy : { } , purge : { destroy : 1 } } ) ; $ . verto . enum . state = $ . verto . ENUM ( "new requesting trying recovering ringing answering early active held hangup destroy purge" ) ; $ . verto . enum . direction = $ . verto . ENUM ( "inbound outbound" ) ; $ . verto . enum . message = $ . verto . ENUM ( "display info pvtEvent" ) ; $ . verto . enum = Object . freeze ( $ . verto . enum ) ; $ . verto . saved = [ ] ; $ . verto . unloadJobs = [ ] ; $ ( window ) . bind ( 'beforeunload' , function ( ) { for ( var f in $ . verto . unloadJobs ) { $ . verto . unloadJobs [ f ] ( ) ; }
2016-10-10 19:02:33 +00:00
if ( $ . verto . haltClosure )
return $ . verto . haltClosure ( ) ; for ( var i in $ . verto . saved ) { var verto = $ . verto . saved [ i ] ; if ( verto ) { verto . purge ( ) ; verto . logout ( ) ; } }
2015-07-21 15:59:17 +00:00
return $ . verto . warnOnUnload ; } ) ; $ . verto . videoDevices = [ ] ; $ . verto . audioInDevices = [ ] ; $ . verto . audioOutDevices = [ ] ; var checkDevices = function ( runtime ) { console . info ( "enumerating devices" ) ; var aud _in = [ ] , aud _out = [ ] , vid = [ ] ; if ( ( ! navigator . mediaDevices || ! navigator . mediaDevices . enumerateDevices ) && MediaStreamTrack . getSources ) { MediaStreamTrack . getSources ( function ( media _sources ) { for ( var i = 0 ; i < media _sources . length ; i ++ ) { if ( media _sources [ i ] . kind == 'video' ) { vid . push ( media _sources [ i ] ) ; } else { aud _in . push ( media _sources [ i ] ) ; } }
2015-09-26 01:00:03 +00:00
$ . verto . videoDevices = vid ; $ . verto . audioInDevices = aud _in ; console . info ( "Audio Devices" , $ . verto . audioInDevices ) ; console . info ( "Video Devices" , $ . verto . videoDevices ) ; runtime ( true ) ; } ) ; } else { if ( ! navigator . mediaDevices || ! navigator . mediaDevices . enumerateDevices ) { console . log ( "enumerateDevices() not supported." ) ; return ; }
navigator . mediaDevices . enumerateDevices ( ) . then ( function ( devices ) { devices . forEach ( function ( device ) { console . log ( device ) ; console . log ( device . kind + ": " + device . label + " id = " + device . deviceId ) ; if ( device . kind === "videoinput" ) { vid . push ( { id : device . deviceId , kind : "video" , label : device . label } ) ; } else if ( device . kind === "audioinput" ) { aud _in . push ( { id : device . deviceId , kind : "audio_in" , label : device . label } ) ; } else if ( device . kind === "audiooutput" ) { aud _out . push ( { id : device . deviceId , kind : "audio_out" , label : device . label } ) ; } } ) ; $ . verto . videoDevices = vid ; $ . verto . audioInDevices = aud _in ; $ . verto . audioOutDevices = aud _out ; console . info ( "Audio IN Devices" , $ . verto . audioInDevices ) ; console . info ( "Audio Out Devices" , $ . verto . audioOutDevices ) ; console . info ( "Video Devices" , $ . verto . videoDevices ) ; runtime ( true ) ; } ) . catch ( function ( err ) { console . log ( " Device Enumeration ERROR: " + err . name + ": " + err . message ) ; runtime ( false ) ; } ) ; } } ; $ . verto . refreshDevices = function ( runtime ) { checkDevices ( runtime ) ; }
$ . verto . init = function ( obj , runtime ) { if ( ! obj ) { obj = { } ; }
if ( ! obj . skipPermCheck && ! obj . skipDeviceCheck ) { $ . FSRTC . checkPerms ( function ( status ) { checkDevices ( runtime ) ; } , true , true ) ; } else if ( obj . skipPermCheck && ! obj . skipDeviceCheck ) { checkDevices ( runtime ) ; } else if ( ! obj . skipPermCheck && obj . skipDeviceCheck ) { $ . FSRTC . checkPerms ( function ( status ) { runtime ( status ) ; } , true , true ) ; } else { runtime ( null ) ; } }
2016-06-08 21:20:18 +00:00
$ . verto . genUUID = function ( ) { return generateGUID ( ) ; } } ) ( jQuery ) ; var AdapterJS = AdapterJS || { } ; if ( typeof exports !== 'undefined' ) { module . exports = AdapterJS ; }
AdapterJS . options = AdapterJS . options || { } ; AdapterJS . VERSION = '0.13.3' ; AdapterJS . onwebrtcready = AdapterJS . onwebrtcready || function ( isUsingPlugin ) { } ; AdapterJS . _onwebrtcreadies = [ ] ; AdapterJS . webRTCReady = function ( callback ) { if ( typeof callback !== 'function' ) { throw new Error ( 'Callback provided is not a function' ) ; }
if ( true === AdapterJS . onwebrtcreadyDone ) { callback ( null !== AdapterJS . WebRTCPlugin . plugin ) ; } else { AdapterJS . _onwebrtcreadies . push ( callback ) ; } } ; AdapterJS . WebRTCPlugin = AdapterJS . WebRTCPlugin || { } ; AdapterJS . WebRTCPlugin . pluginInfo = { prefix : 'Tem' , plugName : 'TemWebRTCPlugin' , pluginId : 'plugin0' , type : 'application/x-temwebrtcplugin' , onload : '__TemWebRTCReady0' , portalLink : 'http://skylink.io/plugin/' , downloadLink : null , companyName : 'Temasys' } ; if ( ! ! navigator . platform . match ( /^Mac/i ) ) { AdapterJS . WebRTCPlugin . pluginInfo . downloadLink = 'http://bit.ly/1n77hco' ; }
else if ( ! ! navigator . platform . match ( /^Win/i ) ) { AdapterJS . WebRTCPlugin . pluginInfo . downloadLink = 'http://bit.ly/1kkS4FN' ; }
AdapterJS . WebRTCPlugin . TAGS = { NONE : 'none' , AUDIO : 'audio' , VIDEO : 'video' } ; AdapterJS . WebRTCPlugin . pageId = Math . random ( ) . toString ( 36 ) . slice ( 2 ) ; AdapterJS . WebRTCPlugin . plugin = null ; AdapterJS . WebRTCPlugin . setLogLevel = null ; AdapterJS . WebRTCPlugin . defineWebRTCInterface = null ; AdapterJS . WebRTCPlugin . isPluginInstalled = null ; AdapterJS . WebRTCPlugin . pluginInjectionInterval = null ; AdapterJS . WebRTCPlugin . injectPlugin = null ; AdapterJS . WebRTCPlugin . PLUGIN _STATES = { NONE : 0 , INITIALIZING : 1 , INJECTING : 2 , INJECTED : 3 , READY : 4 } ; AdapterJS . WebRTCPlugin . pluginState = AdapterJS . WebRTCPlugin . PLUGIN _STATES . NONE ; AdapterJS . onwebrtcreadyDone = false ; AdapterJS . WebRTCPlugin . PLUGIN _LOG _LEVELS = { NONE : 'NONE' , ERROR : 'ERROR' , WARNING : 'WARNING' , INFO : 'INFO' , VERBOSE : 'VERBOSE' , SENSITIVE : 'SENSITIVE' } ; AdapterJS . WebRTCPlugin . WaitForPluginReady = null ; AdapterJS . WebRTCPlugin . callWhenPluginReady = null ; _ _TemWebRTCReady0 = function ( ) { if ( document . readyState === 'complete' ) { AdapterJS . WebRTCPlugin . pluginState = AdapterJS . WebRTCPlugin . PLUGIN _STATES . READY ; AdapterJS . maybeThroughWebRTCReady ( ) ; } else { var timer = setInterval ( function ( ) { if ( document . readyState === 'complete' ) { clearInterval ( timer ) ; AdapterJS . WebRTCPlugin . pluginState = AdapterJS . WebRTCPlugin . PLUGIN _STATES . READY ; AdapterJS . maybeThroughWebRTCReady ( ) ; } } , 100 ) ; } } ; AdapterJS . maybeThroughWebRTCReady = function ( ) { if ( ! AdapterJS . onwebrtcreadyDone ) { AdapterJS . onwebrtcreadyDone = true ; if ( AdapterJS . _onwebrtcreadies . length ) { AdapterJS . _onwebrtcreadies . forEach ( function ( callback ) { if ( typeof ( callback ) === 'function' ) { callback ( AdapterJS . WebRTCPlugin . plugin !== null ) ; } } ) ; } else if ( typeof ( AdapterJS . onwebrtcready ) === 'function' ) { AdapterJS . onwebrtcready ( AdapterJS . WebRTCPlugin . plugin !== null ) ; } } } ; AdapterJS . TEXT = { PLUGIN : { REQUIRE _INSTALLATION : 'This website requires you to install a WebRTC-enabling plugin ' + 'to work on this browser.' , NOT _SUPPORTED : 'Your browser does not support WebRTC.' , BUTTON : 'Install Now' } , REFRESH : { REQUIRE _REFRESH : 'Please refresh page' , BUTTON : 'Refresh Page' } } ; AdapterJS . _iceConnectionStates = { starting : 'starting' , checking : 'checking' , connected : 'connected' , completed : 'connected' , done : 'completed' , disconnected : 'disconnected' , failed : 'failed' , closed : 'closed' } ; AdapterJS . _iceConnectionFiredStates = [ ] ; AdapterJS . isDefined = null ; AdapterJS . parseWebrtcDetectedBrowser = function ( ) { var hasMatch = null ; if ( ( ! ! window . opr && ! ! opr . addons ) || ! ! window . opera || navigator . userAgent . indexOf ( ' OPR/' ) >= 0 ) { webrtcDetectedBrowser = 'opera' ; webrtcDetectedType = 'webkit' ; webrtcMinimumVersion = 26 ; hasMatch = /OPR\/(\d+)/i . exec ( navigator . userAgent ) || [ ] ; webrtcDetectedVersion = parseInt ( hasMatch [ 1 ] , 10 ) ; } else if ( typeof InstallTrigger !== 'undefined' ) { webrtcDetectedType = 'moz' ; } else if ( Object . prototype . toString . call ( window . HTMLElement ) . indexOf ( 'Constructor' ) > 0 ) { webrtcDetectedBrowser = 'safari' ; webrtcDetectedType = 'plugin' ; webrtcMinimumVersion = 7 ; hasMatch = /version\/(\d+)/i . exec ( navigator . userAgent ) || [ ] ; webrtcDetectedVersion = parseInt ( hasMatch [ 1 ] , 10 ) ; } else if ( false || ! ! document . documentMode ) { webrtcDetectedBrowser = 'IE' ; webrtcDetectedType = 'plugin' ; webrtcMinimumVersion = 9 ; hasMatch = /\brv[ :]+(\d+)/g . exec ( navigator . userAgent ) || [ ] ; webrtcDetectedVersion = parseInt ( hasMatch [ 1 ] || '0' , 10 ) ; if ( ! webrtcDetectedVersion ) { hasMatch = /\bMSIE[ :]+(\d+)/g . exec ( navigator . userAgent ) || [ ] ; webrtcDetectedVersion = parseInt ( hasMatch [ 1 ] || '0' , 10 ) ; } } else if ( ! ! window . StyleMedia ) { webrtcDetectedType = '' ; } else if ( ! ! window . chrome && ! ! window . chrome . webstore ) { webrtcDetectedType = 'webkit' ; } else if ( ( webrtcDetectedBrowser === 'chrome' || webrtcDetectedBrowser === 'opera' ) && ! ! window . CSS ) { webrtcDetectedBrowser = 'blink' ; }
window . webrtcDetectedBrowser = webrtcDetectedBrowser ; window . webrtcDetectedVersion = webrtcDetectedVersion ; window . webrtcMinimumVersion = webrtcMinimumVersion ; } ; AdapterJS . addEvent = function ( elem , evnt , func ) { if ( elem . addEventListener ) { elem . addEventListener ( evnt , func , false ) ; } else if ( elem . attachEvent ) { elem . attachEvent ( 'on' + evnt , func ) ; } else { elem [ evnt ] = func ; } } ; AdapterJS . renderNotificationBar = function ( text , buttonText , buttonLink , openNewTab , displayRefreshBar ) { if ( document . readyState !== 'complete' ) { return ; }
var w = window ; var i = document . createElement ( 'iframe' ) ; i . name = 'adapterjs-alert' ; i . style . position = 'fixed' ; i . style . top = '-41px' ; i . style . left = 0 ; i . style . right = 0 ; i . style . width = '100%' ; i . style . height = '40px' ; i . style . backgroundColor = '#ffffe1' ; i . style . border = 'none' ; i . style . borderBottom = '1px solid #888888' ; i . style . zIndex = '9999999' ; if ( typeof i . style . webkitTransition === 'string' ) { i . style . webkitTransition = 'all .5s ease-out' ; } else if ( typeof i . style . transition === 'string' ) { i . style . transition = 'all .5s ease-out' ; }
document . body . appendChild ( i ) ; var c = ( i . contentWindow ) ? i . contentWindow : ( i . contentDocument . document ) ? i . contentDocument . document : i . contentDocument ; c . document . open ( ) ; c . document . write ( '<span style="display: inline-block; font-family: Helvetica, Arial,' + 'sans-serif; font-size: .9rem; padding: 4px; vertical-align: ' + 'middle; cursor: default;">' + text + '</span>' ) ; if ( buttonText && buttonLink ) { c . document . write ( '<button id="okay">' + buttonText + '</button><button id="cancel">Cancel</button>' ) ; c . document . close ( ) ; AdapterJS . addEvent ( c . document . getElementById ( 'okay' ) , 'click' , function ( e ) { if ( ! ! displayRefreshBar ) { AdapterJS . renderNotificationBar ( AdapterJS . TEXT . EXTENSION ? AdapterJS . TEXT . EXTENSION . REQUIRE _REFRESH : AdapterJS . TEXT . REFRESH . REQUIRE _REFRESH , AdapterJS . TEXT . REFRESH . BUTTON , 'javascript:location.reload()' ) ; }
window . open ( buttonLink , ! ! openNewTab ? '_blank' : '_top' ) ; e . preventDefault ( ) ; try { e . cancelBubble = true ; } catch ( error ) { }
var pluginInstallInterval = setInterval ( function ( ) { if ( ! isIE ) { navigator . plugins . refresh ( false ) ; }
AdapterJS . WebRTCPlugin . isPluginInstalled ( AdapterJS . WebRTCPlugin . pluginInfo . prefix , AdapterJS . WebRTCPlugin . pluginInfo . plugName , function ( ) { clearInterval ( pluginInstallInterval ) ; AdapterJS . WebRTCPlugin . defineWebRTCInterface ( ) ; } , function ( ) { } ) ; } , 500 ) ; } ) ; AdapterJS . addEvent ( c . document . getElementById ( 'cancel' ) , 'click' , function ( e ) { w . document . body . removeChild ( i ) ; } ) ; } else { c . document . close ( ) ; }
setTimeout ( function ( ) { if ( typeof i . style . webkitTransform === 'string' ) { i . style . webkitTransform = 'translateY(40px)' ; } else if ( typeof i . style . transform === 'string' ) { i . style . transform = 'translateY(40px)' ; } else { i . style . top = '0px' ; } } , 300 ) ; } ; webrtcDetectedType = null ; checkMediaDataChannelSettings = function ( peerBrowserAgent , peerBrowserVersion , callback , constraints ) { if ( typeof callback !== 'function' ) { return ; }
var beOfferer = true ; var isLocalFirefox = webrtcDetectedBrowser === 'firefox' ; var isLocalFirefoxInterop = webrtcDetectedType === 'moz' && webrtcDetectedVersion > 30 ; var isPeerFirefox = peerBrowserAgent === 'firefox' ; var isPeerFirefoxInterop = peerBrowserAgent === 'firefox' && ( ( peerBrowserVersion ) ? ( peerBrowserVersion > 30 ) : false ) ; if ( ( isLocalFirefox && isPeerFirefox ) || ( isLocalFirefoxInterop ) ) { try { delete constraints . mandatory . MozDontOfferDataChannel ; } catch ( error ) { console . error ( 'Failed deleting MozDontOfferDataChannel' ) ; console . error ( error ) ; } } else if ( ( isLocalFirefox && ! isPeerFirefox ) ) { constraints . mandatory . MozDontOfferDataChannel = true ; }
if ( ! isLocalFirefox ) { for ( var prop in constraints . mandatory ) { if ( constraints . mandatory . hasOwnProperty ( prop ) ) { if ( prop . indexOf ( 'Moz' ) !== - 1 ) { delete constraints . mandatory [ prop ] ; } } } }
if ( isLocalFirefox && ! isPeerFirefox && ! isLocalFirefoxInterop ) { beOfferer = false ; }
callback ( beOfferer , constraints ) ; } ; checkIceConnectionState = function ( peerId , iceConnectionState , callback ) { if ( typeof callback !== 'function' ) { console . warn ( 'No callback specified in checkIceConnectionState. Aborted.' ) ; return ; }
peerId = ( peerId ) ? peerId : 'peer' ; if ( ! AdapterJS . _iceConnectionFiredStates [ peerId ] || iceConnectionState === AdapterJS . _iceConnectionStates . disconnected || iceConnectionState === AdapterJS . _iceConnectionStates . failed || iceConnectionState === AdapterJS . _iceConnectionStates . closed ) { AdapterJS . _iceConnectionFiredStates [ peerId ] = [ ] ; }
iceConnectionState = AdapterJS . _iceConnectionStates [ iceConnectionState ] ; if ( AdapterJS . _iceConnectionFiredStates [ peerId ] . indexOf ( iceConnectionState ) < 0 ) { AdapterJS . _iceConnectionFiredStates [ peerId ] . push ( iceConnectionState ) ; if ( iceConnectionState === AdapterJS . _iceConnectionStates . connected ) { setTimeout ( function ( ) { AdapterJS . _iceConnectionFiredStates [ peerId ] . push ( AdapterJS . _iceConnectionStates . done ) ; callback ( AdapterJS . _iceConnectionStates . done ) ; } , 1000 ) ; }
callback ( iceConnectionState ) ; }
return ; } ; createIceServer = null ; createIceServers = null ; RTCPeerConnection = null ; RTCSessionDescription = ( typeof RTCSessionDescription === 'function' ) ? RTCSessionDescription : null ; RTCIceCandidate = ( typeof RTCIceCandidate === 'function' ) ? RTCIceCandidate : null ; getUserMedia = null ; attachMediaStream = null ; reattachMediaStream = null ; webrtcDetectedBrowser = null ; webrtcDetectedVersion = null ; webrtcMinimumVersion = null ; if ( navigator . mozGetUserMedia || navigator . webkitGetUserMedia || ( navigator . mediaDevices && navigator . userAgent . match ( /Edge\/(\d+).(\d+)$/ ) ) ) { ( function ( f ) { if ( typeof exports === "object" && typeof module !== "undefined" ) { module . exports = f ( ) } else if ( typeof define === "function" && define . amd ) { define ( [ ] , f ) } else { var g ; if ( typeof window !== "undefined" ) { g = window } else if ( typeof global !== "undefined" ) { g = global } else if ( typeof self !== "undefined" ) { g = self } else { g = this } g . adapter = f ( ) } } ) ( function ( ) { var define , module , exports ; return ( function e ( t , n , r ) { function s ( o , u ) { if ( ! n [ o ] ) { if ( ! t [ o ] ) { var a = typeof require == "function" && require ; if ( ! u && a ) return a ( o , ! 0 ) ; if ( i ) return i ( o , ! 0 ) ; var f = new Error ( "Cannot find module '" + o + "'" ) ; throw f . code = "MODULE_NOT_FOUND" , f } var l = n [ o ] = { exports : { } } ; t [ o ] [ 0 ] . call ( l . exports , function ( e ) { var n = t [ o ] [ 1 ] [ e ] ; return s ( n ? n : e ) } , l , l . exports , e , t , n , r ) } return n [ o ] . exports } var i = typeof require == "function" && require ; for ( var o = 0 ; o < r . length ; o ++ ) s ( r [ o ] ) ; return s } ) ( { 1 : [ function ( require , module , exports ) { 'use strict' ; var SDPUtils = { } ; SDPUtils . generateIdentifier = function ( ) { return Math . random ( ) . toString ( 36 ) . substr ( 2 , 10 ) ; } ; SDPUtils . localCName = SDPUtils . generateIdentifier ( ) ; SDPUtils . splitLines = function ( blob ) { return blob . trim ( ) . split ( '\n' ) . map ( function ( line ) { return line . trim ( ) ; } ) ; } ; SDPUtils . splitSections = function ( blob ) { var parts = blob . split ( '\nm=' ) ; return parts . map ( function ( part , index ) { return ( index > 0 ? 'm=' + part : part ) . trim ( ) + '\r\n' ; } ) ; } ; SDPUtils . matchPrefix = function ( blob , prefix ) { return SDPUtils . splitLines ( blob ) . filter ( function ( line ) { return line . indexOf ( prefix ) === 0 ; } ) ; } ; SDPUtils . parseCandidate = function ( line ) { var parts ; if ( line . indexOf ( 'a=candidate:' ) === 0 ) { parts = line . substring ( 12 ) . split ( ' ' ) ; } else { parts = line . substring ( 10 ) . split ( ' ' ) ; }
var candidate = { foundation : parts [ 0 ] , component : parts [ 1 ] , protocol : parts [ 2 ] . toLowerCase ( ) , priority : parseInt ( parts [ 3 ] , 10 ) , ip : parts [ 4 ] , port : parseInt ( parts [ 5 ] , 10 ) , type : parts [ 7 ] } ; for ( var i = 8 ; i < parts . length ; i += 2 ) { switch ( parts [ i ] ) { case 'raddr' : candidate . relatedAddress = parts [ i + 1 ] ; break ; case 'rport' : candidate . relatedPort = parseInt ( parts [ i + 1 ] , 10 ) ; break ; case 'tcptype' : candidate . tcpType = parts [ i + 1 ] ; break ; default : break ; } }
return candidate ; } ; SDPUtils . writeCandidate = function ( candidate ) { var sdp = [ ] ; sdp . push ( candidate . foundation ) ; sdp . push ( candidate . component ) ; sdp . push ( candidate . protocol . toUpperCase ( ) ) ; sdp . push ( candidate . priority ) ; sdp . push ( candidate . ip ) ; sdp . push ( candidate . port ) ; var type = candidate . type ; sdp . push ( 'typ' ) ; sdp . push ( type ) ; if ( type !== 'host' && candidate . relatedAddress && candidate . relatedPort ) { sdp . push ( 'raddr' ) ; sdp . push ( candidate . relatedAddress ) ; sdp . push ( 'rport' ) ; sdp . push ( candidate . relatedPort ) ; }
if ( candidate . tcpType && candidate . protocol . toLowerCase ( ) === 'tcp' ) { sdp . push ( 'tcptype' ) ; sdp . push ( candidate . tcpType ) ; }
return 'candidate:' + sdp . join ( ' ' ) ; } ; SDPUtils . parseRtpMap = function ( line ) { var parts = line . substr ( 9 ) . split ( ' ' ) ; var parsed = { payloadType : parseInt ( parts . shift ( ) , 10 ) } ; parts = parts [ 0 ] . split ( '/' ) ; parsed . name = parts [ 0 ] ; parsed . clockRate = parseInt ( parts [ 1 ] , 10 ) ; parsed . numChannels = parts . length === 3 ? parseInt ( parts [ 2 ] , 10 ) : 1 ; return parsed ; } ; SDPUtils . writeRtpMap = function ( codec ) { var pt = codec . payloadType ; if ( codec . preferredPayloadType !== undefined ) { pt = codec . preferredPayloadType ; }
return 'a=rtpmap:' + pt + ' ' + codec . name + '/' + codec . clockRate +
( codec . numChannels !== 1 ? '/' + codec . numChannels : '' ) + '\r\n' ; } ; SDPUtils . parseExtmap = function ( line ) { var parts = line . substr ( 9 ) . split ( ' ' ) ; return { id : parseInt ( parts [ 0 ] , 10 ) , uri : parts [ 1 ] } ; } ; SDPUtils . writeExtmap = function ( headerExtension ) { return 'a=extmap:' + ( headerExtension . id || headerExtension . preferredId ) + ' ' + headerExtension . uri + '\r\n' ; } ; SDPUtils . parseFmtp = function ( line ) { var parsed = { } ; var kv ; var parts = line . substr ( line . indexOf ( ' ' ) + 1 ) . split ( ';' ) ; for ( var j = 0 ; j < parts . length ; j ++ ) { kv = parts [ j ] . trim ( ) . split ( '=' ) ; parsed [ kv [ 0 ] . trim ( ) ] = kv [ 1 ] ; }
return parsed ; } ; SDPUtils . writeFmtp = function ( codec ) { var line = '' ; var pt = codec . payloadType ; if ( codec . preferredPayloadType !== undefined ) { pt = codec . preferredPayloadType ; }
if ( codec . parameters && Object . keys ( codec . parameters ) . length ) { var params = [ ] ; Object . keys ( codec . parameters ) . forEach ( function ( param ) { params . push ( param + '=' + codec . parameters [ param ] ) ; } ) ; line += 'a=fmtp:' + pt + ' ' + params . join ( ';' ) + '\r\n' ; }
return line ; } ; SDPUtils . parseRtcpFb = function ( line ) { var parts = line . substr ( line . indexOf ( ' ' ) + 1 ) . split ( ' ' ) ; return { type : parts . shift ( ) , parameter : parts . join ( ' ' ) } ; } ; SDPUtils . writeRtcpFb = function ( codec ) { var lines = '' ; var pt = codec . payloadType ; if ( codec . preferredPayloadType !== undefined ) { pt = codec . preferredPayloadType ; }
if ( codec . rtcpFeedback && codec . rtcpFeedback . length ) { codec . rtcpFeedback . forEach ( function ( fb ) { lines += 'a=rtcp-fb:' + pt + ' ' + fb . type +
( fb . parameter && fb . parameter . length ? ' ' + fb . parameter : '' ) + '\r\n' ; } ) ; }
return lines ; } ; SDPUtils . parseSsrcMedia = function ( line ) { var sp = line . indexOf ( ' ' ) ; var parts = { ssrc : parseInt ( line . substr ( 7 , sp - 7 ) , 10 ) } ; var colon = line . indexOf ( ':' , sp ) ; if ( colon > - 1 ) { parts . attribute = line . substr ( sp + 1 , colon - sp - 1 ) ; parts . value = line . substr ( colon + 1 ) ; } else { parts . attribute = line . substr ( sp + 1 ) ; }
return parts ; } ; SDPUtils . getDtlsParameters = function ( mediaSection , sessionpart ) { var lines = SDPUtils . splitLines ( mediaSection ) ; lines = lines . concat ( SDPUtils . splitLines ( sessionpart ) ) ; var fpLine = lines . filter ( function ( line ) { return line . indexOf ( 'a=fingerprint:' ) === 0 ; } ) [ 0 ] . substr ( 14 ) ; var dtlsParameters = { role : 'auto' , fingerprints : [ { algorithm : fpLine . split ( ' ' ) [ 0 ] , value : fpLine . split ( ' ' ) [ 1 ] } ] } ; return dtlsParameters ; } ; SDPUtils . writeDtlsParameters = function ( params , setupType ) { var sdp = 'a=setup:' + setupType + '\r\n' ; params . fingerprints . forEach ( function ( fp ) { sdp += 'a=fingerprint:' + fp . algorithm + ' ' + fp . value + '\r\n' ; } ) ; return sdp ; } ; SDPUtils . getIceParameters = function ( mediaSection , sessionpart ) { var lines = SDPUtils . splitLines ( mediaSection ) ; lines = lines . concat ( SDPUtils . splitLines ( sessionpart ) ) ; var iceParameters = { usernameFragment : lines . filter ( function ( line ) { return line . indexOf ( 'a=ice-ufrag:' ) === 0 ; } ) [ 0 ] . substr ( 12 ) , password : lines . filter ( function ( line ) { return line . indexOf ( 'a=ice-pwd:' ) === 0 ; } ) [ 0 ] . substr ( 10 ) } ; return iceParameters ; } ; SDPUtils . writeIceParameters = function ( params ) { return 'a=ice-ufrag:' + params . usernameFragment + '\r\n' + 'a=ice-pwd:' + params . password + '\r\n' ; } ; SDPUtils . parseRtpParameters = function ( mediaSection ) { var description = { codecs : [ ] , headerExtensions : [ ] , fecMechanisms : [ ] , rtcp : [ ] } ; var lines = SDPUtils . splitLines ( mediaSection ) ; var mline = lines [ 0 ] . split ( ' ' ) ; for ( var i = 3 ; i < mline . length ; i ++ ) { var pt = mline [ i ] ; var rtpmapline = SDPUtils . matchPrefix ( mediaSection , 'a=rtpmap:' + pt + ' ' ) [ 0 ] ; if ( rtpmapline ) { var codec = SDPUtils . parseRtpMap ( rtpmapline ) ; var fmtps = SDPUtils . matchPrefix ( mediaSection , 'a=fmtp:' + pt + ' ' ) ; codec . parameters = fmtps . length ? SDPUtils . parseFmtp ( fmtps [ 0 ] ) : { } ; codec . rtcpFeedback = SDPUtils . matchPrefix ( mediaSection , 'a=rtcp-fb:' + pt + ' ' ) . map ( SDPUtils . parseRtcpFb ) ; description . codecs . push ( codec ) ; switch ( codec . name . toUpperCase ( ) ) { case 'RED' : case 'ULPFEC' : description . fecMechanisms . push ( codec . name . toUpperCase ( ) ) ; break ; default : break ; } } }
SDPUtils . matchPrefix ( mediaSection , 'a=extmap:' ) . forEach ( function ( line ) { description . headerExtensions . push ( SDPUtils . parseExtmap ( line ) ) ; } ) ; return description ; } ; SDPUtils . writeRtpDescription = function ( kind , caps ) { var sdp = '' ; sdp += 'm=' + kind + ' ' ; sdp += caps . codecs . length > 0 ? '9' : '0' ; sdp += ' UDP/TLS/RTP/SAVPF ' ; sdp += caps . codecs . map ( function ( codec ) { if ( codec . preferredPayloadType !== undefined ) { return codec . preferredPayloadType ; }
return codec . payloadType ; } ) . join ( ' ' ) + '\r\n' ; sdp += 'c=IN IP4 0.0.0.0\r\n' ; sdp += 'a=rtcp:9 IN IP4 0.0.0.0\r\n' ; caps . codecs . forEach ( function ( codec ) { sdp += SDPUtils . writeRtpMap ( codec ) ; sdp += SDPUtils . writeFmtp ( codec ) ; sdp += SDPUtils . writeRtcpFb ( codec ) ; } ) ; sdp += 'a=rtcp-mux\r\n' ; return sdp ; } ; SDPUtils . parseRtpEncodingParameters = function ( mediaSection ) { var encodingParameters = [ ] ; var description = SDPUtils . parseRtpParameters ( mediaSection ) ; var hasRed = description . fecMechanisms . indexOf ( 'RED' ) !== - 1 ; var hasUlpfec = description . fecMechanisms . indexOf ( 'ULPFEC' ) !== - 1 ; var ssrcs = SDPUtils . matchPrefix ( mediaSection , 'a=ssrc:' ) . map ( function ( line ) { return SDPUtils . parseSsrcMedia ( line ) ; } ) . filter ( function ( parts ) { return parts . attribute === 'cname' ; } ) ; var primarySsrc = ssrcs . length > 0 && ssrcs [ 0 ] . ssrc ; var secondarySsrc ; var flows = SDPUtils . matchPrefix ( mediaSection , 'a=ssrc-group:FID' ) . map ( function ( line ) { var parts = line . split ( ' ' ) ; parts . shift ( ) ; return parts . map ( function ( part ) { return parseInt ( part , 10 ) ; } ) ; } ) ; if ( flows . length > 0 && flows [ 0 ] . length > 1 && flows [ 0 ] [ 0 ] === primarySsrc ) { secondarySsrc = flows [ 0 ] [ 1 ] ; }
description . codecs . forEach ( function ( codec ) { if ( codec . name . toUpperCase ( ) === 'RTX' && codec . parameters . apt ) { var encParam = { ssrc : primarySsrc , codecPayloadType : parseInt ( codec . parameters . apt , 10 ) , rtx : { payloadType : codec . payloadType , ssrc : secondarySsrc } } ; encodingParameters . push ( encParam ) ; if ( hasRed ) { encParam = JSON . parse ( JSON . stringify ( encParam ) ) ; encParam . fec = { ssrc : secondarySsrc , mechanism : hasUlpfec ? 'red+ulpfec' : 'red' } ; encodingParameters . push ( encParam ) ; } } } ) ; if ( encodingParameters . length === 0 && primarySsrc ) { encodingParameters . push ( { ssrc : primarySsrc } ) ; }
var bandwidth = SDPUtils . matchPrefix ( mediaSection , 'b=' ) ; if ( bandwidth . length ) { if ( bandwidth [ 0 ] . indexOf ( 'b=TIAS:' ) === 0 ) { bandwidth = parseInt ( bandwidth [ 0 ] . substr ( 7 ) , 10 ) ; } else if ( bandwidth [ 0 ] . indexOf ( 'b=AS:' ) === 0 ) { bandwidth = parseInt ( bandwidth [ 0 ] . substr ( 5 ) , 10 ) ; }
encodingParameters . forEach ( function ( params ) { params . maxBitrate = bandwidth ; } ) ; }
return encodingParameters ; } ; SDPUtils . writeSessionBoilerplate = function ( ) { return 'v=0\r\n' + 'o=thisisadapterortc 8169639915646943137 2 IN IP4 127.0.0.1\r\n' + 's=-\r\n' + 't=0 0\r\n' ; } ; SDPUtils . writeMediaSection = function ( transceiver , caps , type , stream ) { var sdp = SDPUtils . writeRtpDescription ( transceiver . kind , caps ) ; sdp += SDPUtils . writeIceParameters ( transceiver . iceGatherer . getLocalParameters ( ) ) ; sdp += SDPUtils . writeDtlsParameters ( transceiver . dtlsTransport . getLocalParameters ( ) , type === 'offer' ? 'actpass' : 'active' ) ; sdp += 'a=mid:' + transceiver . mid + '\r\n' ; if ( transceiver . rtpSender && transceiver . rtpReceiver ) { sdp += 'a=sendrecv\r\n' ; } else if ( transceiver . rtpSender ) { sdp += 'a=sendonly\r\n' ; } else if ( transceiver . rtpReceiver ) { sdp += 'a=recvonly\r\n' ; } else { sdp += 'a=inactive\r\n' ; }
if ( transceiver . rtpSender ) { var msid = 'msid:' + stream . id + ' ' +
transceiver . rtpSender . track . id + '\r\n' ; sdp += 'a=' + msid ; sdp += 'a=ssrc:' + transceiver . sendEncodingParameters [ 0 ] . ssrc + ' ' + msid ; }
sdp += 'a=ssrc:' + transceiver . sendEncodingParameters [ 0 ] . ssrc + ' cname:' + SDPUtils . localCName + '\r\n' ; return sdp ; } ; SDPUtils . getDirection = function ( mediaSection , sessionpart ) { var lines = SDPUtils . splitLines ( mediaSection ) ; for ( var i = 0 ; i < lines . length ; i ++ ) { switch ( lines [ i ] ) { case 'a=sendrecv' : case 'a=sendonly' : case 'a=recvonly' : case 'a=inactive' : return lines [ i ] . substr ( 2 ) ; default : } }
if ( sessionpart ) { return SDPUtils . getDirection ( sessionpart ) ; }
return 'sendrecv' ; } ; module . exports = SDPUtils ; } , { } ] , 2 : [ function ( require , module , exports ) { 'use strict' ; ( function ( ) { var logging = require ( './utils' ) . log ; var browserDetails = require ( './utils' ) . browserDetails ; module . exports . browserDetails = browserDetails ; module . exports . extractVersion = require ( './utils' ) . extractVersion ; module . exports . disableLog = require ( './utils' ) . disableLog ; var chromeShim = require ( './chrome/chrome_shim' ) || null ; var edgeShim = require ( './edge/edge_shim' ) || null ; var firefoxShim = require ( './firefox/firefox_shim' ) || null ; var safariShim = require ( './safari/safari_shim' ) || null ; switch ( browserDetails . browser ) { case 'opera' : case 'chrome' : if ( ! chromeShim || ! chromeShim . shimPeerConnection ) { logging ( 'Chrome shim is not included in this adapter release.' ) ; return ; }
logging ( 'adapter.js shimming chrome.' ) ; module . exports . browserShim = chromeShim ; chromeShim . shimGetUserMedia ( ) ; chromeShim . shimMediaStream ( ) ; chromeShim . shimSourceObject ( ) ; chromeShim . shimPeerConnection ( ) ; chromeShim . shimOnTrack ( ) ; break ; case 'firefox' : if ( ! firefoxShim || ! firefoxShim . shimPeerConnection ) { logging ( 'Firefox shim is not included in this adapter release.' ) ; return ; }
logging ( 'adapter.js shimming firefox.' ) ; module . exports . browserShim = firefoxShim ; firefoxShim . shimGetUserMedia ( ) ; firefoxShim . shimSourceObject ( ) ; firefoxShim . shimPeerConnection ( ) ; firefoxShim . shimOnTrack ( ) ; break ; case 'edge' : if ( ! edgeShim || ! edgeShim . shimPeerConnection ) { logging ( 'MS edge shim is not included in this adapter release.' ) ; return ; }
logging ( 'adapter.js shimming edge.' ) ; module . exports . browserShim = edgeShim ; edgeShim . shimGetUserMedia ( ) ; edgeShim . shimPeerConnection ( ) ; break ; case 'safari' : if ( ! safariShim ) { logging ( 'Safari shim is not included in this adapter release.' ) ; return ; }
logging ( 'adapter.js shimming safari.' ) ; module . exports . browserShim = safariShim ; safariShim . shimGetUserMedia ( ) ; break ; default : logging ( 'Unsupported browser!' ) ; } } ) ( ) ; } , { "./chrome/chrome_shim" : 3 , "./edge/edge_shim" : 5 , "./firefox/firefox_shim" : 7 , "./safari/safari_shim" : 9 , "./utils" : 10 } ] , 3 : [ function ( require , module , exports ) { 'use strict' ; var logging = require ( '../utils.js' ) . log ; var browserDetails = require ( '../utils.js' ) . browserDetails ; var chromeShim = { shimMediaStream : function ( ) { window . MediaStream = window . MediaStream || window . webkitMediaStream ; } , shimOnTrack : function ( ) { if ( typeof window === 'object' && window . RTCPeerConnection && ! ( 'ontrack' in
window . RTCPeerConnection . prototype ) ) { Object . defineProperty ( window . RTCPeerConnection . prototype , 'ontrack' , { get : function ( ) { return this . _ontrack ; } , set : function ( f ) { var self = this ; if ( this . _ontrack ) { this . removeEventListener ( 'track' , this . _ontrack ) ; this . removeEventListener ( 'addstream' , this . _ontrackpoly ) ; }
this . addEventListener ( 'track' , this . _ontrack = f ) ; this . addEventListener ( 'addstream' , this . _ontrackpoly = function ( e ) { e . stream . addEventListener ( 'addtrack' , function ( te ) { var event = new Event ( 'track' ) ; event . track = te . track ; event . receiver = { track : te . track } ; event . streams = [ e . stream ] ; self . dispatchEvent ( event ) ; } ) ; e . stream . getTracks ( ) . forEach ( function ( track ) { var event = new Event ( 'track' ) ; event . track = track ; event . receiver = { track : track } ; event . streams = [ e . stream ] ; this . dispatchEvent ( event ) ; } . bind ( this ) ) ; } . bind ( this ) ) ; } } ) ; } } , shimSourceObject : function ( ) { if ( typeof window === 'object' ) { if ( window . HTMLMediaElement && ! ( 'srcObject' in window . HTMLMediaElement . prototype ) ) { Object . defineProperty ( window . HTMLMediaElement . prototype , 'srcObject' , { get : function ( ) { return this . _srcObject ; } , set : function ( stream ) { var self = this ; this . _srcObject = stream ; if ( this . src ) { URL . revokeObjectURL ( this . src ) ; }
if ( ! stream ) { this . src = '' ; return ; }
this . src = URL . createObjectURL ( stream ) ; stream . addEventListener ( 'addtrack' , function ( ) { if ( self . src ) { URL . revokeObjectURL ( self . src ) ; }
self . src = URL . createObjectURL ( stream ) ; } ) ; stream . addEventListener ( 'removetrack' , function ( ) { if ( self . src ) { URL . revokeObjectURL ( self . src ) ; }
self . src = URL . createObjectURL ( stream ) ; } ) ; } } ) ; } } } , shimPeerConnection : function ( ) { window . RTCPeerConnection = function ( pcConfig , pcConstraints ) { logging ( 'PeerConnection' ) ; if ( pcConfig && pcConfig . iceTransportPolicy ) { pcConfig . iceTransports = pcConfig . iceTransportPolicy ; }
var pc = new webkitRTCPeerConnection ( pcConfig , pcConstraints ) ; var origGetStats = pc . getStats . bind ( pc ) ; pc . getStats = function ( selector , successCallback , errorCallback ) { var self = this ; var args = arguments ; if ( arguments . length > 0 && typeof selector === 'function' ) { return origGetStats ( selector , successCallback ) ; }
var fixChromeStats _ = function ( response ) { var standardReport = { } ; var reports = response . result ( ) ; reports . forEach ( function ( report ) { var standardStats = { id : report . id , timestamp : report . timestamp , type : report . type } ; report . names ( ) . forEach ( function ( name ) { standardStats [ name ] = report . stat ( name ) ; } ) ; standardReport [ standardStats . id ] = standardStats ; } ) ; return standardReport ; } ; var makeMapStats = function ( stats , legacyStats ) { var map = new Map ( Object . keys ( stats ) . map ( function ( key ) { return [ key , stats [ key ] ] ; } ) ) ; legacyStats = legacyStats || stats ; Object . keys ( legacyStats ) . forEach ( function ( key ) { map [ key ] = legacyStats [ key ] ; } ) ; return map ; } ; if ( arguments . length >= 2 ) { var successCallbackWrapper _ = function ( response ) { args [ 1 ] ( makeMapStats ( fixChromeStats _ ( response ) ) ) ; } ; return origGetStats . apply ( this , [ successCallbackWrapper _ , arguments [ 0 ] ] ) ; }
return new Promise ( function ( resolve , reject ) { if ( args . length === 1 && typeof selector === 'object' ) { origGetStats . apply ( self , [ function ( response ) { resolve ( makeMapStats ( fixChromeStats _ ( response ) ) ) ; } , reject ] ) ; } else { origGetStats . apply ( self , [ function ( response ) { resolve ( makeMapStats ( fixChromeStats _ ( response ) , response . result ( ) ) ) ; } , reject ] ) ; } } ) . then ( successCallback , errorCallback ) ; } ; return pc ; } ; window . RTCPeerConnection . prototype = webkitRTCPeerConnection . prototype ; if ( webkitRTCPeerConnection . generateCertificate ) { Object . defineProperty ( window . RTCPeerConnection , 'generateCertificate' , { get : function ( ) { return webkitRTCPeerConnection . generateCertificate ; } } ) ; }
[ 'createOffer' , 'createAnswer' ] . forEach ( function ( method ) { var nativeMethod = webkitRTCPeerConnection . prototype [ method ] ; webkitRTCPeerConnection . prototype [ method ] = function ( ) { var self = this ; if ( arguments . length < 1 || ( arguments . length === 1 && typeof arguments [ 0 ] === 'object' ) ) { var opts = arguments . length === 1 ? arguments [ 0 ] : undefined ; return new Promise ( function ( resolve , reject ) { nativeMethod . apply ( self , [ resolve , reject , opts ] ) ; } ) ; }
return nativeMethod . apply ( this , arguments ) ; } ; } ) ; if ( browserDetails . version < 51 ) { [ 'setLocalDescription' , 'setRemoteDescription' , 'addIceCandidate' ] . forEach ( function ( method ) { var nativeMethod = webkitRTCPeerConnection . prototype [ method ] ; webkitRTCPeerConnection . prototype [ method ] = function ( ) { var args = arguments ; var self = this ; var promise = new Promise ( function ( resolve , reject ) { nativeMethod . apply ( self , [ args [ 0 ] , resolve , reject ] ) ; } ) ; if ( args . length < 2 ) { return promise ; }
return promise . then ( function ( ) { args [ 1 ] . apply ( null , [ ] ) ; } , function ( err ) { if ( args . length >= 3 ) { args [ 2 ] . apply ( null , [ err ] ) ; } } ) ; } ; } ) ; }
[ 'setLocalDescription' , 'setRemoteDescription' , 'addIceCandidate' ] . forEach ( function ( method ) { var nativeMethod = webkitRTCPeerConnection . prototype [ method ] ; webkitRTCPeerConnection . prototype [ method ] = function ( ) { arguments [ 0 ] = new ( ( method === 'addIceCandidate' ) ? RTCIceCandidate : RTCSessionDescription ) ( arguments [ 0 ] ) ; return nativeMethod . apply ( this , arguments ) ; } ; } ) ; var nativeAddIceCandidate = RTCPeerConnection . prototype . addIceCandidate ; RTCPeerConnection . prototype . addIceCandidate = function ( ) { return arguments [ 0 ] === null ? Promise . resolve ( ) : nativeAddIceCandidate . apply ( this , arguments ) ; } ; } } ; module . exports = { shimMediaStream : chromeShim . shimMediaStream , shimOnTrack : chromeShim . shimOnTrack , shimSourceObject : chromeShim . shimSourceObject , shimPeerConnection : chromeShim . shimPeerConnection , shimGetUserMedia : require ( './getusermedia' ) } ; } , { "../utils.js" : 10 , "./getusermedia" : 4 } ] , 4 : [ function ( require , module , exports ) { 'use strict' ; var logging = require ( '../utils.js' ) . log ; module . exports = function ( ) { var constraintsToChrome _ = function ( c ) { if ( typeof c !== 'object' || c . mandatory || c . optional ) { return c ; }
var cc = { } ; Object . keys ( c ) . forEach ( function ( key ) { if ( key === 'require' || key === 'advanced' || key === 'mediaSource' ) { return ; }
var r = ( typeof c [ key ] === 'object' ) ? c [ key ] : { ideal : c [ key ] } ; if ( r . exact !== undefined && typeof r . exact === 'number' ) { r . min = r . max = r . exact ; }
var oldname _ = function ( prefix , name ) { if ( prefix ) { return prefix + name . charAt ( 0 ) . toUpperCase ( ) + name . slice ( 1 ) ; }
return ( name === 'deviceId' ) ? 'sourceId' : name ; } ; if ( r . ideal !== undefined ) { cc . optional = cc . optional || [ ] ; var oc = { } ; if ( typeof r . ideal === 'number' ) { oc [ oldname _ ( 'min' , key ) ] = r . ideal ; cc . optional . push ( oc ) ; oc = { } ; oc [ oldname _ ( 'max' , key ) ] = r . ideal ; cc . optional . push ( oc ) ; } else { oc [ oldname _ ( '' , key ) ] = r . ideal ; cc . optional . push ( oc ) ; } }
if ( r . exact !== undefined && typeof r . exact !== 'number' ) { cc . mandatory = cc . mandatory || { } ; cc . mandatory [ oldname _ ( '' , key ) ] = r . exact ; } else { [ 'min' , 'max' ] . forEach ( function ( mix ) { if ( r [ mix ] !== undefined ) { cc . mandatory = cc . mandatory || { } ; cc . mandatory [ oldname _ ( mix , key ) ] = r [ mix ] ; } } ) ; } } ) ; if ( c . advanced ) { cc . optional = ( cc . optional || [ ] ) . concat ( c . advanced ) ; }
return cc ; } ; var shimConstraints _ = function ( constraints , func ) { constraints = JSON . parse ( JSON . stringify ( constraints ) ) ; if ( constraints && constraints . audio ) { constraints . audio = constraintsToChrome _ ( constraints . audio ) ; }
if ( constraints && typeof constraints . video === 'object' ) { var face = constraints . video . facingMode ; face = face && ( ( typeof face === 'object' ) ? face : { ideal : face } ) ; if ( ( face && ( face . exact === 'user' || face . exact === 'environment' || face . ideal === 'user' || face . ideal === 'environment' ) ) && ! ( navigator . mediaDevices . getSupportedConstraints && navigator . mediaDevices . getSupportedConstraints ( ) . facingMode ) ) { delete constraints . video . facingMode ; if ( face . exact === 'environment' || face . ideal === 'environment' ) { return navigator . mediaDevices . enumerateDevices ( ) . then ( function ( devices ) { devices = devices . filter ( function ( d ) { return d . kind === 'videoinput' ; } ) ; var back = devices . find ( function ( d ) { return d . label . toLowerCase ( ) . indexOf ( 'back' ) !== - 1 ; } ) || ( devices . length && devices [ devices . length - 1 ] ) ; if ( back ) { constraints . video . deviceId = face . exact ? { exact : back . deviceId } : { ideal : back . deviceId } ; }
constraints . video = constraintsToChrome _ ( constraints . video ) ; logging ( 'chrome: ' + JSON . stringify ( constraints ) ) ; return func ( constraints ) ; } ) ; } }
constraints . video = constraintsToChrome _ ( constraints . video ) ; }
logging ( 'chrome: ' + JSON . stringify ( constraints ) ) ; return func ( constraints ) ; } ; var shimError _ = function ( e ) { return { name : { PermissionDeniedError : 'NotAllowedError' , ConstraintNotSatisfiedError : 'OverconstrainedError' } [ e . name ] || e . name , message : e . message , constraint : e . constraintName , toString : function ( ) { return this . name + ( this . message && ': ' ) + this . message ; } } ; } ; var getUserMedia _ = function ( constraints , onSuccess , onError ) { shimConstraints _ ( constraints , function ( c ) { navigator . webkitGetUserMedia ( c , onSuccess , function ( e ) { onError ( shimError _ ( e ) ) ; } ) ; } ) ; } ; navigator . getUserMedia = getUserMedia _ ; var getUserMediaPromise _ = function ( constraints ) { return new Promise ( function ( resolve , reject ) { navigator . getUserMedia ( constraints , resolve , reject ) ; } ) ; } ; if ( ! navigator . mediaDevices ) { navigator . mediaDevices = { getUserMedia : getUserMediaPromise _ , enumerateDevices : function ( ) { return new Promise ( function ( resolve ) { var kinds = { audio : 'audioinput' , video : 'videoinput' } ; return MediaStreamTrack . getSources ( function ( devices ) { resolve ( devices . map ( function ( device ) { return { label : device . label , kind : kinds [ device . kind ] , deviceId : device . id , groupId : '' } ; } ) ) ; } ) ; } ) ; } } ; }
if ( ! navigator . mediaDevices . getUserMedia ) { navigator . mediaDevices . getUserMedia = function ( constraints ) { return getUserMediaPromise _ ( constraints ) ; } ; } else { var origGetUserMedia = navigator . mediaDevices . getUserMedia . bind ( navigator . mediaDevices ) ; navigator . mediaDevices . getUserMedia = function ( cs ) { return shimConstraints _ ( cs , function ( c ) { return origGetUserMedia ( c ) . catch ( function ( e ) { return Promise . reject ( shimError _ ( e ) ) ; } ) ; } ) ; } ; }
if ( typeof navigator . mediaDevices . addEventListener === 'undefined' ) { navigator . mediaDevices . addEventListener = function ( ) { logging ( 'Dummy mediaDevices.addEventListener called.' ) ; } ; }
if ( typeof navigator . mediaDevices . removeEventListener === 'undefined' ) { navigator . mediaDevices . removeEventListener = function ( ) { logging ( 'Dummy mediaDevices.removeEventListener called.' ) ; } ; } } ; } , { "../utils.js" : 10 } ] , 5 : [ function ( require , module , exports ) { 'use strict' ; var SDPUtils = require ( 'sdp' ) ; var edgeShim = { shimPeerConnection : function ( ) { if ( window . RTCIceGatherer ) { if ( ! window . RTCIceCandidate ) { window . RTCIceCandidate = function ( args ) { return args ; } ; }
if ( ! window . RTCSessionDescription ) { window . RTCSessionDescription = function ( args ) { return args ; } ; } }
window . RTCPeerConnection = function ( config ) { var self = this ; var _eventTarget = document . createDocumentFragment ( ) ; [ 'addEventListener' , 'removeEventListener' , 'dispatchEvent' ] . forEach ( function ( method ) { self [ method ] = _eventTarget [ method ] . bind ( _eventTarget ) ; } ) ; this . onicecandidate = null ; this . onaddstream = null ; this . ontrack = null ; this . onremovestream = null ; this . onsignalingstatechange = null ; this . oniceconnectionstatechange = null ; this . onnegotiationneeded = null ; this . ondatachannel = null ; this . localStreams = [ ] ; this . remoteStreams = [ ] ; this . getLocalStreams = function ( ) { return self . localStreams ; } ; this . getRemoteStreams = function ( ) { return self . remoteStreams ; } ; this . localDescription = new RTCSessionDescription ( { type : '' , sdp : '' } ) ; this . remoteDescription = new RTCSessionDescription ( { type : '' , sdp : '' } ) ; this . signalingState = 'stable' ; this . iceConnectionState = 'new' ; this . iceGatheringState = 'new' ; this . iceOptions = { gatherPolicy : 'all' , iceServers : [ ] } ; if ( config && config . iceTransportPolicy ) { switch ( config . iceTransportPolicy ) { case 'all' : case 'relay' : this . iceOptions . gatherPolicy = config . iceTransportPolicy ; break ; case 'none' : throw new TypeError ( 'iceTransportPolicy "none" not supported' ) ; default : break ; } }
this . usingBundle = config && config . bundlePolicy === 'max-bundle' ; if ( config && config . iceServers ) { var iceServers = JSON . parse ( JSON . stringify ( config . iceServers ) ) ; this . iceOptions . iceServers = iceServers . filter ( function ( server ) { if ( server && server . urls ) { var urls = server . urls ; if ( typeof urls === 'string' ) { urls = [ urls ] ; }
urls = urls . filter ( function ( url ) { return url . indexOf ( 'turn:' ) === 0 && url . indexOf ( 'transport=udp' ) !== - 1 && url . indexOf ( 'turn:[' ) === - 1 ; } ) [ 0 ] ; return ! ! urls ; }
return false ; } ) ; }
this . transceivers = [ ] ; this . _localIceCandidatesBuffer = [ ] ; } ; window . RTCPeerConnection . prototype . _emitBufferedCandidates = function ( ) { var self = this ; var sections = SDPUtils . splitSections ( self . localDescription . sdp ) ; this . _localIceCandidatesBuffer . forEach ( function ( event ) { var end = ! event . candidate || Object . keys ( event . candidate ) . length === 0 ; if ( end ) { for ( var j = 1 ; j < sections . length ; j ++ ) { if ( sections [ j ] . indexOf ( '\r\na=end-of-candidates\r\n' ) === - 1 ) { sections [ j ] += 'a=end-of-candidates\r\n' ; } } } else if ( event . candidate . candidate . indexOf ( 'typ endOfCandidates' ) === - 1 ) { sections [ event . candidate . sdpMLineIndex + 1 ] += 'a=' + event . candidate . candidate + '\r\n' ; }
self . localDescription . sdp = sections . join ( '' ) ; self . dispatchEvent ( event ) ; if ( self . onicecandidate !== null ) { self . onicecandidate ( event ) ; }
if ( ! event . candidate && self . iceGatheringState !== 'complete' ) { var complete = self . transceivers . every ( function ( transceiver ) { return transceiver . iceGatherer && transceiver . iceGatherer . state === 'completed' ; } ) ; if ( complete ) { self . iceGatheringState = 'complete' ; } } } ) ; this . _localIceCandidatesBuffer = [ ] ; } ; window . RTCPeerConnection . prototype . addStream = function ( stream ) { this . localStreams . push ( stream . clone ( ) ) ; this . _maybeFireNegotiationNeeded ( ) ; } ; window . RTCPeerConnection . prototype . removeStream = function ( stream ) { var idx = this . localStreams . indexOf ( stream ) ; if ( idx > - 1 ) { this . localStreams . splice ( idx , 1 ) ; this . _maybeFireNegotiationNeeded ( ) ; } } ; window . RTCPeerConnection . prototype . getSenders = function ( ) { return this . transceivers . filter ( function ( transceiver ) { return ! ! transceiver . rtpSender ; } ) . map ( function ( transceiver ) { return transceiver . rtpSender ; } ) ; } ; window . RTCPeerConnection . prototype . getReceivers = function ( ) { return this . transceivers . filter ( function ( transceiver ) { return ! ! transceiver . rtpReceiver ; } ) . map ( function ( transceiver ) { return transceiver . rtpReceiver ; } ) ; } ; window . RTCPeerConnection . prototype . _getCommonCapabilities = function ( localCapabilities , remoteCapabilities ) { var commonCapabilities = { codecs : [ ] , headerExtensions : [ ] , fecMechanisms : [ ] } ; localCapabilities . codecs . forEach ( function ( lCodec ) { for ( var i = 0 ; i < remoteCapabilities . codecs . length ; i ++ ) { var rCodec = remoteCapabilities . codecs [ i ] ; if ( lCodec . name . toLowerCase ( ) === rCodec . name . toLowerCase ( ) && lCodec . clockRate === rCodec . clockRate && lCodec . numChannels === rCodec . numChannels ) { commonCapabilities . codecs . push ( rCodec ) ; rCodec . rtcpFeedback = rCodec . rtcpFeedback . filter ( function ( fb ) { for ( var j = 0 ; j < lCodec . rtcpFeedback . length ; j ++ ) { if ( lCodec . rtcpFeedback [ j ] . type === fb . type && lCodec . rtcpFeedback [ j ] . parameter === fb . parameter ) { return true ; } }
return false ; } ) ; break ; } } } ) ; localCapabilities . headerExtensions . forEach ( function ( lHeaderExtension ) { for ( var i = 0 ; i < remoteCapabilities . headerExtensions . length ; i ++ ) { var rHeaderExtension = remoteCapabilities . headerExtensions [ i ] ; if ( lHeaderExtension . uri === rHeaderExtension . uri ) { commonCapabilities . headerExtensions . push ( rHeaderExtension ) ; break ; } } } ) ; return commonCapabilities ; } ; window . RTCPeerConnection . prototype . _createIceAndDtlsTransports = function ( mid , sdpMLineIndex ) { var self = this ; var iceGatherer = new RTCIceGatherer ( self . iceOptions ) ; var iceTransport = new RTCIceTransport ( iceGatherer ) ; iceGatherer . onlocalcandidate = function ( evt ) { var event = new Event ( 'icecandidate' ) ; event . candidate = { sdpMid : mid , sdpMLineIndex : sdpMLineIndex } ; var cand = evt . candidate ; var end = ! cand || Object . keys ( cand ) . length === 0 ; if ( end ) { if ( iceGatherer . state === undefined ) { iceGatherer . state = 'completed' ; }
event . candidate . candidate = 'candidate:1 1 udp 1 0.0.0.0 9 typ endOfCandidates' ; } else { cand . component = iceTransport . component === 'RTCP' ? 2 : 1 ; event . candidate . candidate = SDPUtils . writeCandidate ( cand ) ; }
var sections = SDPUtils . splitSections ( self . localDescription . sdp ) ; if ( event . candidate . candidate . indexOf ( 'typ endOfCandidates' ) === - 1 ) { sections [ event . candidate . sdpMLineIndex + 1 ] += 'a=' + event . candidate . candidate + '\r\n' ; } else { sections [ event . candidate . sdpMLineIndex + 1 ] += 'a=end-of-candidates\r\n' ; }
self . localDescription . sdp = sections . join ( '' ) ; var complete = self . transceivers . every ( function ( transceiver ) { return transceiver . iceGatherer && transceiver . iceGatherer . state === 'completed' ; } ) ; switch ( self . iceGatheringState ) { case 'new' : self . _localIceCandidatesBuffer . push ( event ) ; if ( end && complete ) { self . _localIceCandidatesBuffer . push ( new Event ( 'icecandidate' ) ) ; }
break ; case 'gathering' : self . _emitBufferedCandidates ( ) ; self . dispatchEvent ( event ) ; if ( self . onicecandidate !== null ) { self . onicecandidate ( event ) ; }
if ( complete ) { self . dispatchEvent ( new Event ( 'icecandidate' ) ) ; if ( self . onicecandidate !== null ) { self . onicecandidate ( new Event ( 'icecandidate' ) ) ; }
self . iceGatheringState = 'complete' ; }
break ; case 'complete' : break ; default : break ; } } ; iceTransport . onicestatechange = function ( ) { self . _updateConnectionState ( ) ; } ; var dtlsTransport = new RTCDtlsTransport ( iceTransport ) ; dtlsTransport . ondtlsstatechange = function ( ) { self . _updateConnectionState ( ) ; } ; dtlsTransport . onerror = function ( ) { dtlsTransport . state = 'failed' ; self . _updateConnectionState ( ) ; } ; return { iceGatherer : iceGatherer , iceTransport : iceTransport , dtlsTransport : dtlsTransport } ; } ; window . RTCPeerConnection . prototype . _transceive = function ( transceiver , send , recv ) { var params = this . _getCommonCapabilities ( transceiver . localCapabilities , transceiver . remoteCapabilities ) ; if ( send && transceiver . rtpSender ) { params . encodings = transceiver . sendEncodingParameters ; params . rtcp = { cname : SDPUtils . localCName } ; if ( transceiver . recvEncodingParameters . length ) { params . rtcp . ssrc = transceiver . recvEncodingParameters [ 0 ] . ssrc ; }
transceiver . rtpSender . send ( params ) ; }
if ( recv && transceiver . rtpReceiver ) { params . encodings = transceiver . recvEncodingParameters ; params . rtcp = { cname : transceiver . cname } ; if ( transceiver . sendEncodingParameters . length ) { params . rtcp . ssrc = transceiver . sendEncodingParameters [ 0 ] . ssrc ; }
transceiver . rtpReceiver . receive ( params ) ; } } ; window . RTCPeerConnection . prototype . setLocalDescription = function ( description ) { var self = this ; var sections ; var sessionpart ; if ( description . type === 'offer' ) { if ( this . _pendingOffer ) { sections = SDPUtils . splitSections ( description . sdp ) ; sessionpart = sections . shift ( ) ; sections . forEach ( function ( mediaSection , sdpMLineIndex ) { var caps = SDPUtils . parseRtpParameters ( mediaSection ) ; self . _pendingOffer [ sdpMLineIndex ] . localCapabilities = caps ; } ) ; this . transceivers = this . _pendingOffer ; delete this . _pendingOffer ; } } else if ( description . type === 'answer' ) { sections = SDPUtils . splitSections ( self . remoteDescription . sdp ) ; sessionpart = sections . shift ( ) ; var isIceLite = SDPUtils . matchPrefix ( sessionpart , 'a=ice-lite' ) . length > 0 ; sections . forEach ( function ( mediaSection , sdpMLineIndex ) { var transceiver = self . transceivers [ sdpMLineIndex ] ; var iceGatherer = transceiver . iceGatherer ; var iceTransport = transceiver . iceTransport ; var dtlsTransport = transceiver . dtlsTransport ; var localCapabilities = transceiver . localCapabilities ; var remoteCapabilities = transceiver . remoteCapabilities ; var rejected = mediaSection . split ( '\n' , 1 ) [ 0 ] . split ( ' ' , 2 ) [ 1 ] === '0' ; if ( ! rejected ) { var remoteIceParameters = SDPUtils . getIceParameters ( mediaSection , sessionpart ) ; if ( isIceLite ) { var cands = SDPUtils . matchPrefix ( mediaSection , 'a=candidate:' ) . map ( function ( cand ) { return SDPUtils . parseCandidate ( cand ) ; } ) . filter ( function ( cand ) { return cand . component === '1' ; } ) ; if ( cands . length ) { iceTransport . setRemoteCandidates ( cands ) ; } }
var remoteDtlsParameters = SDPUtils . getDtlsParameters ( mediaSection , sessionpart ) ; if ( isIceLite ) { remoteDtlsParameters . role = 'server' ; }
if ( ! self . usingBundle || sdpMLineIndex === 0 ) { iceTransport . start ( iceGatherer , remoteIceParameters , isIceLite ? 'controlling' : 'controlled' ) ; dtlsTransport . start ( remoteDtlsParameters ) ; }
var params = self . _getCommonCapabilities ( localCapabilities , remoteCapabilities ) ; self . _transceive ( transceiver , params . codecs . length > 0 , false ) ; } } ) ; }
this . localDescription = { type : description . type , sdp : description . sdp } ; switch ( description . type ) { case 'offer' : this . _updateSignalingState ( 'have-local-offer' ) ; break ; case 'answer' : this . _updateSignalingState ( 'stable' ) ; break ; default : throw new TypeError ( 'unsupported type "' + description . type + '"' ) ; }
var hasCallback = arguments . length > 1 && typeof arguments [ 1 ] === 'function' ; if ( hasCallback ) { var cb = arguments [ 1 ] ; window . setTimeout ( function ( ) { cb ( ) ; if ( self . iceGatheringState === 'new' ) { self . iceGatheringState = 'gathering' ; }
self . _emitBufferedCandidates ( ) ; } , 0 ) ; }
var p = Promise . resolve ( ) ; p . then ( function ( ) { if ( ! hasCallback ) { if ( self . iceGatheringState === 'new' ) { self . iceGatheringState = 'gathering' ; }
window . setTimeout ( self . _emitBufferedCandidates . bind ( self ) , 500 ) ; } } ) ; return p ; } ; window . RTCPeerConnection . prototype . setRemoteDescription = function ( description ) { var self = this ; var stream = new MediaStream ( ) ; var receiverList = [ ] ; var sections = SDPUtils . splitSections ( description . sdp ) ; var sessionpart = sections . shift ( ) ; var isIceLite = SDPUtils . matchPrefix ( sessionpart , 'a=ice-lite' ) . length > 0 ; this . usingBundle = SDPUtils . matchPrefix ( sessionpart , 'a=group:BUNDLE ' ) . length > 0 ; sections . forEach ( function ( mediaSection , sdpMLineIndex ) { var lines = SDPUtils . splitLines ( mediaSection ) ; var mline = lines [ 0 ] . substr ( 2 ) . split ( ' ' ) ; var kind = mline [ 0 ] ; var rejected = mline [ 1 ] === '0' ; var direction = SDPUtils . getDirection ( mediaSection , sessionpart ) ; var transceiver ; var iceGatherer ; var iceTransport ; var dtlsTransport ; var rtpSender ; var rtpReceiver ; var sendEncodingParameters ; var recvEncodingParameters ; var localCapabilities ; var track ; var remoteCapabilities = SDPUtils . parseRtpParameters ( mediaSection ) ; var remoteIceParameters ; var remoteDtlsParameters ; if ( ! rejected ) { remoteIceParameters = SDPUtils . getIceParameters ( mediaSection , sessionpart ) ; remoteDtlsParameters = SDPUtils . getDtlsParameters ( mediaSection , sessionpart ) ; remoteDtlsParameters . role = 'client' ; }
recvEncodingParameters = SDPUtils . parseRtpEncodingParameters ( mediaSection ) ; var mid = SDPUtils . matchPrefix ( mediaSection , 'a=mid:' ) ; if ( mid . length ) { mid = mid [ 0 ] . substr ( 6 ) ; } else { mid = SDPUtils . generateIdentifier ( ) ; }
var cname ; var remoteSsrc = SDPUtils . matchPrefix ( mediaSection , 'a=ssrc:' ) . map ( function ( line ) { return SDPUtils . parseSsrcMedia ( line ) ; } ) . filter ( function ( obj ) { return obj . attribute === 'cname' ; } ) [ 0 ] ; if ( remoteSsrc ) { cname = remoteSsrc . value ; }
var isComplete = SDPUtils . matchPrefix ( mediaSection , 'a=end-of-candidates' , sessionpart ) . length > 0 ; var cands = SDPUtils . matchPrefix ( mediaSection , 'a=candidate:' ) . map ( function ( cand ) { return SDPUtils . parseCandidate ( cand ) ; } ) . filter ( function ( cand ) { return cand . component === '1' ; } ) ; if ( description . type === 'offer' && ! rejected ) { var transports = self . usingBundle && sdpMLineIndex > 0 ? { iceGatherer : self . transceivers [ 0 ] . iceGatherer , iceTransport : self . transceivers [ 0 ] . iceTransport , dtlsTransport : self . transceivers [ 0 ] . dtlsTransport } : self . _createIceAndDtlsTransports ( mid , sdpMLineIndex ) ; if ( isComplete ) { transports . iceTransport . setRemoteCandidates ( cands ) ; }
localCapabilities = RTCRtpReceiver . getCapabilities ( kind ) ; sendEncodingParameters = [ { ssrc : ( 2 * sdpMLineIndex + 2 ) * 1001 } ] ; rtpReceiver = new RTCRtpReceiver ( transports . dtlsTransport , kind ) ; track = rtpReceiver . track ; receiverList . push ( [ track , rtpReceiver ] ) ; stream . addTrack ( track ) ; if ( self . localStreams . length > 0 && self . localStreams [ 0 ] . getTracks ( ) . length >= sdpMLineIndex ) { var localTrack ; if ( kind === 'audio' ) { localTrack = self . localStreams [ 0 ] . getAudioTracks ( ) [ 0 ] ; } else if ( kind === 'video' ) { localTrack = self . localStreams [ 0 ] . getVideoTracks ( ) [ 0 ] ; }
if ( localTrack ) { rtpSender = new RTCRtpSender ( localTrack , transports . dtlsTransport ) ; } }
self . transceivers [ sdpMLineIndex ] = { iceGatherer : transports . iceGatherer , iceTransport : transports . iceTransport , dtlsTransport : transports . dtlsTransport , localCapabilities : localCapabilities , remoteCapabilities : remoteCapabilities , rtpSender : rtpSender , rtpReceiver : rtpReceiver , kind : kind , mid : mid , cname : cname , sendEncodingParameters : sendEncodingParameters , recvEncodingParameters : recvEncodingParameters } ; self . _transceive ( self . transceivers [ sdpMLineIndex ] , false , direction === 'sendrecv' || direction === 'sendonly' ) ; } else if ( description . type === 'answer' && ! rejected ) { transceiver = self . transceivers [ sdpMLineIndex ] ; iceGatherer = transceiver . iceGatherer ; iceTransport = transceiver . iceTransport ; dtlsTransport = transceiver . dtlsTransport ; rtpSender = transceiver . rtpSender ; rtpReceiver = transceiver . rtpReceiver ; sendEncodingParameters = transceiver . sendEncodingParameters ; localCapabilities = transceiver . localCapabilities ; self . transceivers [ sdpMLineIndex ] . recvEncodingParameters = recvEncodingParameters ; self . transceivers [ sdpMLineIndex ] . remoteCapabilities = remoteCapabilities ; self . transceivers [ sdpMLineIndex ] . cname = cname ; if ( ( isIceLite || isComplete ) && cands . length ) { iceTransport . setRemoteCandidates ( cands ) ; }
if ( ! self . usingBundle || sdpMLineIndex === 0 ) { iceTransport . start ( iceGatherer , remoteIceParameters , 'controlling' ) ; dtlsTransport . start ( remoteDtlsParameters ) ; }
self . _transceive ( transceiver , direction === 'sendrecv' || direction === 'recvonly' , direction === 'sendrecv' || direction === 'sendonly' ) ; if ( rtpReceiver && ( direction === 'sendrecv' || direction === 'sendonly' ) ) { track = rtpReceiver . track ; receiverList . push ( [ track , rtpReceiver ] ) ; stream . addTrack ( track ) ; } else { delete transceiver . rtpReceiver ; } } } ) ; this . remoteDescription = { type : description . type , sdp : description . sdp } ; switch ( description . type ) { case 'offer' : this . _updateSignalingState ( 'have-remote-offer' ) ; break ; case 'answer' : this . _updateSignalingState ( 'stable' ) ; break ; default : throw new TypeError ( 'unsupported type "' + description . type + '"' ) ; }
if ( stream . getTracks ( ) . length ) { self . remoteStreams . push ( stream ) ; window . setTimeout ( function ( ) { var event = new Event ( 'addstream' ) ; event . stream = stream ; self . dispatchEvent ( event ) ; if ( self . onaddstream !== null ) { window . setTimeout ( function ( ) { self . onaddstream ( event ) ; } , 0 ) ; }
receiverList . forEach ( function ( item ) { var track = item [ 0 ] ; var receiver = item [ 1 ] ; var trackEvent = new Event ( 'track' ) ; trackEvent . track = track ; trackEvent . receiver = receiver ; trackEvent . streams = [ stream ] ; self . dispatchEvent ( event ) ; if ( self . ontrack !== null ) { window . setTimeout ( function ( ) { self . ontrack ( trackEvent ) ; } , 0 ) ; } } ) ; } , 0 ) ; }
if ( arguments . length > 1 && typeof arguments [ 1 ] === 'function' ) { window . setTimeout ( arguments [ 1 ] , 0 ) ; }
return Promise . resolve ( ) ; } ; window . RTCPeerConnection . prototype . close = function ( ) { this . transceivers . forEach ( function ( transceiver ) { if ( transceiver . iceTransport ) { transceiver . iceTransport . stop ( ) ; }
if ( transceiver . dtlsTransport ) { transceiver . dtlsTransport . stop ( ) ; }
if ( transceiver . rtpSender ) { transceiver . rtpSender . stop ( ) ; }
if ( transceiver . rtpReceiver ) { transceiver . rtpReceiver . stop ( ) ; } } ) ; this . _updateSignalingState ( 'closed' ) ; } ; window . RTCPeerConnection . prototype . _updateSignalingState = function ( newState ) { this . signalingState = newState ; var event = new Event ( 'signalingstatechange' ) ; this . dispatchEvent ( event ) ; if ( this . onsignalingstatechange !== null ) { this . onsignalingstatechange ( event ) ; } } ; window . RTCPeerConnection . prototype . _maybeFireNegotiationNeeded = function ( ) { var event = new Event ( 'negotiationneeded' ) ; this . dispatchEvent ( event ) ; if ( this . onnegotiationneeded !== null ) { this . onnegotiationneeded ( event ) ; } } ; window . RTCPeerConnection . prototype . _updateConnectionState = function ( ) { var self = this ; var newState ; var states = { 'new' : 0 , closed : 0 , connecting : 0 , checking : 0 , connected : 0 , completed : 0 , failed : 0 } ; this . transceivers . forEach ( function ( transceiver ) { states [ transceiver . iceTransport . state ] ++ ; states [ transceiver . dtlsTransport . state ] ++ ; } ) ; states . connected += states . completed ; newState = 'new' ; if ( states . failed > 0 ) { newState = 'failed' ; } else if ( states . connecting > 0 || states . checking > 0 ) { newState = 'connecting' ; } else if ( states . disconnected > 0 ) { newState = 'disconnected' ; } else if ( states . new > 0 ) { newState = 'new' ; } else if ( states . connected > 0 || states . completed > 0 ) { newState = 'connected' ; }
if ( newState !== self . iceConnectionState ) { self . iceConnectionState = newState ; var event = new Event ( 'iceconnectionstatechange' ) ; this . dispatchEvent ( event ) ; if ( this . oniceconnectionstatechange !== null ) { this . oniceconnectionstatechange ( event ) ; } } } ; window . RTCPeerConnection . prototype . createOffer = function ( ) { var self = this ; if ( this . _pendingOffer ) { throw new Error ( 'createOffer called while there is a pending offer.' ) ; }
var offerOptions ; if ( arguments . length === 1 && typeof arguments [ 0 ] !== 'function' ) { offerOptions = arguments [ 0 ] ; } else if ( arguments . length === 3 ) { offerOptions = arguments [ 2 ] ; }
var tracks = [ ] ; var numAudioTracks = 0 ; var numVideoTracks = 0 ; if ( this . localStreams . length ) { numAudioTracks = this . localStreams [ 0 ] . getAudioTracks ( ) . length ; numVideoTracks = this . localStreams [ 0 ] . getVideoTracks ( ) . length ; }
if ( offerOptions ) { if ( offerOptions . mandatory || offerOptions . optional ) { throw new TypeError ( 'Legacy mandatory/optional constraints not supported.' ) ; }
if ( offerOptions . offerToReceiveAudio !== undefined ) { numAudioTracks = offerOptions . offerToReceiveAudio ; }
if ( offerOptions . offerToReceiveVideo !== undefined ) { numVideoTracks = offerOptions . offerToReceiveVideo ; } }
if ( this . localStreams . length ) { this . localStreams [ 0 ] . getTracks ( ) . forEach ( function ( track ) { tracks . push ( { kind : track . kind , track : track , wantReceive : track . kind === 'audio' ? numAudioTracks > 0 : numVideoTracks > 0 } ) ; if ( track . kind === 'audio' ) { numAudioTracks -- ; } else if ( track . kind === 'video' ) { numVideoTracks -- ; } } ) ; }
while ( numAudioTracks > 0 || numVideoTracks > 0 ) { if ( numAudioTracks > 0 ) { tracks . push ( { kind : 'audio' , wantReceive : true } ) ; numAudioTracks -- ; }
if ( numVideoTracks > 0 ) { tracks . push ( { kind : 'video' , wantReceive : true } ) ; numVideoTracks -- ; } }
var sdp = SDPUtils . writeSessionBoilerplate ( ) ; var transceivers = [ ] ; tracks . forEach ( function ( mline , sdpMLineIndex ) { var track = mline . track ; var kind = mline . kind ; var mid = SDPUtils . generateIdentifier ( ) ; var transports = self . usingBundle && sdpMLineIndex > 0 ? { iceGatherer : transceivers [ 0 ] . iceGatherer , iceTransport : transceivers [ 0 ] . iceTransport , dtlsTransport : transceivers [ 0 ] . dtlsTransport } : self . _createIceAndDtlsTransports ( mid , sdpMLineIndex ) ; var localCapabilities = RTCRtpSender . getCapabilities ( kind ) ; var rtpSender ; var rtpReceiver ; var sendEncodingParameters = [ { ssrc : ( 2 * sdpMLineIndex + 1 ) * 1001 } ] ; if ( track ) { rtpSender = new RTCRtpSender ( track , transports . dtlsTransport ) ; }
if ( mline . wantReceive ) { rtpReceiver = new RTCRtpReceiver ( transports . dtlsTransport , kind ) ; }
transceivers [ sdpMLineIndex ] = { iceGatherer : transports . iceGatherer , iceTransport : transports . iceTransport , dtlsTransport : transports . dtlsTransport , localCapabilities : localCapabilities , remoteCapabilities : null , rtpSender : rtpSender , rtpReceiver : rtpReceiver , kind : kind , mid : mid , sendEncodingParameters : sendEncodingParameters , recvEncodingParameters : null } ; } ) ; if ( this . usingBundle ) { sdp += 'a=group:BUNDLE ' + transceivers . map ( function ( t ) { return t . mid ; } ) . join ( ' ' ) + '\r\n' ; }
tracks . forEach ( function ( mline , sdpMLineIndex ) { var transceiver = transceivers [ sdpMLineIndex ] ; sdp += SDPUtils . writeMediaSection ( transceiver , transceiver . localCapabilities , 'offer' , self . localStreams [ 0 ] ) ; } ) ; this . _pendingOffer = transceivers ; var desc = new RTCSessionDescription ( { type : 'offer' , sdp : sdp } ) ; if ( arguments . length && typeof arguments [ 0 ] === 'function' ) { window . setTimeout ( arguments [ 0 ] , 0 , desc ) ; }
return Promise . resolve ( desc ) ; } ; window . RTCPeerConnection . prototype . createAnswer = function ( ) { var self = this ; var sdp = SDPUtils . writeSessionBoilerplate ( ) ; if ( this . usingBundle ) { sdp += 'a=group:BUNDLE ' + this . transceivers . map ( function ( t ) { return t . mid ; } ) . join ( ' ' ) + '\r\n' ; }
this . transceivers . forEach ( function ( transceiver ) { var commonCapabilities = self . _getCommonCapabilities ( transceiver . localCapabilities , transceiver . remoteCapabilities ) ; sdp += SDPUtils . writeMediaSection ( transceiver , commonCapabilities , 'answer' , self . localStreams [ 0 ] ) ; } ) ; var desc = new RTCSessionDescription ( { type : 'answer' , sdp : sdp } ) ; if ( arguments . length && typeof arguments [ 0 ] === 'function' ) { window . setTimeout ( arguments [ 0 ] , 0 , desc ) ; }
return Promise . resolve ( desc ) ; } ; window . RTCPeerConnection . prototype . addIceCandidate = function ( candidate ) { if ( candidate === null ) { this . transceivers . forEach ( function ( transceiver ) { transceiver . iceTransport . addRemoteCandidate ( { } ) ; } ) ; } else { var mLineIndex = candidate . sdpMLineIndex ; if ( candidate . sdpMid ) { for ( var i = 0 ; i < this . transceivers . length ; i ++ ) { if ( this . transceivers [ i ] . mid === candidate . sdpMid ) { mLineIndex = i ; break ; } } }
var transceiver = this . transceivers [ mLineIndex ] ; if ( transceiver ) { var cand = Object . keys ( candidate . candidate ) . length > 0 ? SDPUtils . parseCandidate ( candidate . candidate ) : { } ; if ( cand . protocol === 'tcp' && ( cand . port === 0 || cand . port === 9 ) ) { return ; }
if ( cand . component !== '1' ) { return ; }
if ( cand . type === 'endOfCandidates' ) { cand = { } ; }
transceiver . iceTransport . addRemoteCandidate ( cand ) ; var sections = SDPUtils . splitSections ( this . remoteDescription . sdp ) ; sections [ mLineIndex + 1 ] += ( cand . type ? candidate . candidate . trim ( ) : 'a=end-of-candidates' ) + '\r\n' ; this . remoteDescription . sdp = sections . join ( '' ) ; } }
if ( arguments . length > 1 && typeof arguments [ 1 ] === 'function' ) { window . setTimeout ( arguments [ 1 ] , 0 ) ; }
return Promise . resolve ( ) ; } ; window . RTCPeerConnection . prototype . getStats = function ( ) { var promises = [ ] ; this . transceivers . forEach ( function ( transceiver ) { [ 'rtpSender' , 'rtpReceiver' , 'iceGatherer' , 'iceTransport' , 'dtlsTransport' ] . forEach ( function ( method ) { if ( transceiver [ method ] ) { promises . push ( transceiver [ method ] . getStats ( ) ) ; } } ) ; } ) ; var cb = arguments . length > 1 && typeof arguments [ 1 ] === 'function' && arguments [ 1 ] ; return new Promise ( function ( resolve ) { var results = new Map ( ) ; Promise . all ( promises ) . then ( function ( res ) { res . forEach ( function ( result ) { Object . keys ( result ) . forEach ( function ( id ) { results . set ( id , result [ id ] ) ; results [ id ] = result [ id ] ; } ) ; } ) ; if ( cb ) { window . setTimeout ( cb , 0 , results ) ; }
resolve ( results ) ; } ) ; } ) ; } ; } } ; module . exports = { shimPeerConnection : edgeShim . shimPeerConnection , shimGetUserMedia : require ( './getusermedia' ) } ; } , { "./getusermedia" : 6 , "sdp" : 1 } ] , 6 : [ function ( require , module , exports ) { 'use strict' ; module . exports = function ( ) { var shimError _ = function ( e ) { return { name : { PermissionDeniedError : 'NotAllowedError' } [ e . name ] || e . name , message : e . message , constraint : e . constraint , toString : function ( ) { return this . name ; } } ; } ; var origGetUserMedia = navigator . mediaDevices . getUserMedia . bind ( navigator . mediaDevices ) ; navigator . mediaDevices . getUserMedia = function ( c ) { return origGetUserMedia ( c ) . catch ( function ( e ) { return Promise . reject ( shimError _ ( e ) ) ; } ) ; } ; } ; } , { } ] , 7 : [ function ( require , module , exports ) { 'use strict' ; var browserDetails = require ( '../utils' ) . browserDetails ; var firefoxShim = { shimOnTrack : function ( ) { if ( typeof window === 'object' && window . RTCPeerConnection && ! ( 'ontrack' in
window . RTCPeerConnection . prototype ) ) { Object . defineProperty ( window . RTCPeerConnection . prototype , 'ontrack' , { get : function ( ) { return this . _ontrack ; } , set : function ( f ) { if ( this . _ontrack ) { this . removeEventListener ( 'track' , this . _ontrack ) ; this . removeEventListener ( 'addstream' , this . _ontrackpoly ) ; }
this . addEventListener ( 'track' , this . _ontrack = f ) ; this . addEventListener ( 'addstream' , this . _ontrackpoly = function ( e ) { e . stream . getTracks ( ) . forEach ( function ( track ) { var event = new Event ( 'track' ) ; event . track = track ; event . receiver = { track : track } ; event . streams = [ e . stream ] ; this . dispatchEvent ( event ) ; } . bind ( this ) ) ; } . bind ( this ) ) ; } } ) ; } } , shimSourceObject : function ( ) { if ( typeof window === 'object' ) { if ( window . HTMLMediaElement && ! ( 'srcObject' in window . HTMLMediaElement . prototype ) ) { Object . defineProperty ( window . HTMLMediaElement . prototype , 'srcObject' , { get : function ( ) { return this . mozSrcObject ; } , set : function ( stream ) { this . mozSrcObject = stream ; } } ) ; } } } , shimPeerConnection : function ( ) { if ( typeof window !== 'object' || ! ( window . RTCPeerConnection || window . mozRTCPeerConnection ) ) { return ; }
if ( ! window . RTCPeerConnection ) { window . RTCPeerConnection = function ( pcConfig , pcConstraints ) { if ( browserDetails . version < 38 ) { if ( pcConfig && pcConfig . iceServers ) { var newIceServers = [ ] ; for ( var i = 0 ; i < pcConfig . iceServers . length ; i ++ ) { var server = pcConfig . iceServers [ i ] ; if ( server . hasOwnProperty ( 'urls' ) ) { for ( var j = 0 ; j < server . urls . length ; j ++ ) { var newServer = { url : server . urls [ j ] } ; if ( server . urls [ j ] . indexOf ( 'turn' ) === 0 ) { newServer . username = server . username ; newServer . credential = server . credential ; }
newIceServers . push ( newServer ) ; } } else { newIceServers . push ( pcConfig . iceServers [ i ] ) ; } }
pcConfig . iceServers = newIceServers ; } }
return new mozRTCPeerConnection ( pcConfig , pcConstraints ) ; } ; window . RTCPeerConnection . prototype = mozRTCPeerConnection . prototype ; if ( mozRTCPeerConnection . generateCertificate ) { Object . defineProperty ( window . RTCPeerConnection , 'generateCertificate' , { get : function ( ) { return mozRTCPeerConnection . generateCertificate ; } } ) ; }
window . RTCSessionDescription = mozRTCSessionDescription ; window . RTCIceCandidate = mozRTCIceCandidate ; }
[ 'setLocalDescription' , 'setRemoteDescription' , 'addIceCandidate' ] . forEach ( function ( method ) { var nativeMethod = RTCPeerConnection . prototype [ method ] ; RTCPeerConnection . prototype [ method ] = function ( ) { arguments [ 0 ] = new ( ( method === 'addIceCandidate' ) ? RTCIceCandidate : RTCSessionDescription ) ( arguments [ 0 ] ) ; return nativeMethod . apply ( this , arguments ) ; } ; } ) ; var nativeAddIceCandidate = RTCPeerConnection . prototype . addIceCandidate ; RTCPeerConnection . prototype . addIceCandidate = function ( ) { return arguments [ 0 ] === null ? Promise . resolve ( ) : nativeAddIceCandidate . apply ( this , arguments ) ; } ; var makeMapStats = function ( stats ) { var map = new Map ( ) ; Object . keys ( stats ) . forEach ( function ( key ) { map . set ( key , stats [ key ] ) ; map [ key ] = stats [ key ] ; } ) ; return map ; } ; var nativeGetStats = RTCPeerConnection . prototype . getStats ; RTCPeerConnection . prototype . getStats = function ( selector , onSucc , onErr ) { return nativeGetStats . apply ( this , [ selector || null ] ) . then ( function ( stats ) { return makeMapStats ( stats ) ; } ) . then ( onSucc , onErr ) ; } ; } } ; module . exports = { shimOnTrack : firefoxShim . shimOnTrack , shimSourceObject : firefoxShim . shimSourceObject , shimPeerConnection : firefoxShim . shimPeerConnection , shimGetUserMedia : require ( './getusermedia' ) } ; } , { "../utils" : 10 , "./getusermedia" : 8 } ] , 8 : [ function ( require , module , exports ) { 'use strict' ; var logging = require ( '../utils' ) . log ; var browserDetails = require ( '../utils' ) . browserDetails ; module . exports = function ( ) { var shimError _ = function ( e ) { return { name : { SecurityError : 'NotAllowedError' , PermissionDeniedError : 'NotAllowedError' } [ e . name ] || e . name , message : { 'The operation is insecure.' : 'The request is not allowed by the ' + 'user agent or the platform in the current context.' } [ e . message ] || e . message , constraint : e . constraint , toString : function ( ) { return this . name + ( this . message && ': ' ) + this . message ; } } ; } ; var getUserMedia _ = function ( constraints , onSuccess , onError ) { var constraintsToFF37 _ = function ( c ) { if ( typeof c !== 'object' || c . require ) { return c ; }
var require = [ ] ; Object . keys ( c ) . forEach ( function ( key ) { if ( key === 'require' || key === 'advanced' || key === 'mediaSource' ) { return ; }
var r = c [ key ] = ( typeof c [ key ] === 'object' ) ? c [ key ] : { ideal : c [ key ] } ; if ( r . min !== undefined || r . max !== undefined || r . exact !== undefined ) { require . push ( key ) ; }
if ( r . exact !== undefined ) { if ( typeof r . exact === 'number' ) { r . min = r . max = r . exact ; } else { c [ key ] = r . exact ; }
delete r . exact ; }
if ( r . ideal !== undefined ) { c . advanced = c . advanced || [ ] ; var oc = { } ; if ( typeof r . ideal === 'number' ) { oc [ key ] = { min : r . ideal , max : r . ideal } ; } else { oc [ key ] = r . ideal ; }
c . advanced . push ( oc ) ; delete r . ideal ; if ( ! Object . keys ( r ) . length ) { delete c [ key ] ; } } } ) ; if ( require . length ) { c . require = require ; }
return c ; } ; constraints = JSON . parse ( JSON . stringify ( constraints ) ) ; if ( browserDetails . version < 38 ) { logging ( 'spec: ' + JSON . stringify ( constraints ) ) ; if ( constraints . audio ) { constraints . audio = constraintsToFF37 _ ( constraints . audio ) ; }
if ( constraints . video ) { constraints . video = constraintsToFF37 _ ( constraints . video ) ; }
logging ( 'ff37: ' + JSON . stringify ( constraints ) ) ; }
return navigator . mozGetUserMedia ( constraints , onSuccess , function ( e ) { onError ( shimError _ ( e ) ) ; } ) ; } ; var getUserMediaPromise _ = function ( constraints ) { return new Promise ( function ( resolve , reject ) { getUserMedia _ ( constraints , resolve , reject ) ; } ) ; } ; if ( ! navigator . mediaDevices ) { navigator . mediaDevices = { getUserMedia : getUserMediaPromise _ , addEventListener : function ( ) { } , removeEventListener : function ( ) { } } ; }
navigator . mediaDevices . enumerateDevices = navigator . mediaDevices . enumerateDevices || function ( ) { return new Promise ( function ( resolve ) { var infos = [ { kind : 'audioinput' , deviceId : 'default' , label : '' , groupId : '' } , { kind : 'videoinput' , deviceId : 'default' , label : '' , groupId : '' } ] ; resolve ( infos ) ; } ) ; } ; if ( browserDetails . version < 41 ) { var orgEnumerateDevices = navigator . mediaDevices . enumerateDevices . bind ( navigator . mediaDevices ) ; navigator . mediaDevices . enumerateDevices = function ( ) { return orgEnumerateDevices ( ) . then ( undefined , function ( e ) { if ( e . name === 'NotFoundError' ) { return [ ] ; }
throw e ; } ) ; } ; }
if ( browserDetails . version < 49 ) { var origGetUserMedia = navigator . mediaDevices . getUserMedia . bind ( navigator . mediaDevices ) ; navigator . mediaDevices . getUserMedia = function ( c ) { return origGetUserMedia ( c ) . catch ( function ( e ) { return Promise . reject ( shimError _ ( e ) ) ; } ) ; } ; }
navigator . getUserMedia = function ( constraints , onSuccess , onError ) { if ( browserDetails . version < 44 ) { return getUserMedia _ ( constraints , onSuccess , onError ) ; }
console . warn ( 'navigator.getUserMedia has been replaced by ' + 'navigator.mediaDevices.getUserMedia' ) ; navigator . mediaDevices . getUserMedia ( constraints ) . then ( onSuccess , onError ) ; } ; } ; } , { "../utils" : 10 } ] , 9 : [ function ( require , module , exports ) { 'use strict' ; var safariShim = { shimGetUserMedia : function ( ) { navigator . getUserMedia = navigator . webkitGetUserMedia ; } } ; module . exports = { shimGetUserMedia : safariShim . shimGetUserMedia } ; } , { } ] , 10 : [ function ( require , module , exports ) { 'use strict' ; var logDisabled _ = true ; var utils = { disableLog : function ( bool ) { if ( typeof bool !== 'boolean' ) { return new Error ( 'Argument type: ' + typeof bool + '. Please use a boolean.' ) ; }
logDisabled _ = bool ; return ( bool ) ? 'adapter.js logging disabled' : 'adapter.js logging enabled' ; } , log : function ( ) { if ( typeof window === 'object' ) { if ( logDisabled _ ) { return ; }
if ( typeof console !== 'undefined' && typeof console . log === 'function' ) { console . log . apply ( console , arguments ) ; } } } , extractVersion : function ( uastring , expr , pos ) { var match = uastring . match ( expr ) ; return match && match . length >= pos && parseInt ( match [ pos ] , 10 ) ; } , detectBrowser : function ( ) { var result = { } ; result . browser = null ; result . version = null ; if ( typeof window === 'undefined' || ! window . navigator ) { result . browser = 'Not a browser.' ; return result ; }
if ( navigator . mozGetUserMedia ) { result . browser = 'firefox' ; result . version = this . extractVersion ( navigator . userAgent , /Firefox\/([0-9]+)\./ , 1 ) ; } else if ( navigator . webkitGetUserMedia ) { if ( window . webkitRTCPeerConnection ) { result . browser = 'chrome' ; result . version = this . extractVersion ( navigator . userAgent , /Chrom(e|ium)\/([0-9]+)\./ , 2 ) ; } else { if ( navigator . userAgent . match ( /Version\/(\d+).(\d+)/ ) ) { result . browser = 'safari' ; result . version = this . extractVersion ( navigator . userAgent , /AppleWebKit\/([0-9]+)\./ , 1 ) ; } else { result . browser = 'Unsupported webkit-based browser ' + 'with GUM support but no WebRTC support.' ; return result ; } } } else if ( navigator . mediaDevices && navigator . userAgent . match ( /Edge\/(\d+).(\d+)$/ ) ) { result . browser = 'edge' ; result . version = this . extractVersion ( navigator . userAgent , /Edge\/(\d+).(\d+)$/ , 2 ) ; } else { result . browser = 'Not a supported browser.' ; return result ; }
return result ; } } ; module . exports = { log : utils . log , disableLog : utils . disableLog , browserDetails : utils . detectBrowser ( ) , extractVersion : utils . extractVersion } ; } , { } ] } , { } , [ 2 ] ) ( 2 ) } ) ; AdapterJS . parseWebrtcDetectedBrowser ( ) ; if ( navigator . mozGetUserMedia ) { MediaStreamTrack . getSources = function ( successCb ) { setTimeout ( function ( ) { var infos = [ { kind : 'audio' , id : 'default' , label : '' , facing : '' } , { kind : 'video' , id : 'default' , label : '' , facing : '' } ] ; successCb ( infos ) ; } , 0 ) ; } ; createIceServer = function ( url , username , password ) { console . warn ( 'createIceServer is deprecated. It should be replaced with an application level implementation.' ) ; var iceServer = null ; var urlParts = url . split ( ':' ) ; if ( urlParts [ 0 ] . indexOf ( 'stun' ) === 0 ) { iceServer = { urls : [ url ] } ; } else if ( urlParts [ 0 ] . indexOf ( 'turn' ) === 0 ) { if ( webrtcDetectedVersion < 27 ) { var turnUrlParts = url . split ( '?' ) ; if ( turnUrlParts . length === 1 || turnUrlParts [ 1 ] . indexOf ( 'transport=udp' ) === 0 ) { iceServer = { urls : [ turnUrlParts [ 0 ] ] , credential : password , username : username } ; } } else { iceServer = { urls : [ url ] , credential : password , username : username } ; } }
return iceServer ; } ; createIceServers = function ( urls , username , password ) { console . warn ( 'createIceServers is deprecated. It should be replaced with an application level implementation.' ) ; var iceServers = [ ] ; for ( i = 0 ; i < urls . length ; i ++ ) { var iceServer = createIceServer ( urls [ i ] , username , password ) ; if ( iceServer !== null ) { iceServers . push ( iceServer ) ; } }
return iceServers ; } ; } else if ( navigator . webkitGetUserMedia ) { createIceServer = function ( url , username , password ) { console . warn ( 'createIceServer is deprecated. It should be replaced with an application level implementation.' ) ; var iceServer = null ; var urlParts = url . split ( ':' ) ; if ( urlParts [ 0 ] . indexOf ( 'stun' ) === 0 ) { iceServer = { 'url' : url } ; } else if ( urlParts [ 0 ] . indexOf ( 'turn' ) === 0 ) { iceServer = { 'url' : url , 'credential' : password , 'username' : username } ; }
return iceServer ; } ; createIceServers = function ( urls , username , password ) { console . warn ( 'createIceServers is deprecated. It should be replaced with an application level implementation.' ) ; var iceServers = [ ] ; if ( webrtcDetectedVersion >= 34 ) { iceServers = { 'urls' : urls , 'credential' : password , 'username' : username } ; } else { for ( i = 0 ; i < urls . length ; i ++ ) { var iceServer = createIceServer ( urls [ i ] , username , password ) ; if ( iceServer !== null ) { iceServers . push ( iceServer ) ; } } }
return iceServers ; } ; }
if ( navigator . mediaDevices && navigator . userAgent . match ( /Edge\/(\d+).(\d+)$/ ) ) { getUserMedia = window . getUserMedia = navigator . getUserMedia . bind ( navigator ) ; attachMediaStream = function ( element , stream ) { element . srcObject = stream ; return element ; } ; reattachMediaStream = function ( to , from ) { to . srcObject = from . srcObject ; return to ; } ; }
if ( attachMediaStream ) { attachMediaStream _base = attachMediaStream ; attachMediaStream = function ( element , stream ) { if ( ( webrtcDetectedBrowser === 'chrome' || webrtcDetectedBrowser === 'opera' ) && ! stream ) { element . src = '' ; } else { attachMediaStream _base ( element , stream ) ; }
return element ; } ; }
if ( reattachMediaStream ) { reattachMediaStream _base = reattachMediaStream ; reattachMediaStream = function ( to , from ) { reattachMediaStream _base ( to , from ) ; return to ; } ; }
window . attachMediaStream = attachMediaStream ; window . reattachMediaStream = reattachMediaStream ; window . getUserMedia = getUserMedia ; AdapterJS . attachMediaStream = attachMediaStream ; AdapterJS . reattachMediaStream = reattachMediaStream ; AdapterJS . getUserMedia = getUserMedia ; if ( typeof Promise === 'undefined' ) { requestUserMedia = null ; }
AdapterJS . maybeThroughWebRTCReady ( ) ; } else { if ( typeof console !== 'object' || typeof console . log !== 'function' ) { console = { } || console ; console . log = function ( arg ) { } ; console . info = function ( arg ) { } ; console . error = function ( arg ) { } ; console . dir = function ( arg ) { } ; console . exception = function ( arg ) { } ; console . trace = function ( arg ) { } ; console . warn = function ( arg ) { } ; console . count = function ( arg ) { } ; console . debug = function ( arg ) { } ; console . count = function ( arg ) { } ; console . time = function ( arg ) { } ; console . timeEnd = function ( arg ) { } ; console . group = function ( arg ) { } ; console . groupCollapsed = function ( arg ) { } ; console . groupEnd = function ( arg ) { } ; }
AdapterJS . parseWebrtcDetectedBrowser ( ) ; isIE = webrtcDetectedBrowser === 'IE' ; AdapterJS . WebRTCPlugin . WaitForPluginReady = function ( ) { while ( AdapterJS . WebRTCPlugin . pluginState !== AdapterJS . WebRTCPlugin . PLUGIN _STATES . READY ) { } } ; AdapterJS . WebRTCPlugin . callWhenPluginReady = function ( callback ) { if ( AdapterJS . WebRTCPlugin . pluginState === AdapterJS . WebRTCPlugin . PLUGIN _STATES . READY ) { callback ( ) ; } else { var checkPluginReadyState = setInterval ( function ( ) { if ( AdapterJS . WebRTCPlugin . pluginState === AdapterJS . WebRTCPlugin . PLUGIN _STATES . READY ) { clearInterval ( checkPluginReadyState ) ; callback ( ) ; } } , 100 ) ; } } ; AdapterJS . WebRTCPlugin . setLogLevel = function ( logLevel ) { AdapterJS . WebRTCPlugin . callWhenPluginReady ( function ( ) { AdapterJS . WebRTCPlugin . plugin . setLogLevel ( logLevel ) ; } ) ; } ; AdapterJS . WebRTCPlugin . injectPlugin = function ( ) { if ( document . readyState !== 'complete' ) { return ; }
if ( AdapterJS . WebRTCPlugin . pluginState !== AdapterJS . WebRTCPlugin . PLUGIN _STATES . INITIALIZING ) { return ; }
AdapterJS . WebRTCPlugin . pluginState = AdapterJS . WebRTCPlugin . PLUGIN _STATES . INJECTING ; if ( webrtcDetectedBrowser === 'IE' && webrtcDetectedVersion <= 10 ) { var frag = document . createDocumentFragment ( ) ; AdapterJS . WebRTCPlugin . plugin = document . createElement ( 'div' ) ; AdapterJS . WebRTCPlugin . plugin . innerHTML = '<object id="' +
AdapterJS . WebRTCPlugin . pluginInfo . pluginId + '" type="' +
AdapterJS . WebRTCPlugin . pluginInfo . type + '" ' + 'width="1" height="1">' + '<param name="pluginId" value="' +
AdapterJS . WebRTCPlugin . pluginInfo . pluginId + '" /> ' + '<param name="windowless" value="false" /> ' + '<param name="pageId" value="' + AdapterJS . WebRTCPlugin . pageId + '" /> ' + '<param name="onload" value="' + AdapterJS . WebRTCPlugin . pluginInfo . onload + '" />' + '<param name="tag" value="' + AdapterJS . WebRTCPlugin . TAGS . NONE + '" />' +
( AdapterJS . options . getAllCams ? '<param name="forceGetAllCams" value="True" />' : '' ) + '</object>' ; while ( AdapterJS . WebRTCPlugin . plugin . firstChild ) { frag . appendChild ( AdapterJS . WebRTCPlugin . plugin . firstChild ) ; }
document . body . appendChild ( frag ) ; AdapterJS . WebRTCPlugin . plugin = document . getElementById ( AdapterJS . WebRTCPlugin . pluginInfo . pluginId ) ; } else { AdapterJS . WebRTCPlugin . plugin = document . createElement ( 'object' ) ; AdapterJS . WebRTCPlugin . plugin . id = AdapterJS . WebRTCPlugin . pluginInfo . pluginId ; if ( isIE ) { AdapterJS . WebRTCPlugin . plugin . width = '1px' ; AdapterJS . WebRTCPlugin . plugin . height = '1px' ; } else { AdapterJS . WebRTCPlugin . plugin . width = '0px' ; AdapterJS . WebRTCPlugin . plugin . height = '0px' ; }
AdapterJS . WebRTCPlugin . plugin . type = AdapterJS . WebRTCPlugin . pluginInfo . type ; AdapterJS . WebRTCPlugin . plugin . innerHTML = '<param name="onload" value="' +
AdapterJS . WebRTCPlugin . pluginInfo . onload + '">' + '<param name="pluginId" value="' +
AdapterJS . WebRTCPlugin . pluginInfo . pluginId + '">' + '<param name="windowless" value="false" /> ' +
( AdapterJS . options . getAllCams ? '<param name="forceGetAllCams" value="True" />' : '' ) + '<param name="pageId" value="' + AdapterJS . WebRTCPlugin . pageId + '">' + '<param name="tag" value="' + AdapterJS . WebRTCPlugin . TAGS . NONE + '" />' ; document . body . appendChild ( AdapterJS . WebRTCPlugin . plugin ) ; }
AdapterJS . WebRTCPlugin . pluginState = AdapterJS . WebRTCPlugin . PLUGIN _STATES . INJECTED ; } ; AdapterJS . WebRTCPlugin . isPluginInstalled = function ( comName , plugName , installedCb , notInstalledCb ) { if ( ! isIE ) { var pluginArray = navigator . plugins ; for ( var i = 0 ; i < pluginArray . length ; i ++ ) { if ( pluginArray [ i ] . name . indexOf ( plugName ) >= 0 ) { installedCb ( ) ; return ; } }
notInstalledCb ( ) ; } else { try { var axo = new ActiveXObject ( comName + '.' + plugName ) ; } catch ( e ) { notInstalledCb ( ) ; return ; }
installedCb ( ) ; } } ; AdapterJS . WebRTCPlugin . defineWebRTCInterface = function ( ) { if ( AdapterJS . WebRTCPlugin . pluginState === AdapterJS . WebRTCPlugin . PLUGIN _STATES . READY ) { console . error ( 'AdapterJS - WebRTC interface has already been defined' ) ; return ; }
AdapterJS . WebRTCPlugin . pluginState = AdapterJS . WebRTCPlugin . PLUGIN _STATES . INITIALIZING ; AdapterJS . isDefined = function ( variable ) { return variable !== null && variable !== undefined ; } ; createIceServer = function ( url , username , password ) { var iceServer = null ; var urlParts = url . split ( ':' ) ; if ( urlParts [ 0 ] . indexOf ( 'stun' ) === 0 ) { iceServer = { 'url' : url , 'hasCredentials' : false } ; } else if ( urlParts [ 0 ] . indexOf ( 'turn' ) === 0 ) { iceServer = { 'url' : url , 'hasCredentials' : true , 'credential' : password , 'username' : username } ; }
return iceServer ; } ; createIceServers = function ( urls , username , password ) { var iceServers = [ ] ; for ( var i = 0 ; i < urls . length ; ++ i ) { iceServers . push ( createIceServer ( urls [ i ] , username , password ) ) ; }
return iceServers ; } ; RTCSessionDescription = function ( info ) { AdapterJS . WebRTCPlugin . WaitForPluginReady ( ) ; return AdapterJS . WebRTCPlugin . plugin . ConstructSessionDescription ( info . type , info . sdp ) ; } ; RTCPeerConnection = function ( servers , constraints ) { if ( ! ( servers === undefined || servers === null || Array . isArray ( servers . iceServers ) ) ) { throw new Error ( 'Failed to construct \'RTCPeerConnection\': Malformed RTCConfiguration' ) ; }
if ( typeof constraints !== 'undefined' && constraints !== null ) { var invalidConstraits = false ; invalidConstraits |= typeof constraints !== 'object' ; invalidConstraits |= constraints . hasOwnProperty ( 'mandatory' ) && constraints . mandatory !== undefined && constraints . mandatory !== null && constraints . mandatory . constructor !== Object ; invalidConstraits |= constraints . hasOwnProperty ( 'optional' ) && constraints . optional !== undefined && constraints . optional !== null && ! Array . isArray ( constraints . optional ) ; if ( invalidConstraits ) { throw new Error ( 'Failed to construct \'RTCPeerConnection\': Malformed constraints object' ) ; } }
AdapterJS . WebRTCPlugin . WaitForPluginReady ( ) ; var iceServers = null ; if ( servers && Array . isArray ( servers . iceServers ) ) { iceServers = servers . iceServers ; for ( var i = 0 ; i < iceServers . length ; i ++ ) { if ( iceServers [ i ] . urls && ! iceServers [ i ] . url ) { iceServers [ i ] . url = iceServers [ i ] . urls ; }
iceServers [ i ] . hasCredentials = AdapterJS . isDefined ( iceServers [ i ] . username ) && AdapterJS . isDefined ( iceServers [ i ] . credential ) ; } }
if ( AdapterJS . WebRTCPlugin . plugin . PEER _CONNECTION _VERSION && AdapterJS . WebRTCPlugin . plugin . PEER _CONNECTION _VERSION > 1 ) { if ( iceServers ) { servers . iceServers = iceServers ; }
return AdapterJS . WebRTCPlugin . plugin . PeerConnection ( servers ) ; } else { var mandatory = ( constraints && constraints . mandatory ) ? constraints . mandatory : null ; var optional = ( constraints && constraints . optional ) ? constraints . optional : null ; return AdapterJS . WebRTCPlugin . plugin . PeerConnection ( AdapterJS . WebRTCPlugin . pageId , iceServers , mandatory , optional ) ; } } ; MediaStreamTrack = function ( ) { } ; MediaStreamTrack . getSources = function ( callback ) { AdapterJS . WebRTCPlugin . callWhenPluginReady ( function ( ) { AdapterJS . WebRTCPlugin . plugin . GetSources ( callback ) ; } ) ; } ; getUserMedia = function ( constraints , successCallback , failureCallback ) { constraints . audio = constraints . audio || false ; constraints . video = constraints . video || false ; AdapterJS . WebRTCPlugin . callWhenPluginReady ( function ( ) { AdapterJS . WebRTCPlugin . plugin . getUserMedia ( constraints , successCallback , failureCallback ) ; } ) ; } ; window . navigator . getUserMedia = getUserMedia ; if ( ! navigator . mediaDevices && typeof Promise !== 'undefined' ) { requestUserMedia = function ( constraints ) { return new Promise ( function ( resolve , reject ) { getUserMedia ( constraints , resolve , reject ) ; } ) ; } ; navigator . mediaDevices = { getUserMedia : requestUserMedia , enumerateDevices : function ( ) { return new Promise ( function ( resolve ) { var kinds = { audio : 'audioinput' , video : 'videoinput' } ; return MediaStreamTrack . getSources ( function ( devices ) { resolve ( devices . map ( function ( device ) { return { label : device . label , kind : kinds [ device . kind ] , id : device . id , deviceId : device . id , groupId : '' } ; } ) ) ; } ) ; } ) ; } } ; }
attachMediaStream = function ( element , stream ) { if ( ! element || ! element . parentNode ) { return ; }
var streamId ; if ( stream === null ) { streamId = '' ; } else { if ( typeof stream . enableSoundTracks !== 'undefined' ) { stream . enableSoundTracks ( true ) ; }
streamId = stream . id ; }
var elementId = element . id . length === 0 ? Math . random ( ) . toString ( 36 ) . slice ( 2 ) : element . id ; var nodeName = element . nodeName . toLowerCase ( ) ; if ( nodeName !== 'object' ) { var tag ; switch ( nodeName ) { case 'audio' : tag = AdapterJS . WebRTCPlugin . TAGS . AUDIO ; break ; case 'video' : tag = AdapterJS . WebRTCPlugin . TAGS . VIDEO ; break ; default : tag = AdapterJS . WebRTCPlugin . TAGS . NONE ; }
var frag = document . createDocumentFragment ( ) ; var temp = document . createElement ( 'div' ) ; var classHTML = '' ; if ( element . className ) { classHTML = 'class="' + element . className + '" ' ; } else if ( element . attributes && element . attributes [ 'class' ] ) { classHTML = 'class="' + element . attributes [ 'class' ] . value + '" ' ; }
temp . innerHTML = '<object id="' + elementId + '" ' + classHTML + 'type="' + AdapterJS . WebRTCPlugin . pluginInfo . type + '">' + '<param name="pluginId" value="' + elementId + '" /> ' + '<param name="pageId" value="' + AdapterJS . WebRTCPlugin . pageId + '" /> ' + '<param name="windowless" value="true" /> ' + '<param name="streamId" value="' + streamId + '" /> ' + '<param name="tag" value="' + tag + '" /> ' + '</object>' ; while ( temp . firstChild ) { frag . appendChild ( temp . firstChild ) ; }
var height = '' ; var width = '' ; if ( element . clientWidth || element . clientHeight ) { width = element . clientWidth ; height = element . clientHeight ; }
else if ( element . width || element . height ) { width = element . width ; height = element . height ; }
element . parentNode . insertBefore ( frag , element ) ; frag = document . getElementById ( elementId ) ; frag . width = width ; frag . height = height ; element . parentNode . removeChild ( element ) ; } else { var children = element . children ; for ( var i = 0 ; i !== children . length ; ++ i ) { if ( children [ i ] . name === 'streamId' ) { children [ i ] . value = streamId ; break ; } }
element . setStreamId ( streamId ) ; }
var newElement = document . getElementById ( elementId ) ; AdapterJS . forwardEventHandlers ( newElement , element , Object . getPrototypeOf ( element ) ) ; return newElement ; } ; reattachMediaStream = function ( to , from ) { var stream = null ; var children = from . children ; for ( var i = 0 ; i !== children . length ; ++ i ) { if ( children [ i ] . name === 'streamId' ) { AdapterJS . WebRTCPlugin . WaitForPluginReady ( ) ; stream = AdapterJS . WebRTCPlugin . plugin . getStreamWithId ( AdapterJS . WebRTCPlugin . pageId , children [ i ] . value ) ; break ; } }
if ( stream !== null ) { return attachMediaStream ( to , stream ) ; } else { console . log ( 'Could not find the stream associated with this element' ) ; } } ; window . attachMediaStream = attachMediaStream ; window . reattachMediaStream = reattachMediaStream ; window . getUserMedia = getUserMedia ; AdapterJS . attachMediaStream = attachMediaStream ; AdapterJS . reattachMediaStream = reattachMediaStream ; AdapterJS . getUserMedia = getUserMedia ; AdapterJS . forwardEventHandlers = function ( destElem , srcElem , prototype ) { properties = Object . getOwnPropertyNames ( prototype ) ; for ( var prop in properties ) { if ( prop ) { propName = properties [ prop ] ; if ( typeof propName . slice === 'function' && propName . slice ( 0 , 2 ) === 'on' && typeof srcElem [ propName ] === 'function' ) { AdapterJS . addEvent ( destElem , propName . slice ( 2 ) , srcElem [ propName ] ) ; } } }
var subPrototype = Object . getPrototypeOf ( prototype ) ; if ( ! ! subPrototype ) { AdapterJS . forwardEventHandlers ( destElem , srcElem , subPrototype ) ; } } ; RTCIceCandidate = function ( candidate ) { if ( ! candidate . sdpMid ) { candidate . sdpMid = '' ; }
AdapterJS . WebRTCPlugin . WaitForPluginReady ( ) ; return AdapterJS . WebRTCPlugin . plugin . ConstructIceCandidate ( candidate . sdpMid , candidate . sdpMLineIndex , candidate . candidate ) ; } ; AdapterJS . addEvent ( document , 'readystatechange' , AdapterJS . WebRTCPlugin . injectPlugin ) ; AdapterJS . WebRTCPlugin . injectPlugin ( ) ; } ; AdapterJS . WebRTCPlugin . pluginNeededButNotInstalledCb = AdapterJS . WebRTCPlugin . pluginNeededButNotInstalledCb || function ( ) { AdapterJS . addEvent ( document , 'readystatechange' , AdapterJS . WebRTCPlugin . pluginNeededButNotInstalledCbPriv ) ; AdapterJS . WebRTCPlugin . pluginNeededButNotInstalledCbPriv ( ) ; } ; AdapterJS . WebRTCPlugin . pluginNeededButNotInstalledCbPriv = function ( ) { if ( AdapterJS . options . hidePluginInstallPrompt ) { return ; }
var downloadLink = AdapterJS . WebRTCPlugin . pluginInfo . downloadLink ; if ( downloadLink ) { var popupString ; if ( AdapterJS . WebRTCPlugin . pluginInfo . portalLink ) { popupString = 'This website requires you to install the ' + ' <a href="' + AdapterJS . WebRTCPlugin . pluginInfo . portalLink + '" target="_blank">' + AdapterJS . WebRTCPlugin . pluginInfo . companyName + ' WebRTC Plugin</a>' + ' to work on this browser.' ; } else { popupString = AdapterJS . TEXT . PLUGIN . REQUIRE _INSTALLATION ; }
AdapterJS . renderNotificationBar ( popupString , AdapterJS . TEXT . PLUGIN . BUTTON , downloadLink ) ; } else { AdapterJS . renderNotificationBar ( AdapterJS . TEXT . PLUGIN . NOT _SUPPORTED ) ; } } ; AdapterJS . WebRTCPlugin . isPluginInstalled ( AdapterJS . WebRTCPlugin . pluginInfo . prefix , AdapterJS . WebRTCPlugin . pluginInfo . plugName , AdapterJS . WebRTCPlugin . defineWebRTCInterface , AdapterJS . WebRTCPlugin . pluginNeededButNotInstalledCb ) ; }