WebRTC - टेक्स्ट डेमो
इस अध्याय में, हम एक ग्राहक अनुप्रयोग बनाने जा रहे हैं जो दो उपयोगकर्ताओं को अलग-अलग उपकरणों पर वेबआरटीसी का उपयोग करके एक-दूसरे को संदेश भेजने की अनुमति देता है। हमारे आवेदन में दो पेज होंगे। एक लॉगिन के लिए और दूसरा दूसरे उपयोगकर्ता को संदेश भेजने के लिए।
दो पेज div टैग होंगे। अधिकांश इनपुट सरल ईवेंट हैंडलर के माध्यम से किया जाता है।
सिग्नलिंग सर्वर
WebRTC कनेक्शन बनाने के लिए क्लाइंट को WebRTC सहकर्मी कनेक्शन का उपयोग किए बिना संदेशों को स्थानांतरित करने में सक्षम होना चाहिए। यह वह जगह है जहां हम HTML5 WebSockets - दो एंडपॉइंट के बीच एक द्विदिश सॉकेट कनेक्शन - एक वेब सर्वर और एक वेब ब्राउज़र का उपयोग करेंगे। अब WebSocket लाइब्रेरी का उपयोग शुरू करते हैं। Server.js फ़ाइल बनाएँ और निम्नलिखित कोड डालें -
//require our websocket library
var WebSocketServer = require('ws').Server;
//creating a websocket server at port 9090
var wss = new WebSocketServer({port: 9090});
//when a user connects to our sever
wss.on('connection', function(connection) {
console.log("user connected");
//when server gets a message from a connected user
connection.on('message', function(message) {
console.log("Got message from a user:", message);
});
connection.send("Hello from server");
});
पहली पंक्ति के लिए वेबस्केट लाइब्रेरी की आवश्यकता होती है जिसे हमने पहले ही स्थापित कर दिया है। फिर हम पोर्ट 9090 पर एक सॉकेट सर्वर बनाते हैं। इसके बाद, हम कनेक्शन ईवेंट को सुनते हैं । यह कोड तब निष्पादित किया जाएगा जब कोई उपयोगकर्ता सर्वर से WebSocket कनेक्शन बनाता है। फिर हम उपयोगकर्ता द्वारा भेजे गए किसी भी संदेश को सुनते हैं। अंत में, हम कनेक्टेड यूज़र को "सर्वर से हैलो" कहते हुए एक प्रतिक्रिया भेजें।
हमारे सिग्नलिंग सर्वर में, हम प्रत्येक कनेक्शन के लिए एक स्ट्रिंग-आधारित उपयोगकर्ता नाम का उपयोग करेंगे ताकि हमें पता चले कि संदेश कहां भेजना है। आइए हमारे कनेक्शन हैंडलर को थोड़ा बदलें -
connection.on('message', function(message) {
var data;
//accepting only JSON messages
try {
data = JSON.parse(message);
} catch (e) {
console.log("Invalid JSON");
data = {};
}
});
इस तरह हम केवल JSON संदेशों को स्वीकार करते हैं। अगला, हमें सभी कनेक्ट किए गए उपयोगकर्ताओं को कहीं स्टोर करने की आवश्यकता है। हम इसके लिए एक सरल जावास्क्रिप्ट ऑब्जेक्ट का उपयोग करेंगे। हमारी फ़ाइल के शीर्ष को बदलें -
//require our websocket library
var WebSocketServer = require('ws').Server;
//creating a websocket server at port 9090
var wss = new WebSocketServer({port: 9090});
//all connected to the server users
var users = {};
हम क्लाइंट से आने वाले प्रत्येक संदेश के लिए एक प्रकार का फ़ील्ड जोड़ने जा रहे हैं । उदाहरण के लिए यदि कोई उपयोगकर्ता लॉगिन करना चाहता है, तो वह लॉगिन प्रकार संदेश भेजता है । आइए इसे परिभाषित करते हैं -
connection.on('message', function(message) {
var data;
//accepting only JSON messages
try {
data = JSON.parse(message);
} catch (e) {
console.log("Invalid JSON");
data = {};
}
//switching type of the user message
switch (data.type) {
//when a user tries to login
case "login":
console.log("User logged:", data.name);
//if anyone is logged in with this username then refuse
if(users[data.name]) {
sendTo(connection, {
type: "login",
success: false
});
} else {
//save user connection on the server
users[data.name] = connection;
connection.name = data.name;
sendTo(connection, {
type: "login",
success: true
});
}
break;
default:
sendTo(connection, {
type: "error",
message: "Command no found: " + data.type
});
break;
}
});
यदि उपयोगकर्ता लॉगिन प्रकार के साथ एक संदेश भेजता है , तो हम -
- जांचें कि क्या कोई पहले से ही इस उपयोगकर्ता नाम के साथ लॉग इन कर चुका है।
- यदि ऐसा है, तो उपयोगकर्ता को बताएं कि उसने सफलतापूर्वक लॉगिन नहीं किया है।
- यदि कोई इस उपयोगकर्ता नाम का उपयोग नहीं कर रहा है, तो हम कनेक्शन ऑब्जेक्ट की कुंजी के रूप में उपयोगकर्ता नाम जोड़ते हैं।
- यदि कोई कमांड नहीं पहचानी जाती है तो हम एक त्रुटि भेजते हैं।
निम्नलिखित कोड एक कनेक्शन के लिए संदेश भेजने के लिए एक सहायक कार्य है। इसे server.js फ़ाइल में जोड़ें -
function sendTo(connection, message) {
connection.send(JSON.stringify(message));
}
जब उपयोगकर्ता डिस्कनेक्ट करता है तो हमें उसके कनेक्शन को साफ करना चाहिए। जब करीबी घटना को निकाल दिया जाता है तो हम उपयोगकर्ता को हटा सकते हैं । कनेक्शन हैंडलर में निम्न कोड जोड़ें -
connection.on("close", function() {
if(connection.name) {
delete users[connection.name];
}
});
सफल लॉगिन के बाद उपयोगकर्ता दूसरे को कॉल करना चाहता है। उसे इसे प्राप्त करने के लिए किसी अन्य उपयोगकर्ता को एक प्रस्ताव देना चाहिए । प्रस्ताव हैंडलर जोड़ें -
case "offer":
//for ex. UserA wants to call UserB
console.log("Sending offer to: ", data.name);
//if UserB exists then send him offer details
var conn = users[data.name];
if(conn != null){
//setting that UserA connected with UserB
connection.otherName = data.name;
sendTo(conn, {
type: "offer",
offer: data.offer,
name: connection.name
});
break;
सबसे पहले, हम उस उपयोगकर्ता का कनेक्शन प्राप्त करते हैं जिसे हम कॉल करने का प्रयास कर रहे हैं। यदि यह मौजूद है तो हम उसे प्रस्ताव विवरण भेजते हैं । हम यह भी जोड़ना otherName को कनेक्शन वस्तु। इसे बाद में खोजने की सरलता के लिए बनाया गया है।
प्रतिक्रिया के जवाब में एक समान पैटर्न है जो हमने प्रस्ताव हैंडलर में उपयोग किया था । हमारा सर्वर अन्य उपयोगकर्ताओं के उत्तर के रूप में सभी संदेशों से गुजरता है । प्रस्ताव हैंडलर के बाद निम्नलिखित कोड जोड़ें -
case "answer":
console.log("Sending answer to: ", data.name);
//for ex. UserB answers UserA
var conn = users[data.name];
if(conn != null) {
connection.otherName = data.name;
sendTo(conn, {
type: "answer",
answer: data.answer
});
}
break;
अंतिम भाग उपयोगकर्ताओं के बीच आईसीई उम्मीदवार को संभाल रहा है। हम उसी तकनीक का उपयोग करते हैं जो उपयोगकर्ताओं के बीच संदेश भेजती है मुख्य अंतर यह है कि उम्मीदवार संदेश किसी भी क्रम में प्रति उपयोगकर्ता कई बार हो सकता है। उम्मीदवार हैंडलर जोड़ें -
case "candidate":
console.log("Sending candidate to:",data.name);
var conn = users[data.name];
if(conn != null) {
sendTo(conn, {
type: "candidate",
candidate: data.candidate
});
}
break;
हमारे उपयोगकर्ताओं को किसी अन्य उपयोगकर्ता से डिस्कनेक्ट करने की अनुमति देने के लिए हमें हैंगिंग फ़ंक्शन को लागू करना चाहिए। यह सभी उपयोगकर्ता संदर्भों को हटाने के लिए सर्वर को भी बताएगा। छुट्टी हैंडलर जोड़ें -
case "leave":
console.log("Disconnecting from", data.name);
var conn = users[data.name];
conn.otherName = null;
//notify the other user so he can disconnect his peer connection
if(conn != null) {
sendTo(conn, {
type: "leave"
});
}
break;
यह अन्य उपयोगकर्ता को छुट्टी की घटना भी भेजेगा ताकि वह अपने सहकर्मी कनेक्शन को तदनुसार काट सके। हमें उस मामले को भी संभालना चाहिए जब कोई उपयोगकर्ता सिग्नलिंग सर्वर से अपना कनेक्शन छोड़ देता है। आइए हमारे करीबी हैंडलर को संशोधित करें -
connection.on("close", function() {
if(connection.name) {
delete users[connection.name];
if(connection.otherName) {
console.log("Disconnecting from ", connection.otherName);
var conn = users[connection.otherName];
conn.otherName = null;
if(conn != null) {
sendTo(conn, {
type: "leave"
});
}
}
}
});
हमारे सिग्नलिंग सर्वर का पूरा कोड निम्नलिखित है -
//require our websocket library
var WebSocketServer = require('ws').Server;
//creating a websocket server at port 9090
var wss = new WebSocketServer({port: 9090});
//all connected to the server users
var users = {};
//when a user connects to our sever
wss.on('connection', function(connection) {
console.log("User connected");
//when server gets a message from a connected user
connection.on('message', function(message) {
var data;
//accepting only JSON messages
try {
data = JSON.parse(message);
} catch (e) {
console.log("Invalid JSON");
data = {};
}
//switching type of the user message
switch (data.type) {
//when a user tries to login
case "login":
console.log("User logged", data.name);
//if anyone is logged in with this username then refuse
if(users[data.name]) {
sendTo(connection, {
type: "login",
success: false
});
} else {
//save user connection on the server
users[data.name] = connection;
connection.name = data.name;
sendTo(connection, {
type: "login",
success: true
});
}
break;
case "offer":
//for ex. UserA wants to call UserB
console.log("Sending offer to: ", data.name);
//if UserB exists then send him offer details
var conn = users[data.name];
if(conn != null) {
//setting that UserA connected with UserB
connection.otherName = data.name;
sendTo(conn, {
type: "offer",
offer: data.offer,
name: connection.name
});
}
break;
case "answer":
console.log("Sending answer to: ", data.name);
//for ex. UserB answers UserA
var conn = users[data.name];
if(conn != null) {
connection.otherName = data.name;
sendTo(conn, {
type: "answer",
answer: data.answer
});
}
break;
case "candidate":
console.log("Sending candidate to:",data.name);
var conn = users[data.name];
if(conn != null) {
sendTo(conn, {
type: "candidate",
candidate: data.candidate
});
}
break;
case "leave":
console.log("Disconnecting from", data.name);
var conn = users[data.name];
conn.otherName = null;
//notify the other user so he can disconnect his peer connection
if(conn != null) {
sendTo(conn, {
type: "leave"
});
}
break;
default:
sendTo(connection, {
type: "error",
message: "Command not found: " + data.type
});
break;
}
});
//when user exits, for example closes a browser window
//this may help if we are still in "offer","answer" or "candidate" state
connection.on("close", function() {
if(connection.name) {
delete users[connection.name];
if(connection.otherName) {
console.log("Disconnecting from ", connection.otherName);
var conn = users[connection.otherName];
conn.otherName = null;
if(conn != null) {
sendTo(conn, {
type: "leave"
});
}
}
}
});
connection.send("Hello world");
});
function sendTo(connection, message) {
connection.send(JSON.stringify(message));
}
ग्राहक आवेदन
इस एप्लिकेशन का परीक्षण करने का एक तरीका दो ब्राउज़र टैब खोल रहा है और एक दूसरे को संदेश भेजने की कोशिश कर रहा है।
सबसे पहले, हमें बूटस्ट्रैप लाइब्रेरी स्थापित करने की आवश्यकता है । बूटस्ट्रैप वेब अनुप्रयोगों को विकसित करने के लिए एक दृश्यमान रूपरेखा है। आप और अधिक सीख सकते हैंhttp://getbootstrap.com/.उदाहरण के लिए, "टेक्स्टचैट" नामक एक फ़ोल्डर बनाएँ। यह हमारा रूट एप्लीकेशन फोल्डर होगा। इस फ़ोल्डर के अंदर एक फ़ाइल पैकेज बनाएँ । Json (यह npm निर्भरता के प्रबंधन के लिए आवश्यक है) और निम्नलिखित जोड़ें -
{
"name": "webrtc-textochat",
"version": "0.1.0",
"description": "webrtc-textchat",
"author": "Author",
"license": "BSD-2-Clause"
}
फिर npm इंस्टॉल बूटस्ट्रैप चलाएं । यह टेक्स्टचैट / node_modules फ़ोल्डर में बूटस्ट्रैप लाइब्रेरी स्थापित करेगा ।
अब हमें एक मूल HTML पेज बनाने की आवश्यकता है। निम्नलिखित कोड के साथ रूट फ़ोल्डर में एक index.html फ़ाइल बनाएँ -
<html>
<head>
<title>WebRTC Text Demo</title>
<link rel = "stylesheet" href = "node_modules/bootstrap/dist/css/bootstrap.min.css"/>
</head>
<style>
body {
background: #eee;
padding: 5% 0;
}
</style>
<body>
<div id = "loginPage" class = "container text-center">
<div class = "row">
<div class = "col-md-4 col-md-offset-4">
<h2>WebRTC Text Demo. Please sign in</h2>
<label for = "usernameInput" class = "sr-only">Login</label>
<input type = "email" id = "usernameInput"
class = "form-control formgroup" placeholder = "Login"
required = "" autofocus = "">
<button id = "loginBtn" class = "btn btn-lg btn-primary btnblock">
Sign in</button>
</div>
</div>
</div>
<div id = "callPage" class = "call-page container">
<div class = "row">
<div class = "col-md-4 col-md-offset-4 text-center">
<div class = "panel panel-primary">
<div class = "panel-heading">Text chat</div>
<div id = "chatarea" class = "panel-body text-left"></div>
</div>
</div>
</div>
<div class = "row text-center form-group">
<div class = "col-md-12">
<input id = "callToUsernameInput" type = "text"
placeholder = "username to call" />
<button id = "callBtn" class = "btn-success btn">Call</button>
<button id = "hangUpBtn" class = "btn-danger btn">Hang Up</button>
</div>
</div>
<div class = "row text-center">
<div class = "col-md-12">
<input id = "msgInput" type = "text" placeholder = "message" />
<button id = "sendMsgBtn" class = "btn-success btn">Send</button>
</div>
</div>
</div>
<script src = "client.js"></script>
</body>
</html>
यह पृष्ठ आपको परिचित होना चाहिए। हमने बूटस्ट्रैप css फ़ाइल जोड़ दी है । हमने दो पृष्ठ भी परिभाषित किए हैं। अंत में, हमने उपयोगकर्ता से जानकारी प्राप्त करने के लिए कई पाठ फ़ील्ड और बटन बनाए हैं। "चैट" पृष्ठ पर आपको "चैटरेया" आईडी के साथ div टैग देखना चाहिए जहां हमारे सभी संदेश प्रदर्शित किए जाएंगे। ध्यान दें कि हमने एक client.js फ़ाइल के लिए एक लिंक जोड़ा है ।
अब हमें अपने सिग्नलिंग सर्वर के साथ एक कनेक्शन स्थापित करने की आवश्यकता है। निम्नलिखित कोड के साथ रूट फ़ोल्डर में client.js फ़ाइल बनाएँ -
//our username
var name;
var connectedUser;
//connecting to our signaling server
var conn = new WebSocket('ws://localhost:9090');
conn.onopen = function () {
console.log("Connected to the signaling server");
};
//when we got a message from a signaling server
conn.onmessage = function (msg) {
console.log("Got message", msg.data);
var data = JSON.parse(msg.data);
switch(data.type) {
case "login":
handleLogin(data.success);
break;
//when somebody wants to call us
case "offer":
handleOffer(data.offer, data.name);
break;
case "answer":
handleAnswer(data.answer);
break;
//when a remote peer sends an ice candidate to us
case "candidate":
handleCandidate(data.candidate);
break;
case "leave":
handleLeave();
break;
default:
break;
}
};
conn.onerror = function (err) {
console.log("Got error", err);
};
//alias for sending JSON encoded messages
function send(message) {
//attach the other peer username to our messages
if (connectedUser) {
message.name = connectedUser;
}
conn.send(JSON.stringify(message));
};
अब नोड सर्वर के माध्यम से हमारे सिग्नलिंग सर्वर को चलाएं । फिर, रूट फ़ोल्डर के अंदर स्टेटिक कमांड चलाएं और ब्राउज़र के अंदर पेज खोलें। आपको निम्न कंसोल आउटपुट देखना चाहिए -
अगला चरण एक अद्वितीय उपयोगकर्ता नाम के साथ एक उपयोगकर्ता लॉग को लागू कर रहा है। हम केवल सर्वर पर एक उपयोगकर्ता नाम भेजते हैं, जो तब हमें बताता है कि यह लिया गया है या नहीं। निम्न कोड को अपने client.js फ़ाइल में जोड़ें -
//******
//UI selectors block
//******
var loginPage = document.querySelector('#loginPage');
var usernameInput = document.querySelector('#usernameInput');
var loginBtn = document.querySelector('#loginBtn');
var callPage = document.querySelector('#callPage');
var callToUsernameInput = document.querySelector('#callToUsernameInput');
var callBtn = document.querySelector('#callBtn');
var hangUpBtn = document.querySelector('#hangUpBtn');
callPage.style.display = "none";
// Login when the user clicks the button
loginBtn.addEventListener("click", function (event) {
name = usernameInput.value;
if (name.length > 0) {
send({
type: "login",
name: name
});
}
});
function handleLogin(success) {
if (success === false) {
alert("Ooops...try a different username");
} else {
loginPage.style.display = "none";
callPage.style.display = "block";
//**********************
//Starting a peer connection
//**********************
}
};
सबसे पहले, हम पृष्ठ के तत्वों के कुछ संदर्भों का चयन करते हैं। हम कॉल पेज छिपाते हैं। फिर, हम लॉगिन बटन पर एक ईवेंट श्रोता जोड़ते हैं। जब उपयोगकर्ता इसे क्लिक करता है, तो हम सर्वर पर उसका उपयोगकर्ता नाम भेजते हैं। अंत में, हम हैंडललॉग कॉलबैक लागू करते हैं। यदि लॉगिन सफल था, तो हम कॉल पेज दिखाते हैं, एक सहकर्मी कनेक्शन सेट करते हैं, और एक डेटा चैनल बनाते हैं।
डेटा चैनल के साथ सहकर्मी कनेक्शन शुरू करने के लिए हमें चाहिए -
- RTCPeerConnection ऑब्जेक्ट बनाएँ
- हमारे RTCPeerConnection ऑब्जेक्ट के अंदर एक डेटा चैनल बनाएं
निम्नलिखित कोड को "UI चयनकर्ताओं ब्लॉक" में जोड़ें -
var msgInput = document.querySelector('#msgInput');
var sendMsgBtn = document.querySelector('#sendMsgBtn');
var chatArea = document.querySelector('#chatarea');
var yourConn;
var dataChannel;
हैंडललोगिन फ़ंक्शन को संशोधित करें -
function handleLogin(success) {
if (success === false) {
alert("Ooops...try a different username");
} else {
loginPage.style.display = "none";
callPage.style.display = "block";
//**********************
//Starting a peer connection
//**********************
//using Google public stun server
var configuration = {
"iceServers": [{ "url": "stun:stun2.1.google.com:19302" }]
};
yourConn = new webkitRTCPeerConnection(configuration, {optional: [{RtpDataChannels: true}]});
// Setup ice handling
yourConn.onicecandidate = function (event) {
if (event.candidate) {
send({
type: "candidate",
candidate: event.candidate
});
}
};
//creating data channel
dataChannel = yourConn.createDataChannel("channel1", {reliable:true});
dataChannel.onerror = function (error) {
console.log("Ooops...error:", error);
};
//when we receive a message from the other peer, display it on the screen
dataChannel.onmessage = function (event) {
chatArea.innerHTML += connectedUser + ": " + event.data + "<br />";
};
dataChannel.onclose = function () {
console.log("data channel is closed");
};
}
};
यदि लॉगिन सफल होता है तो एप्लिकेशन RTCPeerConnection ऑब्जेक्ट और सेटअप onicecandidate हैंडलर बनाता है, जो अन्य पाए गए सभी icecandidates को भेजता है। यह एक डेटाचैनल भी बनाता है। ध्यान दें, कि जब RTCPeerConnection ऑब्जेक्ट को कंस्ट्रक्टर वैकल्पिक में दूसरा तर्क बनाते हैं: [{RtpDataChannels: true}] अनिवार्य है यदि आप क्रोम या ओपेरा का उपयोग कर रहे हैं। अगला कदम दूसरे सहकर्मी को एक प्रस्ताव बनाना है। एक बार उपयोगकर्ता को प्रस्ताव मिलने के बाद, वह एक उत्तर बनाता है और ICE उम्मीदवारों का व्यापार शुरू करता है । निम्नलिखित कोड को client.js फ़ाइल में जोड़ें -
//initiating a call
callBtn.addEventListener("click", function () {
var callToUsername = callToUsernameInput.value;
if (callToUsername.length > 0) {
connectedUser = callToUsername;
// create an offer
yourConn.createOffer(function (offer) {
send({
type: "offer",
offer: offer
});
yourConn.setLocalDescription(offer);
}, function (error) {
alert("Error when creating an offer");
});
}
});
//when somebody sends us an offer
function handleOffer(offer, name) {
connectedUser = name;
yourConn.setRemoteDescription(new RTCSessionDescription(offer));
//create an answer to an offer
yourConn.createAnswer(function (answer) {
yourConn.setLocalDescription(answer);
send({
type: "answer",
answer: answer
});
}, function (error) {
alert("Error when creating an answer");
});
};
//when we got an answer from a remote user
function handleAnswer(answer) {
yourConn.setRemoteDescription(new RTCSessionDescription(answer));
};
//when we got an ice candidate from a remote user
function handleCandidate(candidate) {
yourConn.addIceCandidate(new RTCIceCandidate(candidate));
};
हम कॉल बटन पर एक क्लिक हैंडलर जोड़ते हैं , जो एक प्रस्ताव शुरू करता है। फिर हम onmessage हैंडलर द्वारा अपेक्षित कई हैंडलर लागू करते हैं । जब तक दोनों उपयोगकर्ताओं ने एक कनेक्शन नहीं किया है, तब तक उन्हें एसिंक्रोनस रूप से संसाधित किया जाएगा।
अगला कदम हैंग-अप सुविधा को लागू करना है। यह डेटा प्रसारित करना बंद कर देगा और अन्य उपयोगकर्ता को डेटा चैनल बंद करने के लिए कहेगा। निम्नलिखित कोड जोड़ें -
//hang up
hangUpBtn.addEventListener("click", function () {
send({
type: "leave"
});
handleLeave();
});
function handleLeave() {
connectedUser = null;
yourConn.close();
yourConn.onicecandidate = null;
};
जब उपयोगकर्ता हैंग अप बटन पर क्लिक करता है -
- यह दूसरे उपयोगकर्ता को "छुट्टी" संदेश भेजेगा।
- यह RTCPeerConnection और साथ ही डेटा चैनल को बंद कर देगा।
अंतिम चरण दूसरे सहकर्मी को संदेश भेज रहा है। "संदेश भेजें" बटन पर "क्लिक करें" हैंडलर जोड़ें -
//when user clicks the "send message" button
sendMsgBtn.addEventListener("click", function (event) {
var val = msgInput.value;
chatArea.innerHTML += name + ": " + val + "<br />";
//sending a message to a connected peer
dataChannel.send(val);
msgInput.value = "";
});
अब कोड को रन करें। आपको दो ब्राउज़र टैब का उपयोग करके सर्वर में लॉग इन करने में सक्षम होना चाहिए। फिर आप दूसरे उपयोगकर्ता के लिए सहकर्मी कनेक्शन सेट कर सकते हैं और उसे एक संदेश भेज सकते हैं और साथ ही "हैंग अप" बटन पर क्लिक करके डेटा चैनल बंद कर सकते हैं।
निम्नलिखित संपूर्ण ग्राहक है। जेएस फाइल -
//our username
var name;
var connectedUser;
//connecting to our signaling server
var conn = new WebSocket('ws://localhost:9090');
conn.onopen = function () {
console.log("Connected to the signaling server");
};
//when we got a message from a signaling server
conn.onmessage = function (msg) {
console.log("Got message", msg.data);
var data = JSON.parse(msg.data);
switch(data.type) {
case "login":
handleLogin(data.success);
break;
//when somebody wants to call us
case "offer":
handleOffer(data.offer, data.name);
break;
case "answer":
handleAnswer(data.answer);
break;
//when a remote peer sends an ice candidate to us
case "candidate":
handleCandidate(data.candidate);
break;
case "leave":
handleLeave();
break;
default:
break;
}
};
conn.onerror = function (err) {
console.log("Got error", err);
};
//alias for sending JSON encoded messages
function send(message) {
//attach the other peer username to our messages
if (connectedUser) {
message.name = connectedUser;
}
conn.send(JSON.stringify(message));
};
//******
//UI selectors block
//******
var loginPage = document.querySelector('#loginPage');
var usernameInput = document.querySelector('#usernameInput');
var loginBtn = document.querySelector('#loginBtn');
var callPage = document.querySelector('#callPage');
var callToUsernameInput = document.querySelector('#callToUsernameInput');
var callBtn = document.querySelector('#callBtn');
var hangUpBtn = document.querySelector('#hangUpBtn');
var msgInput = document.querySelector('#msgInput');
var sendMsgBtn = document.querySelector('#sendMsgBtn');
var chatArea = document.querySelector('#chatarea');
var yourConn;
var dataChannel;
callPage.style.display = "none";
// Login when the user clicks the button
loginBtn.addEventListener("click", function (event) {
name = usernameInput.value;
if (name.length > 0) {
send({
type: "login",
name: name
});
}
});
function handleLogin(success) {
if (success === false) {
alert("Ooops...try a different username");
} else {
loginPage.style.display = "none";
callPage.style.display = "block";
//**********************
//Starting a peer connection
//**********************
//using Google public stun server
var configuration = {
"iceServers": [{ "url": "stun:stun2.1.google.com:19302" }]
};
yourConn = new webkitRTCPeerConnection(configuration, {optional: [{RtpDataChannels: true}]});
// Setup ice handling
yourConn.onicecandidate = function (event) {
if (event.candidate) {
send({
type: "candidate",
candidate: event.candidate
});
}
};
//creating data channel
dataChannel = yourConn.createDataChannel("channel1", {reliable:true});
dataChannel.onerror = function (error) {
console.log("Ooops...error:", error);
};
//when we receive a message from the other peer, display it on the screen
dataChannel.onmessage = function (event) {
chatArea.innerHTML += connectedUser + ": " + event.data + "<br />";
};
dataChannel.onclose = function () {
console.log("data channel is closed");
};
}
};
//initiating a call
callBtn.addEventListener("click", function () {
var callToUsername = callToUsernameInput.value;
if (callToUsername.length > 0) {
connectedUser = callToUsername;
// create an offer
yourConn.createOffer(function (offer) {
send({
type: "offer",
offer: offer
});
yourConn.setLocalDescription(offer);
}, function (error) {
alert("Error when creating an offer");
});
}
});
//when somebody sends us an offer
function handleOffer(offer, name) {
connectedUser = name;
yourConn.setRemoteDescription(new RTCSessionDescription(offer));
//create an answer to an offer
yourConn.createAnswer(function (answer) {
yourConn.setLocalDescription(answer);
send({
type: "answer",
answer: answer
});
}, function (error) {
alert("Error when creating an answer");
});
};
//when we got an answer from a remote user
function handleAnswer(answer) {
yourConn.setRemoteDescription(new RTCSessionDescription(answer));
};
//when we got an ice candidate from a remote user
function handleCandidate(candidate) {
yourConn.addIceCandidate(new RTCIceCandidate(candidate));
};
//hang up
hangUpBtn.addEventListener("click", function () {
send({
type: "leave"
});
handleLeave();
});
function handleLeave() {
connectedUser = null;
yourConn.close();
yourConn.onicecandidate = null;
};
//when user clicks the "send message" button
sendMsgBtn.addEventListener("click", function (event) {
var val = msgInput.value;
chatArea.innerHTML += name + ": " + val + "<br />";
//sending a message to a connected peer
dataChannel.send(val);
msgInput.value = "";
});