放縱了這麼多天,又到了開學的時候,終於想到不能這麼無所事事下去,正巧遇到同學在寫Python聊天室,想到能不能實現一個web版的聊天室呢?
後臺用nodejs,客戶端與服務端通信用socket.io —— 這是一個比較成熟的websocket框架了。
nodeJs是一個好東西,尤其是在處理消息通訊,網絡編程方面,天生的異步IO配合V8引擎,嘖嘖
初始工作
- 安裝express, 用這個來託管socket.io,以及靜態頁面,命令npm install express --save,–save可以使包添加到package.json文件裏.
- 安裝socket.io,命令npm install socket.io --save.
編寫服務端代碼
首先我們通過express來託管網站,並附加到socket.io實例裏,因爲socket.io初次連接需要http協議。代碼如下:
var express = require('express'),
io = require('socket.io');
var app = express();
app.use(express.static(__dirname));
var server = app.listen(8888);
var ws = io.listen(server);
當客戶端連接成功之後,發公告告訴所有在線用戶,並且,當用戶發送消息時,發廣播通知其它用戶 —— 這需要用到“監聽連接事件”:
ws.on('connection', function(client){
client.on('join', function(msg){
// 檢查是否有重複
if(checkNickname(msg)){
client.emit('nickname', '暱稱有重複!');
}else{
client.nickname = msg;
ws.sockets.emit('announcement', '用戶 ', msg + ' 加入了聊天室!');
}
});
// 監聽發送消息
client.on('send.message', function(msg){
client.broadcast.emit('send.message',client.nickname, msg);
});
// 斷開連接時,通知其它用戶
client.on('disconnect', function(){
if(client.nickname){
client.broadcast.emit('send.message','用戶', client.nickname + '已離開聊天室!');
}
})
})
上面代碼我用了這麼一個函數checkNickname
:由於客戶端是通過暱稱來標識的,所以服務端需要一個檢測暱稱重複的函數
// 檢查暱稱是否重複
var checkNickname = function(name){
for(var k in ws.sockets.sockets){
if(ws.sockets.sockets.hasOwnProperty(k)){
if(ws.sockets.sockets[k] && ws.sockets.sockets[k].nickname == name){
return true;
}
}
}
return false;
}
至此,服務端代碼算是開發完成。
編寫客戶端代碼
由於服務端採用第三方websokcet框架,所以前端頁面需要單獨引用socket.io客戶端代碼,源文件可以從socket.io模塊裏找,windows下路徑爲node_modules\socket.io\node_modules\socket.io-client\dist。筆者下載後發現有開發版和壓縮版的,默認引用開發版就行.
前端主要處理輸入暱稱檢查,消息處理,完整代碼如下:
<!DOCTYPE html>
<html>
<head>
<title>socket.io實現聊天室</title>
<meta charset="utf-8">
</head>
<body>
<div class="wrapper">
<div class="content" id="chat">
<ul id="chat_conatiner">
</ul>
</div>
<div class="action">
<textarea ></textarea>
<button class="btn btn-success" id="clear">清屏</button>
<button class="btn btn-success" id="send">發送</button>
</div>
</div>
<script type="text/javascript" src="js/socket.io.js"></script>
<script type="text/javascript">
var ws = io.connect('http://172.0.0.1:8081');
var sendMsg = function(msg){
ws.emit('send.message', msg);
}
var addMessage = function(from, msg){
var li = document.createElement('li');
li.innerHTML = '<span>' + from + '</span>' + ' : ' + msg;
document.querySelector('#chat_conatiner').appendChild(li);
// 設置內容區的滾動條到底部
document.querySelector('#chat').scrollTop = document.querySelector('#chat').scrollHeight;
// 並設置焦點
document.querySelector('textarea').focus();
}
var send = function(){
var ele_msg = document.querySelector('textarea');
var msg = ele_msg.value.replace('\r\n', '').trim();
console.log(msg);
if(!msg) return;
sendMsg(msg);
// 添加消息到自己的內容區
addMessage('你', msg);
ele_msg.value = '';
}
ws.on('connect', function(){
var nickname = window.prompt('輸入你的暱稱!');
while(!nickname){
nickname = window.prompt('暱稱不能爲空,請重新輸入!')
}
ws.emit('join', nickname);
});
// 暱稱有重複
ws.on('nickname', function(){
var nickname = window.prompt('暱稱有重複,請重新輸入!');
while(!nickname){
nickname = window.prompt('暱稱不能爲空,請重新輸入!')
}
ws.emit('join', nickname);
});
ws.on('send.message', function(from, msg){
addMessage(from, msg);
});
ws.on('announcement', function(from, msg){
addMessage(from, msg);
});
document.querySelector('textarea').addEventListener('keypress', function(event){
if(event.which == 13){
send();
}
});
document.querySelector('textarea').addEventListener('keydown', function(event){
if(event.which == 13){
send();
}
});
document.querySelector('#send').addEventListener('click', function(){
send();
});
document.querySelector('#clear').addEventListener('click', function(){
document.querySelector('#chat_conatiner').innerHTML = '';
});
</script>
</body>
</html>