真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能

使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能?相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。

為措美等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計(jì)制作服務(wù),及措美網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作、措美網(wǎng)站設(shè)計(jì),以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會(huì)得到認(rèn)可,從而選擇與我們長(zhǎng)期合作。這樣,我們也可以走得更遠(yuǎn)!

npx create-react-app socketio-demo

進(jìn)入socketio-demo目錄 運(yùn)行eject進(jìn)行拆包,本項(xiàng)目也可以不拆,這是個(gè)人習(xí)慣。 注意如果運(yùn)行eject命令最好在項(xiàng)目初始階段執(zhí)行,已經(jīng)開始編寫后不要再使用容易出現(xiàn)bug,新人謹(jǐn)慎使用eject命令

yarn eject

項(xiàng)目拆包后創(chuàng)建服務(wù)器文件夾和文件

mkdir server
type null>index.js

創(chuàng)建完成后目錄如下

使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能 

編寫即時(shí)通訊(聊天室)后臺(tái)

安裝nodejs插件

npm i express http socket.io nodemon

進(jìn)入server文件夾下的index.js頁面開始編寫后臺(tái)程序

const app = require('express')(); 
const server = require('http').Server(app); 
const io = require('socket.io')(server); 
//設(shè)置端口9093 
server.listen(9093); 
//創(chuàng)建socket.io連接 
io.on('connection', function (socket) { 
 //獲取messages事件 
 socket.on('messages', function (data) { 
  //向所有連接進(jìn)行廣播 
 socket.broadcast.emit('messages', data) 
  //對(duì)發(fā)出者進(jìn)行廣播,用戶名加上我 
 data.user=data.user+'[我]' 
 socket.emit('messages', data) 
 }); 
});

編寫即時(shí)通訊(聊天室)前臺(tái)

后臺(tái)編寫完畢,可以在src目錄中編寫前臺(tái)內(nèi)容 安裝需要用到的react-router和redux依賴

npm i redux react-redux react-router react-router-dom

在src中創(chuàng)建io文件夾 在io文件夾中創(chuàng)建所需要的文件

cd src
mkdir io
cd io
type null>login.js
type null>socket-demo.js
type null>socket-demo.css
mkdir auth
cd auth
type null>auth.js

創(chuàng)建完成后目錄如下

使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能 

這里auth.js文件是用來判斷用戶是否輸入昵稱,如已輸入昵稱可以進(jìn)入聊天室,如沒有輸入昵稱則跳回登錄界面要求輸入昵稱

本項(xiàng)目當(dāng)中我們把昵稱存在redux里實(shí)現(xiàn)登錄界面和聊天室界面的共用,當(dāng)然現(xiàn)這個(gè)項(xiàng)目比較小,如果想用localStorage存在本地也可以,不過考慮到后期的擴(kuò)展性以及加深對(duì)redux的理解我還是選擇存在redux當(dāng)中

src文件夾下創(chuàng)建redux.js文件

src文件夾下創(chuàng)建redux文件夾,在redux文件夾下創(chuàng)建user.redux.js文件

cd src
type null>redux.js
mkdir redux
cd redux
type null>user.redux.js

新建目錄如下

使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能 

在redux文件夾下的user.redux.js中創(chuàng)建存儲(chǔ)用戶昵稱的reducer

const SET_USERNAME='SET_USERNAME' 
//初始化倉庫 
const initState={user:''} 
//根據(jù)動(dòng)作改變倉庫  
export function User(state = initState, action) { 
 switch (action.type) { 
  case SET_USERNAME: 
   return {...state,user:action.payload} 
  default: 
   return state 
 } 
} 
//寫入昵稱動(dòng)作 
export function setUserName(user) { 
 return { 
  type:SET_USERNAME, 
 payload:user 
 } 
}

在src/redux.js文件中創(chuàng)建倉庫 combineReducers用于多個(gè)reducer的合并,這個(gè)項(xiàng)目中也可以不加,單為了后期擴(kuò)展加入使用

import { combineReducers, createStore } from 'redux' 
import {User} from './redux/user.redux' 
//window.__REDUX_DEVTOOLS_EXTENSION__ && window.__REDUX_DEVTOOLS_EXTENSION__() 用于chrome redux的擴(kuò)展項(xiàng)
let reducer = combineReducers({ User }) 
let store = createStore( 
 reducer,window.__REDUX_DEVTOOLS_EXTENSION__ && window.__REDUX_DEVTOOLS_EXTENSION__()) 
 
export default store

這樣就可以在頁面當(dāng)中使用redux了

下一步在app.js中引入redux,并把路由搭建起來 在src/app.js中寫入

import React from 'react';
import {HashRouter as Router,Route,Switch} from 'react-router-dom'
import Login from "./io/login";
import SocketDemo from "./io/socket-demo";
import {Provider} from 'react-redux'
import store from './redux'
import Auth from "./io/auth/auth";
function App() {
 return (
  
   
    
    
     
     
    
   
  
 );
}

export default App;

在寫頁面之前我們先安裝修飾符插件

npm i babel-plugin-transform-decorators-legacy

Babel >= 7.x 時(shí)安裝 @babel/plugin-proposal-decorators

npm i @babel/plugin-proposal-decorators

在package.json中babel項(xiàng)中配置,注意plugins放在presets前否則容易報(bào)錯(cuò)

"babel": { 
 "plugins": [ 
  ["@babel/plugin-proposal-decorators", { "legacy": true }] 
 ], 
 "presets": [ 
  "react-app" 
 ] 
}

好了這樣就可以使用裝飾付了

下面我們來編寫判斷是否設(shè)置用戶名的程序 打開src/io/auth下的auth.js文件

import React from 'react'; 
import {connect} from 'react-redux' 
import {withRouter} from 'react-router-dom' 
//獲取reducer 
@connect( 
 state=>state, 
 {} 
) 
//獲取router 
@withRouter 
class Auth extends React.Component{ 
 componentDidMount() { 
  //如果有用戶名就跳到聊天頁,如沒有則跳到登陸頁。 
 if(this.props.User.user){ 
   this.props.history.push('/talk') 
  }else { 
   this.props.history.push('/') 
  } 
 } 
 render() { 
  return null 
 } 
} 
 
export default Auth

編寫輸入昵稱并跳轉(zhuǎn)步驟 打開src/io/login.js文件

import React from 'react';
import './socket-demo.css';
import {connect} from 'react-redux'
import {setUserName} from '../redux/user.redux'
@connect(
 null,
 {setUserName}
)
class Login extends React.Component{
 constructor(props) {
  super(props);
  this.state={
   user:''
  }
  this.login=this.login.bind(this)
  this.onKeyDown=this.onKeyDown.bind(this)
 }
 //鍵盤點(diǎn)擊跳轉(zhuǎn) 
 onKeyDown(e){
  switch (e.keyCode) {
   case 13:
    this.login();
    return;
   default:
    return;
  }
 }
 //添加鍵盤事件 
 componentDidMount() {
  document.addEventListener("keydown", this.onKeyDown)
 }
 //賦值state 
 handleChange(title,target){
  this.setState({
   [title]:target.target.value
  })
 }
 //賦值并跳轉(zhuǎn)到聊天室頁面 
 login(){
  let {user}=this.state;
  if(user!==null && user.trim()!==''){
   this.props.setUserName(user);
   this.props.history.push('/talk')
  }
 }
 render() {
  return (
   
    this.handleChange('user',v)} />
    進(jìn)入聊天室
   
 );  } } export default Login

下面是重頭戲,聊天室的前端展示的核心代碼 打開src/iosocket-demo.js文件

import React from 'react'
import io from 'socket.io-client'
import {connect} from 'react-redux'
import './socket-demo.css'
const url='ws://localhost:9093'
const socket = io(url);
@connect(
 state=>state,
 {}
)
class SocketDemo extends React.Component{
 constructor(props) {
  super(props);
  this.state={
   message:'',
   user:this.props.User.user,
   messages:[]
  }
  this.send=this.send.bind(this)
  this.login=this.login.bind(this)
  this.onKeyDown=this.onKeyDown.bind(this)
 }
 componentDidMount() {
  //輸入歡迎信息 
  this.login()
  //增加回車事件 
  document.addEventListener("keydown", this.onKeyDown)
  //socket.io連接后臺(tái) 
  io(url).on('connect', ()=>{
   console.log('connect');
   socket.on('messages', data => {
    //返回用戶列表 
    this.setState({
     messages:[...this.state.messages,data]
    })
    if(this.refs.showDiv){
     this.refs.showDiv.scrollTop=2000
    }
   });
  });
 }
 componentWillUnmount() {
  //斷開socket io連接 
  io('ws://localhost:9093').on('disconnect', function(){
   console.log('disconntect');
  });
  document.removeEventListener("keydown", this.onKeyDown)
 }
 //鼠標(biāo)回車事件 
 onKeyDown(e){
  switch (e.keyCode) {
   case 13:
    this.send();
    return; default:
    return;
  }
 }
 //向后臺(tái)發(fā)送信息 
 send(){
  let {user,message}=this.state;
  console.log(this.refs.showDiv);
  socket.emit('messages', {user,message});
  this.setState({
   message:''
  })
 }
 login(){
  let user=this.props.User.user;
  const obj={user:'作者',message:`歡迎${user}來到聊天室`}
  if(user.trim()!==''){
   this.setState({
    user:user,
    messages:[obj]
   })
  }
 }
 //賦值state 
 handleChange(title,target){
  this.setState({
   [title]:target.target.value
  })
 }
 render() {
  let cn='showInfo'
  return (
   
                 this.handleChange('message',v)}           value={this.state.message}       />       發(fā)送鏈接      
       {       this.state.messages.map((v,index)=>{        if(index===0){         cn='titleInfo'        }else{         cn='showInfo'        }        return (                   {v.user}:          {v.message}         
 )       })      }     
         );  } } export default SocketDemo;

最后加上src/iosocket-demo.css

body{ 
  background: #008DB7; 
 font-family: 'Microsoft YaHei UI'; 
 
} 
.loginDiv{ 
  text-align: center; 
 margin: 150px auto 0; 
 width: 250px; 
} 
.loginDiv input[type='text']{ 
  display: inline-block; 
 box-sizing: border-box; 
 border-radius: 5px; 
 padding-left: 5px; 
 border: none; 
 width: 250px; 
 height: 35px; 
 line-height: 35px; 
} 
.loginDiv button{ 
  display: inline-block; 
 box-sizing: border-box; 
 border-radius: 5px; 
 padding-left: 5px; 
 border: none; 
 width: 250px; 
 height: 35px; 
 line-height: 35px; 
 margin-top: 10px; 
 background: #0067A2; 
 color: #ffffff; 
} 
 
.talkDiv{ 
  position: fixed; 
 top: 0; 
 left: 0; 
 right: 0; 
 bottom: 0; 
} 
 
.talkDiv .operatingDiv{ 
  position: fixed; 
 bottom: 0; 
 left: 0; 
 right: 0; 
 height: 40px; 
 display: flex; 
} 
 
.talkDiv .operatingDiv input[type='text']{ 
  flex: 1; 
 height: 40px; 
 line-height: 40px; 
 box-sizing: border-box; 
 padding-left: 10px; 
} 
.talkDiv .operatingDiv button{ 
  display: inline-block; 
 box-sizing: border-box; 
 border-radius: 5px; 
 border: none; 
 width: 250px; 
 height: 40px; 
 line-height: 40px; 
 background: #0067A2; 
 color: #ffffff; 
} 
 
.talkDiv .showDiv{ 
  position: fixed; 
 bottom: 40px; 
 left: 0; 
 right: 0; 
 top: 0; 
 font-size: 16px; 
 color: #ffffff; 
 overflow: auto; 
} 
.talkDiv .showDiv .titleInfo{ 
  padding: 10px; 
 color: yellow; 
 font-size: 20px; 
} 
.talkDiv .showDiv .showInfo{ 
  padding: 10px; 
}

在package.json中加入命令行

"scripts": { 
 "start": "node scripts/start.js", 
 "build": "node scripts/build.js", 
 "server": "nodemon server/index.js"
},

啟動(dòng)程序

看完上述內(nèi)容,你們掌握使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!


網(wǎng)站欄目:使用nodejs和react怎么實(shí)現(xiàn)一個(gè)即時(shí)通訊聊天室功能
當(dāng)前URL:http://weahome.cn/article/jpdoeh.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部