2013-02-27 23:22:10 +00:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<title>Data Channel Demo 1</title>
|
|
|
|
<style>
|
|
|
|
button {
|
|
|
|
font: 18px sans-serif;
|
|
|
|
padding: 8px;
|
|
|
|
}
|
|
|
|
textarea {
|
|
|
|
font-family: monospace;
|
|
|
|
margin: 2px;
|
2013-09-23 18:03:33 +00:00
|
|
|
height: 400px;
|
|
|
|
width: 300px;
|
|
|
|
}
|
|
|
|
div#send {
|
|
|
|
float: left;
|
|
|
|
margin-right: 20px;
|
|
|
|
}
|
|
|
|
div#receive {
|
|
|
|
}
|
|
|
|
div#sendreceive {
|
|
|
|
margin: 0 0 20px 0;
|
|
|
|
}
|
|
|
|
h2 {
|
|
|
|
margin: 0 0 10px 0;
|
2013-02-27 23:22:10 +00:00
|
|
|
}
|
|
|
|
</style>
|
|
|
|
</head>
|
|
|
|
<body>
|
2013-09-23 18:03:33 +00:00
|
|
|
<div id="sendreceive">
|
|
|
|
<div id="send">
|
2013-02-27 23:22:10 +00:00
|
|
|
<h2>Send data</h2>
|
2013-09-23 18:03:33 +00:00
|
|
|
<textarea id="dataChannelSendId" rows="5" cols="15" disabled="true"
|
|
|
|
placeholder="Press Start, enter some text, then press Send Data.">
|
|
|
|
</textarea>
|
2013-02-27 23:22:10 +00:00
|
|
|
</div>
|
2013-09-23 18:03:33 +00:00
|
|
|
<div id="receive">
|
2013-02-27 23:22:10 +00:00
|
|
|
<h2>Received Data</h2>
|
2013-09-23 18:03:33 +00:00
|
|
|
<textarea id="dataChannelReceiveId" rows="5" cols="15" disabled="true">
|
|
|
|
</textarea>
|
|
|
|
</div>
|
2013-02-27 23:22:10 +00:00
|
|
|
</div>
|
2013-09-23 18:03:33 +00:00
|
|
|
<form>
|
|
|
|
<p>Choose SCTP or RTP for transmitting data.</p>
|
|
|
|
<input type="radio" id="useSctp" name="transportbtn" checked/>
|
|
|
|
<label for="useSctp">Use SCTP</label>
|
|
|
|
<input type="radio" id="useRtp" name="transportbtn"/>
|
|
|
|
<label for="useRtp">Use RTP</label>
|
|
|
|
</form>
|
|
|
|
<button id="startButton">Start</button>
|
|
|
|
<button id="sendButton" disabled>Send Data</button>
|
|
|
|
<button id="closeButton" disabled>Stop</button>
|
|
|
|
<!-- Load the polyfill to switch-hit between Chrome and Firefox -->
|
|
|
|
<script src='../../base/adapter.js'></script>
|
2013-02-27 23:22:10 +00:00
|
|
|
<script>
|
2013-09-23 18:03:33 +00:00
|
|
|
var pc1, pc2, sendChannel, receiveChannel, pcConstraint, dataConstraint;
|
|
|
|
var dataChannelSend = document.getElementById("dataChannelSendId");
|
|
|
|
var dataChannelReceive = document.getElementById("dataChannelReceiveId");
|
|
|
|
var sctp_select = document.getElementById('useSctp');
|
|
|
|
var rtp_select = document.getElementById('useRtp');
|
|
|
|
var startButton = document.querySelector('button#startButton');
|
|
|
|
var sendButton = document.querySelector('button#sendButton');
|
|
|
|
var closeButton = document.querySelector('button#closeButton');
|
|
|
|
startButton.onclick = createConnection;
|
|
|
|
sendButton.onclick = sendData;
|
|
|
|
closeButton.onclick = closeDataChannels;
|
|
|
|
rtp_select.onclick = enableStartButton;
|
|
|
|
sctp_select.onclick = enableStartButton;
|
2013-02-27 23:22:10 +00:00
|
|
|
|
2013-09-23 18:03:33 +00:00
|
|
|
function enableStartButton() {
|
|
|
|
startButton.disabled = false;
|
2013-02-27 23:22:10 +00:00
|
|
|
}
|
2013-09-23 18:03:33 +00:00
|
|
|
|
2013-10-24 19:38:47 +00:00
|
|
|
function disableSendButton() {
|
|
|
|
sendButton.disabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
rtp_select.onclick = sctp_select.onclick = function() {
|
|
|
|
dataChannelReceive.value = '';
|
|
|
|
dataChannelSend.value = '';
|
|
|
|
disableSendButton();
|
|
|
|
enableStartButton();
|
|
|
|
};
|
|
|
|
|
2013-02-27 23:22:10 +00:00
|
|
|
function createConnection() {
|
2013-09-23 18:03:33 +00:00
|
|
|
dataChannelSendId.placeholder = "";
|
2013-02-27 23:22:10 +00:00
|
|
|
var servers = null;
|
2013-10-24 19:38:47 +00:00
|
|
|
pcConstraint = null;
|
|
|
|
dataConstraint = null;
|
2013-09-23 18:03:33 +00:00
|
|
|
if (sctp_select.checked &&
|
|
|
|
webrtcDetectedBrowser === 'chrome' &&
|
|
|
|
webrtcDetectedVersion >= 31) {
|
2013-10-24 19:38:47 +00:00
|
|
|
// SCTP is supported from Chrome M31.
|
|
|
|
// No need to pass DTLS constraint as it is on by default in Chrome M31.
|
|
|
|
// For SCTP, reliable and ordered is true by default.
|
2013-09-23 18:03:33 +00:00
|
|
|
trace('Using SCTP based Data Channels');
|
|
|
|
} else {
|
2013-10-24 19:38:47 +00:00
|
|
|
pcConstraint = {optional: [{RtpDataChannels: true}]};
|
2013-09-23 18:03:33 +00:00
|
|
|
if (!rtp_select.checked) {
|
|
|
|
// Use rtp data channels for chrome versions older than M31.
|
|
|
|
trace('Using RTP based Data Channels,' +
|
|
|
|
'as you are on an older version than M31.');
|
|
|
|
alert('Reverting to RTP based data channels,' +
|
|
|
|
'as you are on an older version than M31.');
|
|
|
|
rtp_select.checked = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pc1 = new RTCPeerConnection(servers, pcConstraint);
|
2013-02-27 23:22:10 +00:00
|
|
|
trace('Created local peer connection object pc1');
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Data Channel api supported from Chrome M25.
|
2013-09-23 18:03:33 +00:00
|
|
|
// You might need to start chrome with --enable-data-channels flag.
|
|
|
|
sendChannel = pc1.createDataChannel("sendDataChannel", dataConstraint);
|
2013-02-27 23:22:10 +00:00
|
|
|
trace('Created send data channel');
|
|
|
|
} catch (e) {
|
|
|
|
alert('Failed to create data channel. ' +
|
|
|
|
'You need Chrome M25 or later with --enable-data-channels flag');
|
|
|
|
trace('Create Data channel failed with exception: ' + e.message);
|
|
|
|
}
|
|
|
|
pc1.onicecandidate = iceCallback1;
|
|
|
|
sendChannel.onopen = onSendChannelStateChange;
|
|
|
|
sendChannel.onclose = onSendChannelStateChange;
|
|
|
|
|
2013-09-23 18:03:33 +00:00
|
|
|
pc2 = new RTCPeerConnection(servers, pcConstraint);
|
2013-02-27 23:22:10 +00:00
|
|
|
trace('Created remote peer connection object pc2');
|
|
|
|
|
|
|
|
pc2.onicecandidate = iceCallback2;
|
|
|
|
pc2.ondatachannel = receiveChannelCallback;
|
|
|
|
|
|
|
|
pc1.createOffer(gotDescription1);
|
|
|
|
startButton.disabled = true;
|
|
|
|
closeButton.disabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function sendData() {
|
2013-09-23 18:03:33 +00:00
|
|
|
var data = dataChannelSend.value;
|
2013-02-27 23:22:10 +00:00
|
|
|
sendChannel.send(data);
|
|
|
|
trace('Sent Data: ' + data);
|
|
|
|
}
|
|
|
|
|
|
|
|
function closeDataChannels() {
|
|
|
|
trace('Closing data Channels');
|
|
|
|
sendChannel.close();
|
|
|
|
trace('Closed data channel with label: ' + sendChannel.label);
|
|
|
|
receiveChannel.close();
|
|
|
|
trace('Closed data channel with label: ' + receiveChannel.label);
|
|
|
|
pc1.close();
|
|
|
|
pc2.close();
|
|
|
|
pc1 = null;
|
|
|
|
pc2 = null;
|
|
|
|
trace('Closed peer connections');
|
|
|
|
startButton.disabled = false;
|
|
|
|
sendButton.disabled = true;
|
|
|
|
closeButton.disabled = true;
|
2013-09-23 18:03:33 +00:00
|
|
|
dataChannelSend.value = "";
|
|
|
|
dataChannelReceive.value = "";
|
|
|
|
dataChannelSend.disabled = true;
|
2013-02-27 23:22:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function gotDescription1(desc) {
|
|
|
|
pc1.setLocalDescription(desc);
|
|
|
|
trace('Offer from pc1 \n' + desc.sdp);
|
|
|
|
pc2.setRemoteDescription(desc);
|
|
|
|
pc2.createAnswer(gotDescription2);
|
|
|
|
}
|
|
|
|
|
|
|
|
function gotDescription2(desc) {
|
|
|
|
pc2.setLocalDescription(desc);
|
|
|
|
trace('Answer from pc2 \n' + desc.sdp);
|
|
|
|
pc1.setRemoteDescription(desc);
|
|
|
|
}
|
|
|
|
|
|
|
|
function iceCallback1(event) {
|
|
|
|
trace('local ice callback');
|
|
|
|
if (event.candidate) {
|
|
|
|
pc2.addIceCandidate(event.candidate);
|
|
|
|
trace('Local ICE candidate: \n' + event.candidate.candidate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function iceCallback2(event) {
|
|
|
|
trace('remote ice callback');
|
|
|
|
if (event.candidate) {
|
|
|
|
pc1.addIceCandidate(event.candidate);
|
|
|
|
trace('Remote ICE candidate: \n ' + event.candidate.candidate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function receiveChannelCallback(event) {
|
|
|
|
trace('Receive Channel Callback');
|
|
|
|
receiveChannel = event.channel;
|
|
|
|
receiveChannel.onmessage = onReceiveMessageCallback;
|
|
|
|
receiveChannel.onopen = onReceiveChannelStateChange;
|
|
|
|
receiveChannel.onclose = onReceiveChannelStateChange;
|
|
|
|
}
|
|
|
|
|
|
|
|
function onReceiveMessageCallback(event) {
|
|
|
|
trace('Received Message');
|
2013-09-23 18:03:33 +00:00
|
|
|
dataChannelReceive.value = event.data;
|
2013-02-27 23:22:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function onSendChannelStateChange() {
|
|
|
|
var readyState = sendChannel.readyState;
|
|
|
|
trace('Send channel state is: ' + readyState);
|
|
|
|
if (readyState == "open") {
|
2013-09-23 18:03:33 +00:00
|
|
|
dataChannelSend.disabled = false;
|
|
|
|
dataChannelSendId.focus();
|
2013-02-27 23:22:10 +00:00
|
|
|
sendButton.disabled = false;
|
|
|
|
closeButton.disabled = false;
|
|
|
|
} else {
|
2013-09-23 18:03:33 +00:00
|
|
|
dataChannelSend.disabled = true;
|
2013-02-27 23:22:10 +00:00
|
|
|
sendButton.disabled = true;
|
|
|
|
closeButton.disabled = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onReceiveChannelStateChange() {
|
|
|
|
var readyState = receiveChannel.readyState;
|
|
|
|
trace('Receive channel state is: ' + readyState);
|
|
|
|
}
|
|
|
|
|
|
|
|
</script>
|
|
|
|
</body>
|
|
|
|
</html>
|