前言

前断时间在学习nodejs,自己闲来无事,在网上搜索了一些资料自己搭建了一个属于自己的web聊天室项目。现在把自己的开发过程和心得和大家分享,希望其中涉及到的一些知识对你有用。

项目开源地址:https://github.com/yunchen132/node-websocket-Chatroom

开发工具准备

安装nodejs,你可以在nodejs官网下载安装适合自己操作系统的版本。#nodejs官网
使用npm安装开发环境依赖。在这里我使用了express,socket.io模块。

功能介绍

  • 支持图片发送
  • 支持表情发送
  • 支持键盘回车发送信息
  • 支持在线用户统计和用户列表:
我的目录结构
server.js代码:
var express = require('express'),
    app = express(),
    server = require('http').createServer(app),
    io = require('socket.io').listen(server),
    users = [];

app.use('/', express.static(__dirname + '/app'));

server.listen(process.env.PORT || 3000);
io.sockets.on('connection', function(socket) {//创建用户链接
    socket.on('login', function(user) {console.log(user);
        if (isHave(user)) {socket.emit('nickExisted');
        } else {var address=socket.handshake.address.address+":"+socket.handshake.address.port;
            user.address=address;
            socket.userIndex = users.length;
            socket.user = user;
            users.push(user);
            socket.emit('loginSuccess');
            io.sockets.emit('system', user,users, users.length, 'login');
        };
    });
    //用户注销链接
    socket.on('disconnect', function() {if (socket.user != null) {users.splice(socket.userIndex, 1);
            socket.broadcast.emit('system', socket.user,users,users.length, 'logout');
        }});
    //新建消息
    socket.on('postMsg', function(msg, color) {socket.broadcast.emit('newMsg', socket.user, msg, color);
    });
    //新建图片信息
    socket.on('img', function(imgData, color) {socket.broadcast.emit('newImg', socket.user, imgData, color);
    });
    //判断用户名是否存在
    function isHave(user) {var flag=false;
        for(var i=0;i<users.length;i++){if(users[i].nickName==user.nickName){flag=true;
                break;
            }}return flag;
    }
});

index.html代码:

<!doctype html>
<html>
    <head>
        <meta charset="utf-8">
        <meta name="renderer" content="webkit">
        <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
        <meta name="author" content="cleverqin">
        <meta name="description" content="Chat聊天室,使用nodejs和websocket构建的网页聊天室">
        <meta name="viewport" content="width=device-width, initial-scale=1">
        <title>Chat聊天室</title>
        <link rel="stylesheet" href="styles/main.css">
        <link rel="shortcut icon" href="favicon.ico" type="image/x-icon">
    </head>
    <body>
        <div class='chat-box'>
            <div class='user-box'>
                <div class='tag'>在线用户:</div>
                <ul id='online-list'></ul>
            </div>
            <div class="wrapper">
                <div class="banner">
                    <div class='chat-title'>欢迎进入Chat聊天室</div>
                    <span id="status"></span>
                </div>
                <div id="historyMsg"></div>
                <div class="controls" >
                    <div class="items">
                        <input id="colorStyle" type="color" placeHolder='#000' title="font color" />
                        <input id="emoji" type="button" value="表情" title="emoji" />
                        <label for="sendImage" class="imageLable">
                            <input type="button" value="图片"  />
                            <input id="sendImage" type="file" value=""/>
                        </label>
                        <input id="clearBtn" type="button" value="清除聊天记录" title="clear screen" />
                    </div>
                    <textarea id="messageInput" placeHolder="请输入消息....."></textarea>
                    <input id="sendBtn" type="button" value="发送">
                    <div id="emojiWrapper">
                    </div>
                </div>
            </div>
        </div>
        <div id="loginWrapper">
            <p id="info">链接服务器...</p>
            <div id="nickWrapper" >
                <div class='login-form'>
                    <div class='form-group'>
                        <label>选择头像:</label>
                        <div class="select-pic-box form-control">
                            <ul class="img-list">
                                <li class="checked"><img src="content/headPic/1.jpg"></li>
                                <li><img src="content/headPic/2.jpg"></li>
                                <li><img src="content/headPic/3.jpg"></li>
                                <li><img src="content/headPic/4.jpg"></li>
                                <li><img src="content/headPic/5.jpg"></li>
                                <li><img src="content/headPic/6.jpg"></li>
                                <li><img src="content/headPic/7.jpg"></li>
                                <li><img src="content/headPic/8.jpg"></li>
                                <li><img src="content/headPic/9.jpg"></li>
                                <li><img src="content/headPic/10.jpg"></li>
                            </ul>
                        </div>
                    </div>
                    <div class='form-group'>
                        <label>填写昵称:</label>
                        <div class='form-control'><input type="text" placeHolder="请输入你的昵称" id="nicknameInput" /></div>
                    </div>
                    <div class="form-group btn-box"><input type="button" value="确定" id="loginBtn" /></div>
                </div>
            </div>
        </div>
        <script src="scripts/jquery-1.11.1.min.js"></script>
        <script src="/socket.io/socket.io.js"></script>
        <script src="scripts/canvas.js"></script>
        <script src="scripts/client.js"></script>
    </body>
</html>

client.js代码

window.onload = function() {var hichat = new MyChat();
    hichat.init();
};
var MyChat = function() {this.socket = null;
};
MyChat.prototype = {init: function() {var that = this;
        var _pic=document.body.querySelectorAll("#nickWrapper .img-list li.checked img")[0].src;
        var curUser={nickName:"我",
            pic:_pic,
            address:""
        }this.socket = io.connect();
        this.socket.on('connect', function() {document.getElementById('info').textContent = '请输入一个昵称:';
            document.getElementById('nickWrapper').style.display = 'block';
            document.getElementById('nicknameInput').focus();
        });
        this.socket.on('nickExisted', function() {document.getElementById('info').textContent = '该昵称已被使用,请选用其他昵称';
        });
        this.socket.on('loginSuccess', function() {document.title = 'Chat聊天室 | ' + document.getElementById('nicknameInput').value;
            document.getElementById('loginWrapper').style.display = 'none';
            document.getElementById('messageInput').focus();
            curUser.nickName="我";
        });
        this.socket.on('error', function(err) {if (document.getElementById('loginWrapper').style.display == 'none') {document.getElementById('status').textContent = '链接失败!';
            } else {document.getElementById('info').textContent = '链接失败!';
            }});
        this.socket.on('system', function(user,users, userCount, type) {var msg = user.nickName + (type == 'login' ? ' 加入聊天室' : ' 离开聊天室');
            that._displayMsg(user, msg, 'red','sys');
            document.getElementById('status').innerHTML = "当前在线用户:<span class='num-tag'>"+userCount+"</span>人";
            var listDiv=document.getElementById("online-list");
            var str="";
            for (var i=0;i<users.length;i++){str+="<li class='user-span' title='"+users[i].nickName+"'><img src='"+users[i].pic+"'><div>"+users[i].nickName+"</div></li>";
            }listDiv.innerHTML=str;
        });
        this.socket.on('newMsg', function(user, msg, color) {that._displayMsg(user, msg, color,"msg");
        });
        this.socket.on('newImg', function(user, img, color) {that._displayMsg(user,img,color,"img");
        });
        document.getElementById('loginBtn').addEventListener('click', function() {var nickName = document.getElementById('nicknameInput').value;
            var pic=document.body.querySelectorAll("#nickWrapper .img-list li.checked img")[0].src;
            if (nickName.trim().length != 0&&nickName.trim()!="我") {curUser={nickName:nickName,
                    pic:pic}that.socket.emit('login', curUser);
            } else {document.getElementById('nicknameInput').focus();
                document.getElementById('info').textContent = '请输入昵称,昵称不能为空不能为’我';
            };
        }, false);
        document.getElementById('nicknameInput').addEventListener('keyup', function(e) {if (e.keyCode == 13) {var nickName = document.getElementById('nicknameInput').value;
                var pic=document.body.querySelectorAll("#nickWrapper .img-list li.checked img")[0].src;
                if (nickName.trim().length != 0) {curUser={nickName:nickName,
                        pic:pic}that.socket.emit('login', curUser);
                };
            };
        }, false);
        document.getElementById('sendBtn').addEventListener('click', function() {var messageInput = document.getElementById('messageInput'),
                msg = messageInput.value,
                color = document.getElementById('colorStyle').value;
            messageInput.value = '';
            messageInput.focus();
            if (msg.trim().length != 0) {that.socket.emit('postMsg', msg, color);
                that._displayMsg(curUser, msg, color,"msg");
                return;
            };
        }, false);
        document.getElementById('messageInput').addEventListener('keyup', function(e) {var messageInput = document.getElementById('messageInput'),
                msg = messageInput.value,
                color = document.getElementById('colorStyle').value;
            if (e.keyCode == 13 && msg.trim().length != 0) {messageInput.value = '';
                that.socket.emit('postMsg', msg, color);
                that._displayMsg(curUser, msg, color,"msg");
            };
        }, false);
        document.getElementById('clearBtn').addEventListener('click', function() {document.getElementById('historyMsg').innerHTML = '';
        }, false);
        document.getElementById('sendImage').addEventListener('change', function() {if (this.files.length != 0) {var file = this.files[0],
                    reader = new FileReader(),
                    color = document.getElementById('colorStyle').value;
                if (!reader) {that._displayMsg(curUser, '你的浏览器不支持文件读取!', 'red',"sys");
                    this.value = '';
                    return;
                };
                reader.onload = function(e) {this.value = '';
                    that.socket.emit('img', e.target.result, color);
                    that._displayMsg(curUser, e.target.result, color,"img");
                };
                reader.readAsDataURL(file);
            };
        }, false);
        this._initialEmoji();
        this.initSelectPic();
        document.getElementById('emoji').addEventListener('click', function(e) {var emojiwrapper = document.getElementById('emojiWrapper');
            emojiwrapper.style.display = 'block';
            e.stopPropagation();
        }, false);
        document.body.addEventListener('click', function(e) {var emojiwrapper = document.getElementById('emojiWrapper');
            if (e.target != emojiwrapper) {emojiwrapper.style.display = 'none';
            };
        });
        document.getElementById('emojiWrapper').addEventListener('click', function(e) {var target = e.target;
            if (target.nodeName.toLowerCase() == 'img') {var messageInput = document.getElementById('messageInput');
                messageInput.focus();
                messageInput.value = messageInput.value + '[emoji:' + target.title + ']';
            };
        }, false);

    },
    //初始化表情
    _initialEmoji: function() {var emojiContainer = document.getElementById('emojiWrapper'),
            docFragment = document.createDocumentFragment();
        for (var i = 69; i > 0; i--) {var emojiItem = document.createElement('img');
            emojiItem.src = '../content/emoji/' + i + '.gif';
            emojiItem.title = i;
            docFragment.appendChild(emojiItem);
        };
        emojiContainer.appendChild(docFragment);
    },
    _displayMsg:function(user,msg,color,type) {var container = document.getElementById('historyMsg'),
            msgToDisplay = document.createElement('p'),
            date = new Date().toTimeString().substr(0, 8),
            msgDiv="",
            userDiv="";
        msgToDisplay.style.color = color || '#000';
        if(type=="sys"){msgDiv=msg;
            msgToDisplay.className="sys";
            msgToDisplay.innerHTML = "系统消息" + '<span class="timespan">(' + date + '): </span>' + msgDiv;
        }else if(type=="img"){userDiv="<div class='user-msg'><img src='"+user.pic+"' time='"+date+"' /><span>"+user.nickName+"</span></div>"
            if(user.nickName=="我"){msgToDisplay.className="me";
                userDiv="<div class='user-msg'><span>"+user.nickName+"</span><img src='"+user.pic+"' time='"+date+"' /></div>"
            }msgDiv="<div class='msg-box'><div class=msg-content><span class='msg-arrow'></span><a href='"+msg+"' target='_blank'><img src='"+msg+"'/></a></div></div>";
            msgToDisplay.innerHTML =userDiv+ msgDiv;
        }else {msg = this._showEmoji(msg);
            msgDiv="<div class='msg-box'><div class=msg-content><span class='msg-arrow'></span>"+msg+"</div></div>";
            userDiv="<div class='user-msg'><img src='"+user.pic+"' time='"+date+"' /><span>"+user.nickName+"</span></div>"
            if(user.nickName=="我"){msgToDisplay.className="me";
                userDiv="<div class='user-msg'><span>"+user.nickName+"</span><img src='"+user.pic+"' time='"+date+"' /></div>"
            }msgToDisplay.innerHTML = userDiv + msgDiv;
        }container.appendChild(msgToDisplay);
        container.scrollTop = container.scrollHeight;
    },
    //匹配表情信息
    _showEmoji: function(msg) {var match, result = msg,
            reg = /\[emoji:\d+\]/g,
            emojiIndex,
            totalEmojiNum = document.getElementById('emojiWrapper').children.length;
        while (match = reg.exec(msg)) {emojiIndex = match[0].slice(7, -1);
            if (emojiIndex > totalEmojiNum) {result = result.replace(match[0], '[X]');
            } else {result = result.replace(match[0], '<img class="emoji" src="../content/emoji/' + emojiIndex + '.gif" />');
            };
        };
        return result;
    },
    initSelectPic:function () {var imglist=document.body.querySelectorAll("#nickWrapper .img-list li");
        $(imglist).on("click",function (e) {e.stopPropagation();
            $(imglist).removeClass("checked");
            $(this).addClass("checked");
        })}
};

项目启动

  1. 使用’npm‘安装项目依赖模块
  2. 使用’node server.js‘启动服务
  3. 最后打开浏览器访问 http://localhost:3000

心得总结

这是我使用了express作为服务器路由框架,socket.io构建socket服务器。这其中涉及的了一些socket知识的。在开发过程中也遇到了一些问题。在这里我并未对前端界面显示作优化,仅使用了jQuery进行dom更新。当然你也可以用react,angular,vue来构建的前端界面。当然这里你也可以实现聊天室的创建,单用户的链接服务,单用户之间的私聊功能。甚至创建一个在线的网页客服系统。这里你可以参考socket.io的更多知识。具体请参考socket.io文档

项目截图

最后希望我写的东西对大家有所帮助,感谢你的支持!

使用nodejs搭建你自己的专属web聊天室相关推荐

  1. ASP.NET SignalR 与 LayIM2.0 配合轻松实现Web聊天室(一) 之 基层数据搭建,让数据活起来(数据获取)...

    大家好,本篇是接上一篇 ASP.NET SignalR 与 LayIM2.0 配合轻松实现Web聊天室(零) 前言  ASP.NET SignalR WebIM系列第二篇.本篇会带领大家将 LayIM ...

  2. Spring和WebSocket整合并建立简单的Web聊天室

    Spring和WebSocket整合并建立简单的Web聊天室 官方主页 Spring WebSocket 一.概述 WebSocket 是一种网络通信协议.RFC6455 定义了它的通信标准. Web ...

  3. 基于阿里云用C/C++做了一个http协议与TCP协议的web聊天室的服务器——《干饭聊天室》

    基于阿里云用C/C++做了一个http协议与TCP协议的web聊天室的服务器--<干饭聊天室> 在这里首先感谢前端小伙伴飞鸟 前端技术请看一款基于React.C++,使用TCP/HTTP协 ...

  4. Django项目--web聊天室

    需求 做一个web聊天室,主要练习前端ajax与后台的交互: 一对一聊天和群组聊天 添加用户为好友 搜索并添加群组 管理员可以审批用户加群请求,群管理员可以有多个,群管理员可以删除,添加禁言群友 与聊 ...

  5. 使用FastHttpApi构建多人Web聊天室

    为什么80%的码农都做不了架构师?>>>    一般在dotnet core下构建使用web服务应用都使用asp.net core,但通过FastHttpApi组建也可以方便地构建w ...

  6. 基于.NET SingalR,LayIM2.0实现的web聊天室

    LayIM官网 http://www.layui.com/doc/layim.html 博客教程:http://www.cnblogs.com/panzi/p/5767095.html 项目说明:基于 ...

  7. 利用html 5 websocket做个山寨版web聊天室(手写C#服务器)

    在之前的博客中提到过看到html5 的websocket后很感兴趣,终于可以摆脱长轮询(websocket之前的实现方式可以看看Developer Works上的一篇文章,有简单提到,同时也说了web ...

  8. WebSocket请求过程分析及实现Web聊天室

    WebSocket协议是基于TCP的一种新的协议.WebSocket最初在HTML5规范中被引用为TCP连接,作为基于TCP的套接字API的占位符.它实现了浏览器与服务器全双工(full-duplex ...

  9. LayIM 3.9.1与ASP.NET SignalR实现Web聊天室快速入门(五)之使用RabbitMQ缓存消息

    前言 本系列文章特点:使用ASP.NET SignalR和LayIM快速入门对接,实现一对一聊天,群聊,添加聊天群组,查找聊天记录等功能.源代码不包含LayIM的源代码,因为官方并没开源属于收费资源, ...

最新文章

  1. rancher部署项目Validation failed in API: Deployment.apps“”must be no more than 63 characters问题原因及解决方法
  2. (转)分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间)...
  3. 简单的python抢红包脚本-Python自动抢红包,超详细教程,再也不会错过微信红包了!...
  4. 云炬随笔20180703
  5. 文件上传api——MultipartFile
  6. 难度炸裂!DeepChange:一个新的超大规模的换衣行人再识别数据集
  7. 路由器05---多拨
  8. [转]自定义UITableView各种函数
  9. Excel制作抛硬币动态频率趋势曲线
  10. 可靠性五性分析标准和国军标参考(文件分享)
  11. 如何ubuntu上安装tecplot,教你怎么创建图标哦!
  12. Thinkphp 5.1安装
  13. 英语语法(1)——简单句
  14. [附源码]java毕业设计校园征兵及退役复原管理系统
  15. 一起赚美元⑥ | 创立Discourse开源论坛软件每月赚取12万美元的故事
  16. 【JVM学习篇】剖析JVM类加载机制
  17. 哈尔滨工业大学(深圳)本科毕业设计(论文)LaTeX模板:hitszthesis
  18. 网线8跟全通就能跑千兆?No!只是必要条件而非充分条件
  19. DevExpress皮肤样式
  20. AndroidVideoCache源码解读

热门文章

  1. 快速提高编程能力---一年时间可以改变很多事
  2. Django基础必备三神装(HttpResponse、render、redirect)
  3. python学习手册(第4版)第十一章
  4. Python小白的学习日志:分析化学中pH值计算小程序
  5. 中职计算机系统的基本构成教案,计算机系统组成教案(中职教育)
  6. oracle一个600错误的解决
  7. 代理模式:aspectj静态代理和jdk,cglib的动态代理的对比
  8. 001-决策树算法介绍
  9. python中复数的乘法_为什么复数乘法几乎与python中的整数乘法一样快?
  10. I.MX6 PHY fixup 调用流程 hacking