实战web聊天室(express+socket.io):进入、聊天、重名检测

放纵了这么多天,又到了开学的时候,终于想到不能这么无所事事下去,正巧遇到同学在写Python聊天室,想到能不能实现一个web版的聊天室呢?

后台用nodejs,客户端与服务端通信用socket.io —— 这是一个比较成熟的websocket框架了。


nodeJs是一个好东西,尤其是在处理消息通讯,网络编程方面,天生的异步IO配合V8引擎,啧啧


初始工作

  1. 安装express, 用这个来托管socket.io,以及静态页面,命令npm install express --save,–save可以使包添加到package.json文件里.
  2. 安装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>
发布了200 篇原创文章 · 获赞 417 · 访问量 6万+

猜你喜欢

转载自blog.csdn.net/qq_43624878/article/details/104363119