实时聊天室:基于Laravel+Pusher+Vue通过事件广播实现

发布时间 - 2018-07-31 00:00:00    点击率:

之前有说过要整理出一篇事件广播的教程,今天终于有时间把这篇文章给写了出来,本次的教程是基于laravel+pusher+vue,以事件广播作为核心技术,让你可以快速搭建起一个实时聊天室应用,话不多说,让我们来直接看看具体的内容吧。

应用初始化

安装配置

首先还是通过 Composer 安装一个全新的聊天室应用:

composer create-project laravel/laravel chatroom --prefer-dist

由于要用到事件广播,所以需要取消 config/app.php 中广播服务提供者前面的注释:

修改 .env 中 BROADCAST_DRIVER 配置项为 pusher:

BROADCAST_DRIVER=pusher

尽管 Laravel 开箱支持 Pusher,但是我们还是需要安装对应的 PHP SDK:

composer require pusher/pusher-php-server

设置 Pusher 凭证信息

访问 Pusher 官网,注册并登录到用户后台,创建一个新的 Channels App:

创建完成后即可在跳转页面中获取到 App Keys 相关信息:

将对应字段填充到聊天室应用根目录下的 .env 相应配置项即可。

前端资源初始化

我们使用 Laravel Mix 来编译前端 CSS 和 JavaScript:

npm install

此外,Laravel 还提供了 JavaScript 库 Laravel Echo 来订阅和监听事件:

npm install --save laravel-echo pusher-js

安装完成,还要告知 Laravel Echo 使用 Pusher,Laravel 已经在 resources/assets/js/bootstrap.js 中为我们提供了该实现,只不过默认注释起来了,只需要取消这段注释即可:

import Echo from 'laravel-echo'
window.Pusher = require('pusher-js');
window.Echo = new Echo({
    broadcaster: 'pusher',
    key: process.env.MIX_PUSHER_APP_KEY,
    cluster: process.env.MIX_PUSHER_APP_CLUSTER,
    encrypted: true
});

用户认证脚手架代码

我们设定只有登录用户才能进入聊天室进行聊天,为了简化流程,我们使用 Laravel 默认的用户认证功能:

php artisan make:auth

上述命令会为我们生成用户认证系统所必须的路由、视图、控制器等代码。在功能生效之前,还需要运行数据库迁移命令生成对应数据表,编辑 .env 中数据库相关配置项,保证可以正确连接上数据库,然后运行以下命令:

php artisan migrate

至此,应用初始化准备工作已完成,下面开始编写业务代码。

业务代码实现

消息模型

首先要为发送的消息创建一个模型类及其对应数据库迁移文件:

php artisan make:model Message -m

在新生成的 app/Messaage 模型类中新增下面这行代码以方便批量赋值:

/**
 * Fields that are mass assignable
 *
 * @var array
 */
protected $fillable = ['message'];

然后在 databases/migrations 目录下编写刚生成的 messages 对应迁移文件的 up 方法:

Schema::create('messages', function (Blueprint $table) {
    $table->increments('id');
    $table->integer('user_id')->unsigned();
    $table->text('message');
    $table->timestamps();
});

最后执行迁移命令生成数据表 messages:

php artisan migrate

用户与消息的关联关系

很显然,用户与消息之间是一对多的关系,在 User 模型类中新增关联方法:

/**
 * A user can have many messages
 *
 * @return \Illuminate\Database\Eloquent\Relations\HasMany
 */
public function messages()
{
    return $this->hasMany(Message::class);
}

接下来在 Message 模型类中定义与之相对的关联关系:

/**
 * A message belong to a user
 *
 * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
 */
public function user()
{
    return $this->belongsTo(User::class);
}

控制器代码

创建控制器 ChatsController 实现具体业务逻辑:

php artisan make:controller ChatsController

编写刚生成的控制器类 app/Http/Controllers/ChatsController 代码如下:

middleware('auth');  // 登录用户才能访问
    }
    /**
     * Show chats
     *
     * @return \Illuminate\Http\Response
     */
    public function index()
    {
        return view('chat');
    }
    /**
     * Fetch all messages
     *
     * @return Message
     */
    public function fetchMessages()
    {
        return Message::with('user')->get();
    }
    /**
     * Persist message to database
     *
     * @param  Request $request
     * @return Response
     */
    public function sendMessage(Request $request)
    {
        $user = Auth::user();
        $message = $user->messages()->create([
            'message' => $request->input('message')
        ]);
        return ['status' => 'Message Sent!'];
    }
}

该控制器提供了三个业务方法,index 用于显示聊天室视图,fetchMessages 用户获取所有消息,sendMessage 用于发送消息。

注册应用路由

对应地,我们在 routes/web.php 中注册三个路由:

Route::get('/', 'ChatsController@index');
Route::get('messages', 'ChatsController@fetchMessages');
Route::post('messages', 'ChatsController@sendMessage');

从注册路由中移除 /home 路由,相应地,需要把 app/Http/Controllers/Auth/LoginController.php 和 app/Http/Controllers/Auth/RegisterController.php 中的 $redirectTo 属性进行调整:

protected $redirectTo = '/';

聊天室视图

对于聊天室视图代码,我们基于 Bootsnipp 聊天室代码片段 稍作调整。首先创建 resources/views/chat.blade.php:

@extends('layouts.app')
@section('content')
    
        
            
                
                    聊天室
                    
                        
                    
                    
                        
                    
                
            
        
    
@endsection

该视图用于展示聊天室主体页面。注意到我们在视图中使用了一些 Vue 组件,chat-messages 组件用于显示所有聊天信息,chat-form 组件用于发送消息,稍后会给出这些组件代码。

在编写 Vue 组件之前,我们在 resources/views/layouts/app.blade.php 模板中为 chat 视图添加一些样式代码(添加到 标签之前):

接下来在 resources/assets/js/components 中创建 ChatMessages.vue 组件:


然后在同一目录下创建 ChatForm.vue 组件:


最后我们需要将这两个组件注册到位于 resources/assets/js/app.js 中的 Vue 根实例中:

require('./bootstrap');
window.Vue = require('vue');
Vue.component('chat-messages', require('./components/ChatMessages.vue'));
Vue.component('chat-form', require('./components/ChatForm.vue'));
const app = new Vue({
    el: '#app',
    data: {
        messages: []
    },
    created() {
        this.fetchMessages();
    },
    methods: {
        fetchMessages() {
            axios.get('/messages').then(response => {
                this.messages = response.data;
            });
        },
        addMessage(message) {
            this.messages.push(message);
            axios.post('/messages', message).then(response => {
                console.log(response.data);
            });
        }
    }
});

广播消息发送事件

为了在聊天室中进行实时交互,需要广播某些事件,在本例中,我们会在用户发送消息时触发 MessageSent 事件:

php artisan make:event MessageSent

编写 app/Events/MessageSent 事件类代码如下:

user = $user;
        $this->message = $message;
    }
    /**
     * Get the channels the event should broadcast on.
     *
     * @return \Illuminate\Broadcasting\Channel|array
     */
    public function broadcastOn()
    {
        return new PrivateChannel('chat');
    }
}

由于只有登录用户才能访问我们的应用,所以我们定义了一个私有的频道 chat,只有登录用户才能连接上它。

接下来,我们需要修改 ChatsController 的 sendMessage() 来广播 MessageSent 事件:

public function sendMessage(Request $request)
{
    $user = Auth::user();
    $message = $user->messages()->create([
        'message' => $request->input('message')
    ]);
    broadcast(new MessageSent($user, $message))->toOthers();
    return ['status' => 'Message Sent!'];
}

然后在 routes/channels.php 中授权当前登录用户可以监听该私有频道:

Broadcast::channel('chat', function ($user) {
    return Auth::check();
});

现在,当一条消息发送后,MessageSent 事件就会被广播到 Pusher,使用 toOthers() 是为了将消息发送者从广播接收者中排除。

监听消息发送事件

MessageSent 事件在服务端被广播后,需要在客户端监听这个事件以便将最新发送消息更新到聊天室消息流中,我们可以通过在 resources/assets/js/app.js 中定义的 created() 方法中添加如下代码片段来实现这一功能:

created() {
    this.fetchMessages();
    Echo.private('chat')
        .listen('MessageSent', (e) => {
            this.messages.push({
                message: e.message.message,
                user: e.user
            });
        });
},

我们通过 Laravel Echo 连接到 chat 频道监听 MessageSent 广播事件,如果有新消息则将其推送到当前聊天室消息流中显示。

在正式测试聊天室应用之前,还需要运行以下命令通过 Laravel Mix 来编译前面编写的 JavaScript 代码:

npm run dev

使用示例

完成上述所有业务代码编写工作后,接下来就是见证工作成果的时候了,在项目根目录下运行如下命令启动应用:

php artisan serve

然后在浏览器通过 http://127.0.0.1:8000/ 访问应用,由于系统需要登录后才能访问,所以首先会跳转到登录页面,我们需要先注册一个新用户,注册成功后页面即跳转到聊天室页面,我们发送一条测试消息。

为了测试多个用户聊天的效果,打开另一个浏览器或者在当前浏览器新开一个隐身窗口,还是重复上面的访问注册步骤(注册名不同),注册成功后跳转到聊天室页面,看到的效果和上面一样,我们再发条消息试试。

可以看到两个窗口消息是同步的,所以已经达到我们预期的实时聊天效果,实现了通过事件广播构建实时聊天室的功能。

以上就是本篇文章的全部内容了,更多laravel内容请关注laravel框架入门教程。

相关文章推荐:

laravel框架中TokenMismatchException的异常处理内容

Laravel 5.1框架中的ACL用户授权和权限检查功能的实现

相关课程推荐:

2017年最新的五个Laravel视频教程推荐


# php  # JavaScript  # laravel  # composer  # css  # bootstrap  # echo  # JS  # 事件  # 数据库  # http  # 聊天室  # 发送消息  # 跳转到  # 类中  # 还需要  # 创建一个  # 中为  # 注册成功  # 关联关系  # 这一 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何用西部建站助手快速创建专业网站?  Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程  如何批量查询域名的建站时间记录?  laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法  如何在腾讯云免费申请建站?  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  如何在建站主机中优化服务器配置?  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  微信h5制作网站有哪些,免费微信H5页面制作工具?  高性价比服务器租赁——企业级配置与24小时运维服务  网页制作模板网站推荐,网页设计海报之类的素材哪里好?  手机软键盘弹出时影响布局的解决方法  Laravel如何配置任务调度?(Cron Job示例)  html5源代码发行怎么设置权限_访问权限控制方法与实践【指南】  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  Bootstrap整体框架之CSS12栅格系统  如何注册花生壳免费域名并搭建个人网站?  Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】  Laravel怎么调用外部API_Laravel Http Client客户端使用  Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理  香港服务器部署网站为何提示未备案?  高端建站如何打造兼具美学与转化的品牌官网?  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  Android okhttputils现在进度显示实例代码  如何获取免费开源的自助建站系统源码?  Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】  Laravel软删除怎么实现_Laravel Eloquent SoftDeletes功能使用教程  Thinkphp 中 distinct 的用法解析  弹幕视频网站制作教程下载,弹幕视频网站是什么意思?  html文件怎么打开证书错误_https协议的html打开提示不安全【指南】  Laravel如何设置自定义的日志文件名_Laravel根据日期或用户ID生成动态日志【技巧】  Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)  php 三元运算符实例详细介绍  Win11怎么设置默认图片查看器_Windows11照片应用关联设置  Laravel如何实现API版本控制_Laravel API版本化路由设计策略  百度浏览器网页无法复制文字怎么办 百度浏览器复制修复  如何快速搭建支持数据库操作的智能建站平台?  Laravel路由怎么定义_Laravel核心路由系统完全入门指南  Laravel如何与Docker(Sail)协同开发?(环境搭建教程)  作用域操作符会触发自动加载吗_php类自动加载机制与::调用【教程】  制作电商网页,电商供应链怎么做?  Python自然语言搜索引擎项目教程_倒排索引查询优化案例  如何用已有域名快速搭建网站?  jQuery 常见小例汇总  Gemini手机端怎么发图片_Gemini手机端发图方法【步骤】  Claude怎样写约束型提示词_Claude约束提示词写法【教程】  中山网站制作网页,中山新生登记系统登记流程?  Laravel的HTTP客户端怎么用_Laravel HTTP Client发起API请求教程  Laravel如何生成URL和重定向?(路由助手函数)  Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】