|
/** |
|
* This example code represents a typical workflow for Verto in a client |
|
* browser. |
|
* |
|
* It can be used as a bare-bones starting point to hang your application |
|
* specific code around. |
|
* |
|
* There's a lot of inline documentation below, so read on. :) |
|
*/ |
|
|
|
/* |
|
* Globally instantiated variables. |
|
*/ |
|
// Verto object. |
|
var vertoObj; |
|
// Verto configuration object. |
|
var vertoConf; |
|
// Actively placed call object. |
|
var currentCall; |
|
// Live array object. |
|
var liveArray; |
|
|
|
/* |
|
* Initialize Verto. |
|
*/ |
|
// This includes: |
|
// - Detecting audio/video/sound devices |
|
// - Getting user audio/video permissions |
|
// NOTE: These steps can be done manually instead. |
|
$.verto.init({}, function() { |
|
|
|
// Create a new verto instance: |
|
// This step performs a user login to FreeSWITCH via secure websocket. |
|
// The user must be properly configured in the FreeSWITCH user directory. |
|
vertoObj = new $.verto({ |
|
login: '[email protected]', |
|
passwd: 'supersekret', |
|
// As configured in verto.conf.xml on the server. |
|
socketUrl: 'wss://freeswitch.example.com:8082', |
|
// ID of HTML video tag where the video feed is placed. |
|
tag: "video-feed", |
|
// STUN/TURN server config, more than one is allowed. |
|
iceServers: [ |
|
{ |
|
url: 'stun:stun.example.com', |
|
}, |
|
], |
|
// Internal session ID used by Verto to track the call, eg. for call |
|
// recovery. A random one will be generated if none is provided, and, |
|
// it can be useful to provide a custom ID to store and reference for |
|
// other purposes. |
|
sessid: sessid, |
|
// Google Chrome specific adjustments/filters for audio. |
|
// Official documentation is scant, best to try them out and see! |
|
audioParams: { |
|
googEchoCancellation: true, |
|
googAutoGainControl: true, |
|
googNoiseSuppression: true, |
|
googHighpassFilter: true, |
|
googTypingNoiseDetection: true, |
|
googEchoCancellation2: false, |
|
googAutoGainControl2: false, |
|
}, |
|
// These can be set per-call as well as per-login. |
|
deviceParams: { |
|
useCamera: 'any', |
|
useMic: 'any', |
|
useSpeak: 'any', |
|
}, |
|
}, callbacks); |
|
|
|
}); |
|
|
|
/* |
|
* Handle websocket messages from the freeswitch server. |
|
*/ |
|
var callbacks = { |
|
|
|
// Receives websocket login status from FreeSWITCH. |
|
onWSLogin: function (verto, success) { |
|
if (success) { |
|
// At this point you're connected to the FreeSWITCH server and logged |
|
// in, ready to place a call to the conference, or run a test for the |
|
// user's bandwidth. |
|
testBandwidth(function(bandwidthTestData) { |
|
// Do something with the bandwidth test results... |
|
}); |
|
} |
|
}, |
|
|
|
// Websocket connection to FreeSWITCH closed. |
|
onWSClose: function (verto, success) { |
|
// Perhaps try to reconnect? |
|
}, |
|
|
|
// Receives call state messages from FreeSWITCH. |
|
onDialogState: function (d) { |
|
switch (d.state.name) { |
|
case "trying": |
|
break; |
|
case "answering": |
|
break; |
|
case "active": |
|
break; |
|
case "hangup": |
|
log("Call ended with cause: " + d.cause); |
|
break; |
|
case "destroy": |
|
// Some kind of client side cleanup... |
|
break; |
|
} |
|
}, |
|
|
|
// Receives conference-related messages from FreeSWITCH. |
|
// Note that it's possible to write server-side modules to send customized |
|
// messages via this callback. |
|
onMessage: function (verto, dialog, message, data) { |
|
switch (message) { |
|
case $.verto.enum.message.pvtEvent: |
|
if (data.pvtData) { |
|
switch (data.pvtData.action) { |
|
// This client has joined the live array for the conference. |
|
case "conference-liveArray-join": |
|
// With the initial live array data from the server, you can |
|
// configure/subscribe to the live array. |
|
initLiveArray(verto, dialog, data); |
|
break; |
|
// This client has left the live array for the conference. |
|
case "conference-liveArray-part": |
|
// Some kind of client-side wrapup... |
|
break; |
|
} |
|
} |
|
break; |
|
} |
|
}, |
|
|
|
}; |
|
|
|
/* |
|
* Place a call to the videoconference. |
|
*/ |
|
var call = function(bandwidthTestData) { |
|
|
|
// Sets the parameters for the video stream that will be sent to the |
|
// videoconference. |
|
// Hint: Use the upKPS result in bandwidthTestData to determine the video |
|
// resolution to send! |
|
vertoObj.videoParams({ |
|
// Dimensions of the video feed to send. |
|
minWidth: 640, |
|
minHeight: 480, |
|
maxWidth: 640, |
|
maxHeight: 480, |
|
// The minimum frame rate of the client camera, Verto will fail if it's |
|
// less than this. |
|
minFrameRate: 15, |
|
// The maximum frame rate to send from the camera. |
|
vertoBestFrameRate: 30, |
|
}); |
|
|
|
currentCall = vertoObj.newCall({ |
|
// Extension to dial. |
|
destination_number: 'conference_1', |
|
caller_id_name: 'Bob Smith', |
|
caller_id_number: 'some-caller-or-user-id', |
|
// Enable video support. |
|
useVideo: true, |
|
// User devices to use. |
|
useCamera: 'any', |
|
useMic: 'any', |
|
useSpeak: 'any', |
|
// Data returned from the bandwidth test can be used to set these params, |
|
// which will be used to calculate the best strategy for sending/receiving |
|
// video within these bandwidth limits. |
|
outgoingBandwidth: bandwidthTestData.upKPS, |
|
incomingBandwidth: bandwidthTestData.downKPS, |
|
// Use a dedicated outbound encoder for this user's video. |
|
// NOTE: This is generally only needed if the user has some kind of |
|
// non-standard video setup, and is not recommended to use, as it |
|
// dramatically increases the CPU usage for the conference. |
|
dedEnc: false, |
|
// You can pass any application/call specific variables here, and they will |
|
// be available as a dialplan variable, prefixed with 'verto_dvar_'. |
|
userVariables: { |
|
// Shows up as a 'verto_dvar_foo' dialplan variable. |
|
foo: 'bar', |
|
}, |
|
// Enable stereo audio. |
|
useStereo: true, |
|
// Mirror local user's webcam. |
|
mirrorInput: true, |
|
}); |
|
} |
|
|
|
/* |
|
* Setting up and subscribing to the live array. |
|
*/ |
|
var initLiveArray = function(verto, dialog, data) { |
|
// Set up addtional configuration specific to the call. |
|
vertoConf = new $.verto.conf(verto, { |
|
dialog: dialog, |
|
hasVid: true, |
|
laData: data.pvtData, |
|
// For subscribing to published chat messages. |
|
chatCallback: function(verto, eventObj) { |
|
var from = eventObj.data.fromDisplay || eventObj.data.from || 'Unknown'; |
|
var message = eventObj.data.message || ''; |
|
}, |
|
}); |
|
var config = { |
|
subParams: { |
|
callID: dialog ? dialog.callID : null |
|
}, |
|
}; |
|
// Set up the live array, using the live array data received from FreeSWITCH. |
|
liveArray = new $.verto.liveArray(vertoObj, data.pvtData.laChannel, data.pvtData.laName, config); |
|
// Subscribe to live array changes. |
|
liveArray.onChange = function(liveArrayObj, args) { |
|
console.log("Call UUID is: " + args.key); |
|
console.log("Call data is: ", args.data); |
|
try { |
|
switch (args.action) { |
|
|
|
// Initial list of existing conference users. |
|
case "bootObj": |
|
break; |
|
|
|
// New user joined conference. |
|
case "add": |
|
break; |
|
|
|
// User left conference. |
|
case "del": |
|
break; |
|
|
|
// Existing user's state changed (mute/unmute, talking, floor, etc) |
|
case "modify": |
|
break; |
|
|
|
} |
|
} catch (err) { |
|
console.error("ERROR: " + err); |
|
} |
|
}; |
|
// Called if the live array throws an error. |
|
liveArray.onErr = function (obj, args) { |
|
console.error("Error: ", obj, args); |
|
}; |
|
} |
|
|
|
/* |
|
* Send a command to the conference. |
|
*/ |
|
// This translates to the following conference API command: |
|
// conference [conference id] [command] [id] [value] |
|
var sendCommand = function (command, id, value) { |
|
vertoObj.rpcClient.call("verto.broadcast", { |
|
"eventChannel": vertoConf.params.laData.modChannel, |
|
"data": { |
|
"application": "conf-control", |
|
"command": command, |
|
"id": id, |
|
"value": value |
|
} |
|
}); |
|
} |
|
|
|
// Some examples of using the above sendCommand() function. |
|
var toggleAudioMute = function(conferenceUserId, arg) { |
|
sendCommand('tmute', conferenceUserId, arg); |
|
} |
|
var toggleVideoMute = function(conferenceUserId, arg) { |
|
sendCommand('tvmute', conferenceUserId, arg); |
|
} |
|
var videoLayout = function(layoutId) { |
|
sendCommand("vid-layout", null, layoutId); |
|
} |
|
|
|
/* |
|
* Send a key press to the conference. |
|
*/ |
|
// currentCall is the object returned by vertoObj.newCall() |
|
// The sent key digits map to the <caller-controls> section of |
|
// conference.conf.xml |
|
var memberMute = function() { |
|
// Normally audio mute/unmute. |
|
currentCall.dtmf("0"); |
|
// Normally video mute/unmute. |
|
currentCall.dtmf("*0"); |
|
} |
|
|
|
/* |
|
* Send a chat message. |
|
*/ |
|
var sendConferenceChat = function(message) { |
|
vertoConf.sendChat(message, "message"); |
|
} |
|
|
|
/* |
|
* Perform bandwidth test. |
|
*/ |
|
// This can be performed any time after a successful Verto login to the |
|
// FreeSWITCH server. |
|
var testBandwidth = function(someCallback) { |
|
vertoObj.rpcClient.speedTest(1024 * 256, function(event, data) { |
|
// These values are in kilobits/sec. |
|
var upBand = Math.ceil(data.upKPS); |
|
var downBand = Math.ceil(data.downKPS); |
|
console.log('[BANDWIDTH TEST] Up: ' + upBand + ', Down: ' + downBand); |
|
if(someCallback) { |
|
someCallback(data); |
|
} |
|
}); |
|
} |