首页 web前端 js教程 如何实现node+express个性化聊天室?

如何实现node+express个性化聊天室?

Jun 07, 2018 pm 04:12 PM
express node 聊天室

这篇文章主要介绍了零基础实现node+express个性化聊天室的示例,现在分享给大家,也给大家做个参考。

本篇文章使用node+express+jquery写一个个性化聊天室,一起来get一下~(源码地址见文章末尾)

效果图

项目结构

实现功能

  1. 登录检测

  2. 系统自动提示用户状态(进入/离开)

  3. 显示在线用户

  4. 支持发送和接收消息

  5. 自定义字体颜色

  6. 支持发送表情

  7. 支持发送图片

下面将一一讲解如何实现

前期准备

node及npm环境、express、socket.io

具体实现

1、将聊天室部署到服务器

先用node搭建一个服务器,部署在localhost:3000端口,先尝试向浏览器发送一个“hello world”,新建server.js文件。

var app = require('express')(); // 引入express模块
var http = require('http').Server(app);

app.get('/', function(req, res){ // 路由为localhost:3000时向客户端响应“hello world”
 res.send(&#39;<h1>Hello world</h1>&#39;); // 发送数据
});

http.listen(3000, function(){ // 监听3000端口
 console.log(&#39;listening on *:3000&#39;); 
});
登录后复制

打开浏览器输入网址:localhost:3000是这样的

一个node服务器搭建成功。

接下来用express向浏览器返回一个html页面

#安装express模块
npm install --save express
登录后复制

将server.js的代码改一下:

var express = require(&#39;express&#39;);
var app = express();
var http = require(&#39;http&#39;).Server(app); 

// 路由为/默认www静态文件夹
app.use(&#39;/&#39;, express.static(__dirname + &#39;/www&#39;));
登录后复制

express.static(__dirname + '/www');是将www文件夹托管为静态资源,意味着这个文件夹里的文件(html、css、js)彼此可以用相对路径。在www文件夹中添加index.html文件以及相应的css(相应css代码就不贴了,详情见源码),如下,该页面用了font-awesome小图标

<!doctype html>
<html>
 <head>  
 <meta charset="utf-8">
 <meta name="viewport" content="width=device-width, initial-scale=1">
 <title>chat</title>
 <link rel="stylesheet" href="style/index.css" rel="external nofollow" >
 <link rel="stylesheet" href="style/font-awesome-4.7.0/css/font-awesome.min.css" rel="external nofollow" >
 </head>
 <body>
 <p class="all">
  <p class="name">
  <!-- <h2>请输入你的昵称</h2> -->
  <input type="text" id="name" placeholder="请输入昵称..." autocomplete="off"> 
  <button id="nameBtn">确 定</button>
  </p>
  <p class="main">
  <p class="header">
   <img src="image/logo.jpg">
   happy聊天室
  </p>
  <p id="container">
   <p class="conversation">
    <ul id="messages"></ul>
    <form action="">
     <p class="edit"> 
     <input type="color" id="color" value="#000000">
     <i title="双击取消选择" class="fa fa-smile-o" id="smile">
     </i><i title="双击取消选择" class="fa fa-picture-o" id="img"></i>
     <p class="selectBox"> 
      <p class="smile"> 
      </p>
      <p class="img"> 
      </p>
     </p>
     </p>
     <!-- autocomplete禁用自动完成功能 -->
     <textarea id="m"></textarea>
     <button class="btn rBtn" id="sub">发送</button>
     <button class="btn" id="clear">关闭</button>
    </form>
   </p>
   <p class="contacts">
   <h1>在线人员(<span id="num">0</span>)</h1>
   <ul id="users"></ul>
   <p>当前无人在线哟~</p>
   </p>
  </p>
  </p> 
 </p> 
 </body>
</html>
登录后复制

打开localhost:3000,会看到如下:

聊天室成功部署到服务器。

2、检测登录

在客户端和服务器之间传送消息需要用到socket.io

#安装socket.io模块
npm install --save socket.io
登录后复制

将server.js改动如下:

var app = require(&#39;express&#39;)();
var http = require(&#39;http&#39;).Server(app);
var io = require(&#39;socket.io&#39;)(http);

app.use(&#39;/&#39;, express.static(__dirname + &#39;/www&#39;));

io.on(&#39;connection&#39;, function(socket){ // 用户连接时触发
 console.log(&#39;a user connected&#39;);
});

http.listen(3000, function(){
 console.log(&#39;listening on *:3000&#39;);
});
登录后复制

当打开localhost:3000的时候会触发服务器端io的connection事件,会在服务器打印“a user connected”,但是我们想统计一下连接该服务器的用户人数,如果有用户连接就打印“n users connected”,n为用户人数,怎么办呢?

在server.js设置一个全局数组为user,每当一个用户连接成功就在连接事件中将用户的昵称push进user,打印user.length即可知道已成功连接用户的人数。

等一等。

在用户连接的时输入昵称登录,我们应该检测一下用户的昵称是否已存在,避免昵称相同的情况发生,在服务器监听一个登录事件来判断该情况,由于一切都发生在用户连接之后,所以触发事件应该写在connection事件的回调函数中。

io.on(&#39;connection&#39;, (socket)=> {
 // 渲染在线人员
 io.emit(&#39;disUser&#39;, usersInfo);

 // 登录,检测用户名
 socket.on(&#39;login&#39;, (user)=> {
  if(users.indexOf(user.name) > -1) { // 昵称是否存在
   socket.emit(&#39;loginError&#39;); // 触发客户端的登录失败事件
  } else {
   users.push(user.name); //储存用户的昵称
   usersInfo.push(user); // 储存用户的昵称和头像
   socket.emit(&#39;loginSuc&#39;); // 触发客户端的登录成功事件
   socket.nickname = user.name;
   io.emit(&#39;system&#39;, { // 向所有用户广播该用户进入房间
    name: user.name,
    status: &#39;进入&#39;
   });
   io.emit(&#39;disUser&#39;, usersInfo); // 渲染右侧在线人员信息
   console.log(users.length + &#39; user connect.&#39;); // 打印连接人数
  }
 });
登录后复制

system和disUser事件先不管,之后再说 区分io.emit(foo)、socket.emit(foo)、socket.broadcast.emit(foo)

io.emit(foo); //会触发所有客户端用户的foo事件
socket.emit(foo); //只触发当前客户端用户的foo事件
socket.broadcast.emit(foo); //触发除了当前客户端用户的其他用户的foo事件
登录后复制

接下来是客户端代码chat-client.js

$(function() {
  // io-client
  // 连接成功会触发服务器端的connection事件
  var socket = io();

  // 点击输入昵称
  $(&#39;#nameBtn&#39;).click(()=> { 
   var imgN = Math.floor(Math.random()*4)+1; // 随机分配头像
   if($(&#39;#name&#39;).val().trim()!==&#39;&#39;)
     socket.emit(&#39;login&#39;, { // 触发服务器端登录事件
      name: $(&#39;#name&#39;).val(),
      img: &#39;image/user&#39; + imgN + &#39;.jpg&#39;
     }); 
   return false; 
  });
  // 登录成功,隐藏登录层
  socket.on(&#39;loginSuc&#39;, ()=> { 
   $(&#39;.name&#39;).hide(); 
  })
  socket.on(&#39;loginError&#39;, ()=> {
   alert(&#39;用户名已存在,请重新输入!&#39;);
   $(&#39;#name&#39;).val(&#39;&#39;);
  }); 
});
登录后复制

倘若登录成功,会看到如下页面:

登录检测完成。

3、系统自动提示用户状态(进入/离开)

该功能是为了实现上图所示的系统提示“XXX进入聊天室”,在登录成功时触发system事件,向所有用户广播信息,注意此时用的是io.emit而不是socket.emit,客户端代码如下

// 系统提示消息
socket.on(&#39;system&#39;, (user)=> { 
 var data = new Date().toTimeString().substr(0, 8);
 $(&#39;#messages&#39;).append(`<p class=&#39;system&#39;><span>${data}</span><br /><span>${user.name} ${user.status}了聊天室<span></p>`);
 // 滚动条总是在最底部
 $(&#39;#messages&#39;).scrollTop($(&#39;#messages&#39;)[0].scrollHeight);
});
登录后复制

4、显示在线用户

客户端监听一个显示在线用户的事件disUser,在以下三个时间段服务器端就触发一次该事件重新渲染一次

  1. 程序开始启动时

  2. 每当用户进入房间

  3. 每当用户离开房间

// chat-client.js
// 显示在线人员
socket.on(&#39;disUser&#39;, (usersInfo)=> {
 displayUser(usersInfo);
});
// 显示在线人员
function displayUser(users) {
 $(&#39;#users&#39;).text(&#39;&#39;); // 每次都要重新渲染
 if(!users.length) {
  $(&#39;.contacts p&#39;).show();
 } else {
  $(&#39;.contacts p&#39;).hide();
 }
 $(&#39;#num&#39;).text(users.length);
 for(var i = 0; i < users.length; i++) {
  var $html = `<li>
   <img src="${users[i].img}">
   <span>${users[i].name}</span>
  </li>`;
  $(&#39;#users&#39;).append($html);
 }
}
登录后复制

5、支持发送和接收消息

用户发送消息时触发服务器端的sendMsg事件,并将消息内容作为参数,服务器端监听到sendMsg事件之后向其他所有用户广播该消息,用的socket.broadcast.emit(foo)

 // server.js
  // 发送消息事件
  socket.on(&#39;sendMsg&#39;, (data)=> {
    var img = &#39;&#39;;
    for(var i = 0; i < usersInfo.length; i++) {
      if(usersInfo[i].name == socket.nickname) {
        img = usersInfo[i].img;
      }
    }
    socket.broadcast.emit(&#39;receiveMsg&#39;, { // 向除了发送者之外的其他用户广播
      name: socket.nickname,
      img: img,
      msg: data.msg,
      color: data.color,
      side: &#39;left&#39;
    });
    socket.emit(&#39;receiveMsg&#39;, { // 向发送者发送消息,为什么分开发送?因为css样式不同
      name: socket.nickname,
      img: img,
      msg: data.msg,
      color: data.color,
      side: &#39;right&#39;
    });
  });
登录后复制

服务器端接受到来自用户的消息后会触发客户端的receiveMsg事件,并将用户发送的消息作为参数传递,该事件会向聊天面板添加聊天内容,以下为chat-client.js代码

// 点击按钮或回车键发送消息
  $(&#39;#sub&#39;).click(sendMsg);
  $(&#39;#m&#39;).keyup((ev)=> {
   if(ev.which == 13) {
    sendMsg();
   }
  });

  // 接收消息
  socket.on(&#39;receiveMsg&#39;, (obj)=> { // 将接收到的消息渲染到面板上
   $(&#39;#messages&#39;).append(` 
     <li class=&#39;${obj.side}&#39;>
     <img src="${obj.img}">
     <p>
      <span>${obj.name}</span>
      <p>${obj.msg}</p>
     </p>
    </li>
   `);
   // 滚动条总是在最底部
   $(&#39;#messages&#39;).scrollTop($(&#39;#messages&#39;)[0].scrollHeight);
  });


  // 发送消息
  function sendMsg() { 
   if($(&#39;#m&#39;).val() == &#39;&#39;) { // 输入消息为空
    alert(&#39;请输入内容!&#39;);
    return false;
   }
   socket.emit(&#39;sendMsg&#39;, {
    msg: $(&#39;#m&#39;).val()
   });
   $(&#39;#m&#39;).val(&#39;&#39;); 
   return false; 
  }
登录后复制

6、自定义字体颜色

得益于html5的input新特性,可以通过type为color的input调用系统调色板

<!-- $(&#39;#color&#39;).val();为选中颜色,格式为#FFCCBB -->
<input type=&#39;color&#39; id=&#39;color&#39;>
登录后复制

客户端根据用户选择的颜色渲染内容样式,代码很容易看懂,这里就不赘述了。

7、支持发送表情

发送表情其实很简单,将表情图片放在li中,当用户点击li时就将表情的src中的序号解析出来,用[emoji+表情序号]的格式存放在聊天框里,点击发送后再解析为src。就是一个解析加还原的过程,这一过程中我们的服务器代码不变,需要改变的是客户端监听的receiveMsg事件。

// chat-client.js

  // 显示表情选择面板
  $(&#39;#smile&#39;).click(()=> {
   $(&#39;.selectBox&#39;).css(&#39;display&#39;, "block");
  });
  $(&#39;#smile&#39;).dblclick((ev)=> { 
   $(&#39;.selectBox&#39;).css(&#39;display&#39;, "none");
  }); 
  $(&#39;#m&#39;).click(()=> {
   $(&#39;.selectBox&#39;).css(&#39;display&#39;, "none");
  });

  // 用户点击发送表情
  $(&#39;.emoji li img&#39;).click((ev)=> {
    ev = ev || window.event;
    var src = ev.target.src;
    var emoji = src.replace(/\D*/g, &#39;&#39;).substr(6, 8); // 提取序号
    var old = $(&#39;#m&#39;).val(); // 用户输入的其他内容
    $(&#39;#m&#39;).val(old+&#39;[emoji&#39;+emoji+&#39;]&#39;);
    $(&#39;.selectBox&#39;).css(&#39;display&#39;, "none");
  });
登录后复制

客户端收到之后将表情序号还原为src,更改如下

// chat-client.js

  // 接收消息
  socket.on(&#39;receiveMsg&#39;, (obj)=> { 
   // 提取文字中的表情加以渲染
   var msg = obj.msg;
   var content = &#39;&#39;;
   while(msg.indexOf(&#39;[&#39;) > -1) { // 其实更建议用正则将[]中的内容提取出来
    var start = msg.indexOf(&#39;[&#39;);
    var end = msg.indexOf(&#39;]&#39;);

    content += &#39;<span>&#39;+msg.substr(0, start)+&#39;</span>&#39;;
    content += &#39;<img src="image/emoji/emoji%20(&#39;+msg.substr(start+6, end-start-6)+&#39;).png">&#39;;
    msg = msg.substr(end+1, msg.length);
   }
   content += &#39;<span>&#39;+msg+&#39;</span>&#39;;
   
   $(&#39;#messages&#39;).append(`
    <li class=&#39;${obj.side}&#39;>
     <img src="${obj.img}">
     <p>
      <span>${obj.name}</span>
      <p style="color: ${obj.color};">${content}</p>
     </p>
    </li>
   `);
   // 滚动条总是在最底部
   $(&#39;#messages&#39;).scrollTop($(&#39;#messages&#39;)[0].scrollHeight);
  });
登录后复制

可以成功发送表情了。

8、支持发送图片

首先是图片按钮样式,发送图片的按钮是type为file的input。这里有一个改变样式的小技巧,那就是将input的透明度设为0,z-index为5,将你想要得样式放在p中,z-index设为1覆盖在input上。

<input type="file" id="file">
<i class="fa fa-picture-o" id="img"></i>
css:

.edit #file {
  width: 32.36px;
  height: 29px;
  opacity: 0;
  z-index: 5;
}
.edit #img {
  z-index: 0;
  margin-left: -43px;
}
登录后复制

完美

接下来是点击按钮发送图片,我们用了fileReader对象,这里有一篇不错的文章讲解了fileReader,fileReader是一个对象,可以将我们选中的文件已64位输出然后将结果存放在reader.result中,我们选中图片之后,reader.result就存放的是图片的src

// chat-client.js

  // 用户发送图片
  $(&#39;#file&#39;).change(function() {
   var file = this.files[0]; // 上传单张图片
   var reader = new FileReader();

   //文件读取出错的时候触发
   reader.onerror = function(){
     console.log(&#39;读取文件失败,请重试!&#39;); 
   };
   // 读取成功后
   reader.onload = function() {
    var src = reader.result; // 读取结果
    var img = &#39;<img class="sendImg" src="&#39;+src+&#39;">&#39;;
    socket.emit(&#39;sendMsg&#39;, { // 发送
     msg: img,
     color: color,
     type: &#39;img&#39; // 发送类型为img
    }); 
   };
   reader.readAsDataURL(file); // 读取为64位
  });
登录后复制

由于发送的是图片,所以对页面布局难免有影响,为了页面美观客户端在接收其他用户发送的消息的时候会先判断发送的是文本还是图片,根据不同的结果展示不同布局。判断的方法是在客户发送消息的时候传入一个type,根据type的值来确实发送内容的类型。所以上面发送图片代码中触发了sendMsg事件,传入参数多了一个type属性。

响应的,我们应该在chat-client.js中修改receiveMsg事件监听函数,改为根据传入type做不同操作

chat-client.js
  // 接收消息
  socket.on(&#39;receiveMsg&#39;, (obj)=> { 
   // 发送为图片
   if(obj.type == &#39;img&#39;) {
    $(&#39;#messages&#39;).append(`
     <li class=&#39;${obj.side}&#39;>
      <img src="${obj.img}">
      <p>
       <span>${obj.name}</span>
       <p style="padding: 0;">${obj.msg}</p>
      </p>
     </li>
    `); 
    $(&#39;#messages&#39;).scrollTop($(&#39;#messages&#39;)[0].scrollHeight);
    return;
   }

   // 提取文字中的表情加以渲染
   // 下面不变
  });
登录后复制

现在我们可以发送图片了

上面是我整理给大家的,希望今后会对大家有帮助。

相关文章:

在vue中如何使用cli请求代理与项目打包方面的问题

在vue-cli中使用webpack模板解决项目搭建及打包路径问题

在vue中bus全局事件中心(详细教程)

以上是如何实现node+express个性化聊天室?的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.聊天命令以及如何使用它们
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

nvm 怎么删除node nvm 怎么删除node Dec 29, 2022 am 10:07 AM

nvm删除node的方法:1、下载“nvm-setup.zip”并将其安装在C盘;2、配置环境变量,并通过“nvm -v”命令查看版本号;3、使用“nvm install”命令安装node;4、通过“nvm uninstall”命令删除已安装的node即可。

node项目中如何使用express来处理文件的上传 node项目中如何使用express来处理文件的上传 Mar 28, 2023 pm 07:28 PM

怎么处理文件上传?下面本篇文章给大家介绍一下node项目中如何使用express来处理文件的上传,希望对大家有所帮助!

如何使用MySQL和Java实现一个简单的聊天室功能 如何使用MySQL和Java实现一个简单的聊天室功能 Sep 21, 2023 pm 05:13 PM

如何使用MySQL和Java实现一个简单的聊天室功能引言:在当今社交媒体的盛行下,人们越来越依赖于在线聊天来交流和分享信息。如何使用MySQL和Java实现一个简单的聊天室功能是一个非常有趣和实用的项目。本文将介绍如何使用MySQL和Java来实现这一功能,并提供具体的代码示例。一、搭建数据库首先,我们需要在MySQL中创建一个数据库来存储聊天室的相关信息。

深入浅析Node的进程管理工具“pm2” 深入浅析Node的进程管理工具“pm2” Apr 03, 2023 pm 06:02 PM

本篇文章给大家分享Node的进程管理工具“pm2”,聊聊为什么需要pm2、安装和使用pm2的方法,希望对大家有所帮助!

Pi Node教学:什么是Pi节点?如何安装和设定Pi Node? Pi Node教学:什么是Pi节点?如何安装和设定Pi Node? Mar 05, 2025 pm 05:57 PM

PiNetwork节点详解及安装指南本文将详细介绍PiNetwork生态系统中的关键角色——Pi节点,并提供安装和配置的完整步骤。Pi节点在PiNetwork区块链测试网推出后,成为众多先锋积极参与测试的重要环节,为即将到来的主网发布做准备。如果您还不了解PiNetwork,请参考Pi币是什么?上市价格多少?Pi用途、挖矿及安全性分析。什么是PiNetwork?PiNetwork项目始于2019年,拥有其专属加密货币Pi币。该项目旨在创建一个人人可参与

如何使用Go语言开发Websocket聊天室 如何使用Go语言开发Websocket聊天室 Dec 14, 2023 pm 01:46 PM

如何使用Go语言开发Websocket聊天室Websocket是一种实时通信协议,通过建立一次连接,可以在服务器和客户端之间进行双向通信。在开发聊天室时,Websocket是一个非常好的选择,因为它可以实现实时消息交流,并且能够提供高效的性能。本文将介绍如何使用Go语言开发一个简单的Websocket聊天室,并提供一些具体的代码示例。一、准备工作1.安装Go

基于JavaScript构建实时聊天室 基于JavaScript构建实时聊天室 Aug 10, 2023 pm 11:18 PM

基于JavaScript构建实时聊天室随着互联网的快速发展,人们越来越注重即时通讯和实时互动体验。而实时聊天室作为一种常见的即时通讯工具,对于个人和企业来说都非常重要。本文将介绍如何使用JavaScript构建一个简单的实时聊天室,并提供相应的代码示例。我们首先需要一个前端页面作为聊天室的UI界面。以下是一个简单的HTML结构示例:&lt;!DOCTYPE

聊聊用pkg将Node.js项目打包为可执行文件的方法 聊聊用pkg将Node.js项目打包为可执行文件的方法 Dec 02, 2022 pm 09:06 PM

​如何用pkg打包nodejs可执行文件?下面本篇文章给大家介绍一下使用pkg将Node项目打包为可执行文件的方法,希望对大家有所帮助!

See all articles