Maison interface Web js tutoriel AngularJS Node.js implémente un salon de discussion en ligne_AngularJS

AngularJS Node.js implémente un salon de discussion en ligne_AngularJS

May 16, 2016 pm 03:41 PM
angularjs node.js 在线聊天室

Je dois dire que démarrer avec AngularJS est beaucoup plus difficile que je ne le pensais. Après avoir lu les exemples PhoneCat fournis sur le site officiel, je suis allé au MOOC et j'ai lu la Série pratique AngularJS de Da Mo Qiongqiu. Il y a encore beaucoup de questions floues sur l'utilisation de base, j'ai donc décidé de créer un salon de discussion en ligne pour aider à comprendre. DEMO peut être cliqué →salle de discussion, et le code peut être cliqué →ChatRoom-AngularJS.

Vous pouvez tamponner l'image claire http://files.jb51.net/file_images/article/201508/201508281040051.gif

Fonction

Avant de commencer le développement, clarifiez d'abord les fonctions qui doivent être implémentées :

Un nouvel utilisateur se connecte et diffuse pour informer les autres utilisateurs
L'utilisateur se déconnecte et diffuse pour avertir les autres utilisateurs
Peut afficher le nombre de personnes en ligne et la liste
Chat de groupe et messagerie privée disponibles
Si un utilisateur envoie un message de groupe, la diffusion en informera tous les autres utilisateurs
Si l'utilisateur envoie un message privé, l'interface du destinataire sera notifiée séparément

Parce que je suis un salaud esthétique, je me suis entièrement appuyé sur le bootstrap. De plus, j'ai également imité le design des bulles dans l'historique du chat WeChat.

L'interface est divisée en deux sections : gauche et droite, qui servent respectivement à afficher les listes en ligne et le contenu du chat.

Dans la liste en ligne à gauche, cliquez sur différents éléments pour changer de partenaire de discussion dans la section de droite.

Les enregistrements de conversation avec le partenaire de discussion actuel sont affichés à droite, mais seuls les 30 plus récents sont affichés. Le contenu de chaque enregistrement de discussion comprend le surnom et l'avatar de l'expéditeur, l'heure d'envoi et le contenu du message. Concernant l'avatar, nous allons faire ici un procédé simple et le remplacer par des carrés remplis de couleurs aléatoires. De plus, les styles des messages que vous envoyez et des messages que vous recevez doivent naturellement être conçus différemment. Tous les effets sont visibles dans l'image ci-dessous.

Vous pouvez tamponner l'image claire http://files.jb51.net/file_images/article/201508/201508281040052.png

Serveur

Nous utilisons Node.js et mélangeons express et socket.io pour développer le serveur. Ouvrez le terminal dans le répertoire racine du programme et exécutez :

.

Copier le code Le code est le suivant :
npm init

Suivez les invites et générez un fichier package.json. Ouvrir et configurer les dépendances :

 "dependencies": {
  "express": "^4.13.3",
  "socket.io": "^1.3.6"
 }
Copier après la connexion

Ensuite, exécutez npm install pour installer les modules dépendants.

Ensuite, nous créons un nouveau app.js dans le répertoire racine et y écrivons le code côté serveur. Créez un nouveau dossier public pour stocker le code client.

Le contenu principal de app.js est le suivant :

var express = require('express');
var app = require('express')();
var http = require('http').createServer(app);
var io = require('socket.io')(http);

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


app.get('/', function (req, res) {
  res.sendfile('index.html');
});


io.on('connection',function(socket){
  socket.on('addUser',function(data){ //有新用户进入聊天室
  });

  socket.on('addMessage',function(data){ //有用户发送新消息
  });
  
  socket.on('disconnect', function () { //有用户退出聊天室
  );
});

http.listen(3002, function () {
  console.log('listening on *:3002');
});
Copier après la connexion

Dans le code ci-dessus, nous avons ajouté des auditeurs pour les événements suivants :

-addUser, un nouvel utilisateur entre dans le salon de discussion

Cet événement est déclenché par la saisie d'un pseudo par le client. Après l'avoir reçu, le serveur jugera si le pseudo existe déjà, avertira le client que le pseudo n'est pas valide :

Copier le code Le code est le suivant :
socket.emit('userAddingResult',{result:false});

Au contraire, avertir le client que le pseudo est valide ainsi que toutes les informations de l'utilisateur actuellement connecté, et diffuser les informations du nouvel utilisateur aux autres utilisateurs connectés :

socket.emit('userAddingResult',{result:true});
allUsers.push(data);//allUsers保存了所有用户
socket.emit('allUser',allUsers);//将所有在线用户发给新用户
socket.broadcast.emit('userAdded',data);//广播欢迎新用户,除新用户外都可看到
Copier après la connexion

Vous devez faire attention à la différence entre 'socket.emit' et 'socket.broadcast.emit'. Vous pouvez consulter cet article de blog Explication de plusieurs utilisations de socket.io émet : <🎜. >

// send to current request socket client
socket.emit('message', "this is a test");
// sending to all clients except sender
socket.broadcast.emit('message', "this is a test");

Copier après la connexion

-addMessage, un utilisateur envoie un nouveau message

Dans ce suivi événementiel, deux types de situations sont à gérer :

1.Message privé

Si le message est envoyé à un utilisateur spécifique A, vous devez alors obtenir l'instance de socket correspondant à A, puis appeler sa méthode d'émission. Ainsi, chaque fois qu'un client se connecte au serveur, nous devons sauvegarder son instance de socket pour les besoins ultérieurs.

Copier le code Le code est le suivant :
connectedSockets[nickname]=socket;//以昵称作下标,保存每个socket实例,发私信需要用

需要发私信时,取出socket实例做操作即可:

复制代码 代码如下:
connectedSockets[nickname].emit('messageAdded',data)

2.群发
群发就比较简单了,用broadcast方法即可:

复制代码 代码如下:
socket.broadcast.emit('messageAdded',data);//广播消息,除原发送者外都可看到

-disconnect,有用户退出聊天室
需要做三件事情:

1.通知其他用户“某用户下线”

复制代码 代码如下:
socket.broadcast.emit('userRemoved', data);

2.将用户从保存了所有用户的数组中移除

3.将其socket实例从保存了所有客户端socket实例的数组中移除

复制代码 代码如下:
delete connectedSockets[nickname]; //删除对应的socket实例

运行一下服务端代码,观察有无错误:

复制代码 代码如下:
node app.js

若没什么问题,继续编写客户端的代码。

客户端

在public目录下新建'index.html',客户端需要用到bootstrap、angularjs、socket.io、jQuery以及我们自己的js和css文件,先把这些文件用标签引入。

<!DOCTYPE html>
<html>
<head lang="en">
  <meta charset="UTF-8">
  <title></title>
  <link href="http://cdn.bootcss.com/bootstrap/3.3.5/css/bootstrap.min.css" rel="stylesheet">
  <link rel="stylesheet" href="./assets/style/app.css"/>
  <script src="http://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
  <script src="/socket.io/socket.io.js"></script>
  <script src="//cdn.bootcss.com/angular.js/1.4.3/angular.min.js"></script>
  <script src="./assets/js/app.js"></script>
</head>
<body></body>
</html>
Copier après la connexion

我们并不立即深入逻辑细节,把框架搭好先。
首先,在body上加上ng-app属性,标记一下angularjs的“管辖范围”。这个练习中我们只用到了一个控制器,同样将ng-controller属性加到body标签。

复制代码 代码如下:
<body ng-app="chatRoom" ng-controller="chatCtrl">

接下来在js中,我们来创建module及controller。

var app=angular.module("chatRoom",[]);
app.controller("chatCtrl",['$scope','socket','randomColor',function($scope,socket,randomColor){}]);
Copier après la connexion

注意这里,我们用内联注入添加了socket和randomColor服务依赖。这里我们不用推断式注入,以防部署的时候用uglify或其他工具进行了混淆,变量经过了重命名导致注入失效。
在这个练习中,我们自定义了两个服务,socket和randomColor,前者是对socket.io的包装,让其事件进入angular context,后者是个可以生成随机色的服务,用来给头像指定颜色。

//socket服务
app.factory('socket', function($rootScope) {
  var socket = io(); //默认连接部署网站的服务器
  return {
    on: function(eventName, callback) {...},
    emit: function(eventName, data, callback) {...}
  };
});

//randomcolor服务
app.factory('randomColor', function($rootScope) {
  return {
    newColor: function() {
      return '#'+('00000'+(Math.random()*0x1000000<<0).toString(16)).slice(-6);//返回一个随机色
    }
  };
});
Copier après la connexion

注意socket服务中连接的语句“var socket = io();”,我们并没有传入任何url,是因为其默认连接部署这个网站的服务器。

考虑到聊天记录以及在线人员列表都是一个个逻辑及结构重复的条目,且html结构较复杂,为了其复用性,我们把它们封装成两个指令:

app.directive('message', ['$timeout',function($timeout) {}])
  .directive('user', ['$timeout',function($timeout) {}]);
Copier après la connexion

注意这里两个指令都注入了'$timeout'依赖,其作用后文会解释。

这样一个外层框架就搭好了,现在我们来完成内部的细节。

登录

页面刚加载时只显示登录界面,只有当输入昵称提交后且收到服务端通知昵称有效方可跳转到聊天室。我们将ng-show指令添加到登录界面和聊天室各自的dom节点上,来帮助我们显示或隐藏元素。用'hasLogined'的值控制是显示或隐藏。

<!-- chat room -->
<div class="chat-room-wrapper" ng-show="hasLogined">
...
</div>
<!-- end of chat room -->

<!-- login form -->
<div class="userform-wrapper" ng-show="!hasLogined">
...
</div>
<!-- end of login form -->
Copier après la connexion

JS部分

 $scope.login = function() { //登录
   socket.emit("addUser", {...});
 }

 //收到登录结果
 socket.on('userAddingResult', function(data) {
   if (data.result) {
     $scope.hasLogined = true;
   } else { //昵称被占用
     $scope.hasLogined = false;
   }
 });
Copier après la connexion

这里监听了socket连接上的'userAddingResult'事件,接收服务端的通知,确认是否登录成功。

socket连接监听

成功登录以后,我们还监听socket连接上的其他事件:

复制代码 代码如下:

//接收到欢迎新用户消息,显示系统欢迎辞,刷新在线列表<br /> socket.on('userAdded', function(data) {});<br /> //接收到所有用户信息,初始化在线列表<br /> socket.on('allUser', function(data) {});<br /> //接收到用户退出消息,刷新在线列表<br /> socket.on('userRemoved', function(data) {});<br /> //接收到新消息,添加到聊天记录<br /> socket.on('messageAdded', function(data) {});<br />

接收到事件以后,做相应的刷新动作,这里的socket是socket.io经过包装的服务,内部仅包装了我们需要用到的两个函数on和emit。我们在事件监听里对model做的修改,都会在AngularJS内部得到通知和处理,UI才会得到及时刷新。
监听内做的事情太具体和琐碎了,这里就不列出了,接下来介绍一下message指令。

message 指令

最后分享一下我在写message指令时遇到的问题。首先看一下其代码:

app.directive('message', ['$timeout',function($timeout) {
  return {
    restrict: 'E',
    templateUrl: 'message.html',
    scope:{
      info:"=",
      self:"=",
      scrolltothis:"&"
    },
    link:function(scope, elem, attrs){
        $timeout(scope.scrolltothis);
    }
  };
}])
Copier après la connexion

以及其模板message.html:

<div ng-switch on="info.type">
  <!-- 欢迎消息 -->
  <div class="system-notification" ng-switch-when="welcome">系统{{info.text}}来啦,大家不要放过他~</div>
  <!-- 退出消息 -->
  <div class="system-notification" ng-switch-when="bye">系统:byebye,{{info.text}}</div>
  <!-- 普通消息 -->
  <div class="normal-message" ng-switch-when="normal" ng-class="{others:self!==info.from,self:self===info.from}">
    <div class="name-wrapper">{{info.from}} @ {{time | date: 'HH:mm:ss' }}</div>
    <div class="content-wrapper">{{info.text}}<span class="avatar"></span></div>
  </div>
</div>
Copier après la connexion

模板中我们用ng-switch指令监听info.type变量的值,根据其值的不同显示不同内容。比如,当info.type值为"welcome"时,创建第一个dom节点,删除下方另外两个div。
另外,普通消息下,为了在UI上区分自己发出去的和收到的消息,需要给他们应用不同的样式,这里用ng-class指令实现。

复制代码 代码如下:
ng-class="{others:self!==info.from,self:self===info.from}"

当'self===info.from'返回true时,应用'self'类,否则,应用'others'类。
在此指令中,我们创建了独立作用域,并绑定了三个属性,绑定完后还必须在父作用域的HTML标签上添加相应属性。

scope:{
    info:"=",
    self:"=",
    scrolltothis:"&"
}

<message self="nickname" scrolltothis="scrollToBottom()" info="message" ng-repeat="message in messages"></message>
Copier après la connexion

在link函数中,执行一个动作:每当一个message被加到页面上时,将聊天记录滚动到最下方,一开始我是这样写的:

复制代码 代码如下:
link:function(scope, elem, attrs){ scope.scrolltothis();}

结果发生了一个很奇怪的现象,总是滚动到上一条位置,而不是最新这条。调试之后发现是因为'scrolltothis'函数执行的时候,DOM还没渲染,所以在函数内部获取scrollHeight的时候获得的总是添加DOM节点之前的状态。这时候,可以把代码放到$timeout里延迟0秒执行,延迟0秒并不意味着会立即执行,因为js的单线程特性,代码实际会等到dom渲染完再执行。

复制代码 代码如下:
$timeout(scope.scrolltothis);

完整代码可以戳我的GitHub→ChatRoom-AngularJS,DEMO可以戳→chat room

有任何不妥之处或错误欢迎各位指出,不胜感激~

Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

Video Face Swap

Video Face Swap

Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Outils chauds

Bloc-notes++7.3.1

Bloc-notes++7.3.1

Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise

SublimeText3 version chinoise

Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1

Envoyer Studio 13.0.1

Puissant environnement de développement intégré PHP

Dreamweaver CS6

Dreamweaver CS6

Outils de développement Web visuel

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Sujets chauds

Tutoriel Java
1664
14
Tutoriel PHP
1268
29
Tutoriel C#
1244
24
Explication graphique détaillée de la mémoire et du GC du moteur Node V8 Explication graphique détaillée de la mémoire et du GC du moteur Node V8 Mar 29, 2023 pm 06:02 PM

Cet article vous donnera une compréhension approfondie de la mémoire et du garbage collector (GC) du moteur NodeJS V8. J'espère qu'il vous sera utile !

Un article pour parler du contrôle de la mémoire dans Node Un article pour parler du contrôle de la mémoire dans Node Apr 26, 2023 pm 05:37 PM

Le service Node construit sur une base non bloquante et piloté par les événements présente l'avantage d'une faible consommation de mémoire et est très adapté à la gestion de requêtes réseau massives. Dans le contexte de demandes massives, les questions liées au « contrôle de la mémoire » doivent être prises en compte. 1. Le mécanisme de récupération de place du V8 et les limitations de mémoire Js sont contrôlés par la machine de récupération de place

Parlons de la façon de choisir la meilleure image Docker Node.js ? Parlons de la façon de choisir la meilleure image Docker Node.js ? Dec 13, 2022 pm 08:00 PM

Le choix d'une image Docker pour Node peut sembler trivial, mais la taille et les vulnérabilités potentielles de l'image peuvent avoir un impact significatif sur votre processus CI/CD et votre sécurité. Alors, comment choisir la meilleure image Docker Node.js ?

Comment mettre en œuvre une salle de discussion en ligne en utilisant le langage Go et Redis Comment mettre en œuvre une salle de discussion en ligne en utilisant le langage Go et Redis Oct 27, 2023 pm 03:28 PM

Comment mettre en œuvre un salon de discussion en ligne en utilisant le langage Go et Redis Introduction : Avec le développement rapide d'Internet, les réseaux sociaux sont devenus un élément indispensable de la vie quotidienne des gens. En tant qu'élément important des réseaux sociaux, les forums de discussion en ligne sont populaires parmi les gens pour leur commodité, leur temps réel et leur forte interactivité. Cet article est basé sur le langage Go et Redis et présente comment utiliser ces deux outils pour mettre en œuvre un salon de discussion en ligne simple. 1. Introduction au langage Go : le langage Go est un langage de programmation système open source pour les systèmes d'exploitation modernes.

Parlons en profondeur du module File dans Node Parlons en profondeur du module File dans Node Apr 24, 2023 pm 05:49 PM

Le module de fichiers est une encapsulation des opérations de fichiers sous-jacentes, telles que l'ajout de lecture/écriture/ouverture/fermeture/suppression de fichiers, etc. La plus grande caractéristique du module de fichiers est que toutes les méthodes fournissent deux versions de **synchrone** et ** asynchrone**, with Les méthodes avec le suffixe sync sont toutes des méthodes de synchronisation, et celles qui n'en ont pas sont toutes des méthodes hétérogènes.

Parlons de la boucle d'événements dans Node Parlons de la boucle d'événements dans Node Apr 11, 2023 pm 07:08 PM

La boucle d'événements est un élément fondamental de Node.js et permet une programmation asynchrone en garantissant que le thread principal n'est pas bloqué. Comprendre la boucle d'événements est crucial pour créer des applications efficaces. L'article suivant vous donnera une compréhension approfondie de la boucle d'événements dans Node. J'espère qu'il vous sera utile !

Parlons de la façon d'utiliser pkg pour empaqueter des projets Node.js dans des fichiers exécutables. Parlons de la façon d'utiliser pkg pour empaqueter des projets Node.js dans des fichiers exécutables. Dec 02, 2022 pm 09:06 PM

Comment empaqueter le fichier exécutable nodejs avec pkg ? L'article suivant vous expliquera comment utiliser pkg pour empaqueter un projet Node dans un fichier exécutable. J'espère qu'il vous sera utile !

Que dois-je faire si le nœud ne peut pas utiliser la commande npm ? Que dois-je faire si le nœud ne peut pas utiliser la commande npm ? Feb 08, 2023 am 10:09 AM

La raison pour laquelle le nœud ne peut pas utiliser la commande npm est que les variables d'environnement ne sont pas configurées correctement. La solution est la suivante : 1. Ouvrez "Propriétés système" ; 2. Recherchez "Variables d'environnement" -> "Variables système", puis modifiez l'environnement. variables ; 3. Recherchez l'emplacement du dossier nodejs ; 4. Cliquez sur "OK".

See all articles