項目背景:小程序中實現(xiàn)實時聊天功能
創(chuàng)新互聯(lián)專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務,包含不限于成都做網(wǎng)站、成都網(wǎng)站設計、成都外貿(mào)網(wǎng)站建設、城關網(wǎng)絡推廣、成都微信小程序、城關網(wǎng)絡營銷、城關企業(yè)策劃、城關品牌公關、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運營等,從售前售中售后,我們都將竭誠為您服務,您的肯定,是我們最大的嘉獎;創(chuàng)新互聯(lián)為所有大學生創(chuàng)業(yè)者提供城關建站搭建服務,24小時服務熱線:13518219792,官方網(wǎng)址:www.cdcxhl.com
一、服務器域名配置
配置流程
配置參考URL:https://developers.weixin.qq.com/miniprogram/dev/api/api-network.html
二、nginx中配置反向代理加密websocket(wss)
upstream websocket{ hash $remote_addr consistent; server 127.0.0.1:9090 weight=5 max_fails=3 fail_timeout=30s; } server { listen 80; server_name www.xxxx.cn; rewrite ^(.*)$ https://$host$1 permanent; } server { listen 443; server_name www.xxxx.cn; ssl on; root /home/wwwroot/yzcp; index index.php index.html index.htm; ssl_certificate /usr/local/nginx/conf/cert/1526060965511.pem;#這里是服務端證書路徑 ssl_certificate_key /usr/local/nginx/conf/cert/1526060965511.key;#這里是密鑰路徑 ssl_session_timeout 5m; ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:!aNULL:!MD5:!ADH:!RC4; ssl_protocols TLSv1 TLSv1.1 TLSv1.2; ssl_prefer_server_ciphers on; ssl_verify_client off; #隱藏index.php location / { #index index.php; deny 127.0.0.1; if (!-e $request_filename) { #一級目錄 rewrite ^(.*)$ /index.php?s=$1 last; break; } #wss配置 client_max_body_size 100m; proxy_redirect off; proxy_pass http://websocket;#反向代理轉發(fā)地址 proxy_set_header Host $host;# http請求的主機域名 proxy_set_header X-Real-IP $remote_addr;# 遠程真實IP地址 proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;#反向代理之后轉發(fā)之前的IP地址 proxy_read_timeout 604800s;#websocket心跳時間,默認是60s proxy_http_version 1.1; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection "Upgrade"; } location ~ .+\.php { fastcgi_pass unix:/tmp/php-cgi.sock; fastcgi_index index.php; include fastcgi_params; set $path_info ""; set $real_script_name $fastcgi_script_name; if ($fastcgi_script_name ~ "^(.+?\.php)(/.+)$") { set $real_script_name $1; set $path_info $2; } fastcgi_param SCRIPT_FILENAME $document_root$real_script_name; fastcgi_param SCRIPT_NAME $real_script_name; fastcgi_param PATH_INFO $path_info; } #防盜鏈開始 location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$ { expires 30d; } location ~ .*\.(js|css)?$ { expires 12h; } access_log /home/wwwlogs/www1537ucn.log; }
三、安裝swoole
編譯安裝:
wget http://pecl.php.net/get/swoole-1.9.3.tgz //下載swoole tar -zvxf swoole-1.9.3.tgz //解壓swoole cd swoole-1.9.3/; //進入swoole /usr/local/php54/bin/phpize; //生成configure ./configure --with-php-config=/usr/local/php/bin/php-config make && make install //安裝 cd /phpstudy/server/php/lib/php/extensions/no-debug-non-zts-20121212 //查看是否安轉上了swoole.so (注意:此文件下邊都是你安裝的拓展) vim /phpstudy/server/php/etc/php.ini //在php.ini添加extension=swoole.so加入到文件最后一行 lnmp restart; //重啟nginx php -m; //查看phpinfo,這時候swoole拓展已經(jīng)裝上了
四、服務器端運行程序
1、創(chuàng)建server.php放到項目的根目錄即可
<?php //實例化一個swoole的websocket服務監(jiān)聽本機的9501端口 $server = new swoole_websocket_server("服務器IP", 9090); //只需要綁定要監(jiān)聽的ip和端口。如果ip指定為127.0.0.1,則表示客戶端只能位于本機才能連接,其他計算機無法連接。 //端口這里指定為9090,可以通過netstat查看下該端口是否被占用。如果該端口被占用,可更改為其他端口,如9502,9503等。 $server->on('open', function (swoole_websocket_server $server, $request) { echo "你好連接成功{$request->fd}\n"; }); $server->on('message', function (swoole_websocket_server $server, $frame) { foreach($server->connections as $key => $fd) { $user_message = $frame->data; $server->push($fd, $user_message); } }); $server->on('close', function ($ser, $fd) { echo "client {$fd} closed\n"; }); $server->start(); ?>
2、由于swoole_server只能運行在CLI模式下,所以不要試圖通過瀏覽器進行訪問,這樣是無效的,我們在命令行下面執(zhí)行,注意一定要找到php的絕對路徑php server.php (這行代碼的意思是,把程序在服務器跑起來)
注意:php server.php
命令運行后,下面的黑框關閉后將無法聊天。所以一般使用命令:nohup php server.php &
五、客戶端
1、網(wǎng)頁代碼
聊天 內容: 收信人:
2、小程序端的代碼
Uitls/websocket.js:
var url = 'wss://www.xxx.cn';//服務器地址 function connect(user, func) { wx.connectSocket({ url: url, header: { 'content-type': 'application/json' }, success: function () { console.log('websocket連接成功~') }, fail: function () { console.log('websocket連接失敗~') } }) wx.onSocketOpen(function (res) { wx.showToast({ title: 'websocket已開通~', icon: "success", duration: 2000 }) //接受服務器消息 wx.onSocketMessage(func);//func回調可以拿到服務器返回的數(shù)據(jù) }); wx.onSocketError(function (res) { wx.showToast({ title: 'websocket連接失敗,請檢查!', icon: "none", duration: 2000 }) }) } //發(fā)送消息 function send(msg) { wx.sendSocketMessage({ data: msg }); } module.exports = { connect: connect, send: send }
JS:
// pages/socks/socks.js const app = getApp() var websocket = require('../../utils/websocket.js'); var utils = require('../../utils/util.js'); Page({ /** * 頁面的初始數(shù)據(jù) */ data: { newslist: [], userInfo: {}, scrollTop: 0, increase: false,//圖片添加區(qū)域隱藏 aniStyle: true,//動畫效果 message: "", previewImgList: [] }, /** * 生命周期函數(shù)--監(jiān)聽頁面加載 */ onLoad: function () { var that = this if (app.globalData.userInfo) { this.setData({ userInfo: app.globalData.userInfo }) } //調通接口 websocket.connect(this.data.userInfo, function (res) { // console.log(JSON.parse(res.data)) var list = [] list = that.data.newslist list.push(JSON.parse(res.data)) that.setData({ newslist: list }) }) }, // 頁面卸載 onUnload() { wx.closeSocket(); wx.showToast({ title: '連接已斷開~', icon: "none", duration: 2000 }) }, //事件處理函數(shù) send: function () { var flag = this if (this.data.message.trim() == "") { wx.showToast({ title: '消息不能為空哦~', icon: "none", duration: 2000 }) } else { setTimeout(function () { flag.setData({ increase: false }) }, 500) websocket.send('{ "content": "' + this.data.message + '", "date": "' + utils.formatTime(new Date()) + '","type":"text", "nickName": "' + this.data.userInfo.nickName + '", "avatarUrl": "' + this.data.userInfo.avatarUrl + '" }') this.bottom() } }, //監(jiān)聽input值的改變 bindChange(res) { this.setData({ message: res.detail.value }) }, cleanInput() { //button會自動清空,所以不能再次清空而是應該給他設置目前的input值 this.setData({ message: this.data.message }) }, increase() { this.setData({ increase: true, aniStyle: true }) }, //點擊空白隱藏message下選框 outbtn() { this.setData({ increase: false, aniStyle: true }) }, //發(fā)送圖片 chooseImage() { var that = this wx.chooseImage({ count: 1, // 默認9 sizeType: ['original', 'compressed'], // 可以指定是原圖還是壓縮圖,默認二者都有 sourceType: ['album', 'camera'], // 可以指定來源是相冊還是相機,默認二者都有 success: function (res) { // 返回選定照片的本地文件路徑列表,tempFilePath可以作為img標簽的src屬性顯示圖片 var tempFilePaths = res.tempFilePaths // console.log(tempFilePaths) wx.uploadFile({ url: 'wss://www.xxx.cn', //服務器地址 filePath: tempFilePaths[0], name: 'file', headers: { 'Content-Type': 'form-data' }, success: function (res) { if (res.data) { that.setData({ increase: false }) websocket.send('{"images":"' + res.data + '","date":"' + utils.formatTime(new Date()) + '","type":"image","nickName":"' + that.data.userInfo.nickName + '","avatarUrl":"' + that.data.userInfo.avatarUrl + '"}') that.bottom() } } }) } }) }, //圖片預覽 previewImg(e) { var that = this //必須給對應的wxml的image標簽設置data-set=“圖片路徑”,否則接收不到 var res = e.target.dataset.src var list = this.data.previewImgList //頁面的圖片集合數(shù)組 //判斷res在數(shù)組中是否存在,不存在則push到數(shù)組中, -1表示res不存在 if (list.indexOf(res) == -1) { this.data.previewImgList.push(res) } wx.previewImage({ current: res, // 當前顯示圖片的http鏈接 urls: that.data.previewImgList // 需要預覽的圖片http鏈接列表 }) }, //聊天消息始終顯示最底端 bottom: function () { var query = wx.createSelectorQuery() query.select('#flag').boundingClientRect() query.selectViewport().scrollOffset() query.exec(function (res) { wx.pageScrollTo({ scrollTop: res[0].bottom // #the-id節(jié)點的下邊界坐標 }) res[1].scrollTop // 顯示區(qū)域的豎直滾動位置 }) }, })
WXML:
系統(tǒng)消息: 歡迎 {{ userInfo.nickName }} 加入聊天室 {{item.date}} {{ item.nickName }} {{item.content}} {{ item.nickName }} {{item.content}}
WXSS:
/* pages/socks/socks.wxss */ page { background-color: #f7f7f7; height: 100%; } /* 聊天內容 */ .news { padding-top: 30rpx; text-align: center; /* height:100%; */ box-sizing:border-box; } #flag{ margin-bottom: 100rpx; height: 30rpx; } .chat-notice{ text-align: center; font-size: 30rpx; padding: 10rpx 0; color: #666; } .historycon { height: 100%; width: 100%; /* flex-direction: column; */ display: flex; border-top: 0px; } /* 聊天 */ .chat-news{ width: 100%; overflow: hidden; } .chat-news .my_right { float: right; /* right: 40rpx; */ padding: 10rpx 10rpx; } .chat-news .name{ margin-right: 10rpx; } .chat-news .you_left { float: left; /* left: 5rpx; */ padding: 10rpx 10rpx; } .selectImg{ display: inline-block; width: 150rpx; height: 150rpx; margin-left: 50rpx; } .my_right .selectImg{ margin-right: 80rpx; } .new_img { width: 60rpx; height: 60rpx; border-radius: 50%; vertical-align: middle; margin-right: 10rpx; } .new_txt { max-width: 300rpx; display: inline-block; border-radius: 6rpx; line-height: 60rpx; background-color: #95d4ff; padding: 5rpx 20rpx; margin: 0 10rpx; margin-left: 50rpx; } .my_right .new_txt{ margin-right:60rpx; } .you{ background-color: lightgreen; } .my { border-color: transparent transparent transparent #95d4ff; } .you { border-color: transparent #95d4ff transparent transparent; } .hei{ margin-top: 50px; height: 20rpx; } .history { height: 100%; margin-top: 15px; padding: 10rpx; font-size: 14px; line-height: 40px; word-break: break-all; } ::-webkit-scrollbar { width: 0; height: 0; color: transparent; z-index: -1; } /* 信息輸入?yún)^(qū)域 */ .message{ position: fixed; bottom:0; width: 100%; } .sendMessage{ display: block; height: 80rpx; padding: 10rpx 10rpx; background-color: #fff; border-top: 2rpx solid #eee; border-bottom: 2rpx solid #eee; z-index:3; } .sendMessage input{ float:left; width: 66%; height: 100%; line-height: 80rpx; border-bottom: 1rpx solid #ccc; padding:0 10rpx; font-size: 35rpx; color: #666; } .sendMessage view{ display: inline-block; width: 80rpx; height: 80rpx; line-height: 80rpx; font-size: 60rpx; text-align: center; color: #999; border: 1rpx solid #ccc; border-radius: 50%; margin-left: 10rpx; } .sendMessage button { float: right; font-size: 35rpx; } .increased{ width:100%; /* height: 150rpx; */ padding: 40rpx 30rpx; background-color: #fff; } .increased .image{ width: 100rpx; height: 100rpx; border: 3rpx solid #ccc; line-height: 100rpx; text-align: center; border-radius: 8rpx; font-size:35rpx; } @keyframes slidedown { from { transform: translateY(0); } to { transform: translateY(100%); } } .slidedown { animation: slidedown 0.5s linear ; } .slideup { animation: slideup 0.5s linear ; } @keyframes slideup { from { transform: translateY(100%); } to { transform: translateY(0); } }
總結
以上所述是小編給大家介紹的微信小程序實現(xiàn)即時通信聊天功能的實例代碼,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對創(chuàng)新互聯(lián)網(wǎng)站的支持!