Laravel View Composer - 当 include 一个模板时,自动获取其所需的变量
网站中,许多页面的侧边栏是相同的。例如:
分类列表页,与文章详情页的侧边栏都包含
- 最新文章
- 最新评论
- 统计计数
这些相同的侧边栏数据也是动态的,并不是固定的。
在每个 controller 里都写一遍数据读取逻辑,并赋值给模板,这样做有几个弊端
- 重复编码量不小。即使提取成函数,也要手动将变量传递给模板,controller 中无法专注于业务逻辑
- 不利于模块化
简单粗暴的做法 - 全局 view share
之前,我用了一种简单粗暴的解决方案:
在 app/Providers/AppServiceProvider.php 里全局共享模板变量,即,View::share。
但是,这种做法的弊端是
- 及时某些页面不需要这些变量,也会浪费时间去获取一遍
- 代码越来越臃肿
View Composer - 当 include 一个 view 时,自动获取其所需的变量
Laravel 的 View Composer 很好的解决了这个问题。
官方文档: https://laravel.com/docs/5.5/views#view-composers
View composer 实际上是一个回调,当需要渲染一个指定 blade 模板时,被自动调用。
View Composer 需要在 service provider 中进行注册。即,新建一个 Service Provider, 然后加入到 config/app.php 中,方便 Laravel 启动时自动载入。
这个 view composer 可以偷懒地写在 AppServiceProvider.php 中,但是,带来的问题就是代码乱的一塌糊涂。还是放到独立的文件中比较好。
class based composers VS Closure based composers
class based composers
View::composer(
'profile', 'App\Http\ViewComposers\ProfileComposer'
);
Closure based composers
View::composer('dashboard', function ($view) {
//
});
我更倾向于使用 Closure based composers,这样做的好处有
- 省去新建一个 composer 文件。因为如果没有复用、没有解耦的需求,新建一个代码文件是件我无法理解的事情
- 逻辑一目了然,不需要再切到另外一个文件查看
如果通过 $view->with 给一个模板传递多个变量
// 一个变量
$view->with('count', 1);
// 多个变量
$view->with('count', 1)
->with(‘site’, ‘sunzhongwei.com’);
如何通过 laravel 命令创建 View Composer 文件
View composer 统一放到目录
app/Http/ViewComposers
下。
Laravel 内置的命令
make:provider Create a new service provider class
即
php artisan make:provider SidebarCompoerServiceProvider
为何用这么长的名字?因为可以在 provider 列表里一目了然。
app/Providers/SidebarCompoerServiceProvider.php
namespace App\Providers;
use Illuminate\Support\ServiceProvider;
class SidebarCompoerServiceProvider extends ServiceProvider
{
/**
* Bootstrap the application services.
*
* @return void
*/
public function boot()
{
// View::composer
}
/**
* Register the application services.
*
* @return void
*/
public function register()
{
//
}
}
将该文件从 app/Providers 移动到 app/Http/ViewComposers 目录下。
添加 service provider 到 config.php 中
参考 AppServiceProvider 的做法
App\Providers\AppServiceProvider::class,
一个 View Composer 给多个 blade 模板提供数据
View::composer(
['profile', 'dashboard'],
'App\Http\ViewComposers\MyViewComposer'
);
当然也可以使用星号为所有模板提供数据 (严重不推荐, 这跟之前的 app service provider 中的做法没有本质区别)
Laravel Service Provider 是什么?
https://oomusou.io/laravel/laravel-service-provider/
我发现很难理解 Laravel Service Provider 的概念。
先从最常见的使用场景来看:
我们安装一个新的 Laravel 包,总是先 composer install xxx,然后修改 config/app.php 文件,在 providers 里加入 XXX::class 。
疑问来了,composer install 之后为何还要手动配置 provider.
因为,composer install 只是将三方包的代码安装到 vendor 目录下。
但是,Laravel 并不知道这个三方包的存在,如果想让 laravel 启动时自动加载这个三方包,就需要在 config/app.php 的 providers 中加入这个包。
所以,service provider 存在的意义就是方便管理哪些模块应该在 laravel 启动时自动载入的了?
Register 和 Boot 的使用,我还是看不懂?
一个完整的例子
创建 View Composer Service Provider
php artisan make:provider SidebarCompoerServiceProvider
mkdir app/Http/ViewComposers
mv app/Providers/SidebarComposerServiceProvider.php app/Http/ViewComposers/
代码填充如下
<?php
namespace App\Http\ViewComposers; // 注意修改 namespace
use Illuminate\Support\ServiceProvider;
use Illuminate\Support\Facades\View;
use App\Models\WebExamInfo;
use App\Models\Live;
class SidebarComposerServiceProvider extends ServiceProvider
{
/**
* Bootstrap the application services.
*
* @return void
*/
public function boot()
{
View::composer('detail_sidebar', function ($view) {
// 招考信息
$exams = WebExamInfo::orderBy('id', 'desc')
->limit(6)
->get();
// 直播信息
$lives = Live::where('is_deleted', 0)
->orderBy('id', 'desc')
->limit(6)
->get();
$view->with('exams', $exams)
->with('lives', $lives);
});
}
/**
* Register the application services.
*
* @return void
*/
public function register()
{
//
}
}
最后在 config/app.php 的 providers 中添加
App\Http\ViewComposers\SidebarComposerServiceProvider::class,
对应的修改 blade 渲染逻辑即可。
Laravel View Composer - 当 include 一个模板时,自动获取其所需的变量的更多相关文章
- zTree异步生成数据时无法获取到子节点的选中状态
最近在项目中遇到一个问题,需求如下: 根据选中不同的人员(ID)向后台发送ajax请求,通过返回的数据来生成该人员的权限访问树,该树目录最少为3级目录,在生成的时候会自动勾选上次保存过的选中状态,点击 ...
- VMware vSphere 服务器虚拟化之二十二桌面虚拟化之创建View Composer链接克隆的虚拟桌面池
VMware vSphere 服务器虚拟化之二十二桌面虚拟化之创建View Composer链接克隆的虚拟桌面池 在上一节我们创建了完整克隆的自动专有桌面池,在创建过程比较缓慢,这次我们将学习创建Vi ...
- win7,Mindmanager2012使用模板时弹出Runtime error R6025解决方法
Mindjet.MindManager2012.v10.0在应用个别模板时提示"参数错误",然后自动关闭. 解决办法: 如果是win7系统,可以进入C:\Users\(用户名)\A ...
- iOS开发UI篇—使用picker View控件完成一个简单的选餐应用
iOS开发UI篇—使用picker View控件完成一个简单的选餐应用 一.实现效果 说明:点击随机按钮,能够自动选取,下方数据自动刷新. 二.实现思路 1.picker view的有默认高度为162 ...
- php包含(include/require)文件时的作用域
当一个php脚本被require/include时,他的变量作用域根据其上下文环境决定. 1.如果是在一个函数中require/include一个文件,则此文件中的变量作用域是这个函数的范围.也就是说 ...
- listview当选中某一个item时设置背景色其他的不变
listview当选中某一个item时设置背景色其他的不变: 可以使用listview.setOnFoucsChangeListener(listener) ; /** * listview获得焦点和 ...
- VMware vSphere 服务器虚拟化之十八桌面虚拟化之安装View Composer服务器
VMware vSphere 服务器虚拟化之十八桌面虚拟化之安装View Composer服务器 View Compose服务可安装在管理虚拟机的vC ...
- thinkphp 使每一个模板页都包括一个header文件和一个footer文件
在开发的过程中,常常遇到要使每一个模板页都包括一个header文件和一个footer文件.thinkPHP的模板布局为我们提供了一个叫全局配置方式可以解决问题. 1. 在配置文件里开启LAYOUT_O ...
- FreeMarker 语法 include 引用模板
一.java 代码 @Test public void testFreeMarker() throws Exception { //1.创建一个模板文件 //2.创建一个Configuration对象 ...
随机推荐
- Spring(2)
前言:控制反转(Inversion of Control,英文缩写为IoC)是一个重要的面向对象编程的法则来削减计算机程序的耦合问题,也是轻量级的Spring框架的核心. 控制反转一般分为两种类型,依 ...
- Hadoop生态圈-Cloudera Manager扫盲篇
Hadoop生态圈-Cloudera Manager扫盲篇 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在讨论CDH.cloueramamager之前,我要问大家几个问题: 1& ...
- H3C配置FTP服务器
H3C配置FTP服务器 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.FTP协议简介 1.FTP协议是互联网上广泛使用的文件传输协议 FTP文件传送协议(File Transf ...
- JS 中函数名后面加与不加括号的区别
a.onmouseover = fn1; a.onmouseout = fn2; function fn1(){ div.className = "erweima show"; } ...
- javascript 模拟京东关闭广告栏
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- JAVA记录-消息队列介绍
1.JMS概述 JMS即Java消息服务(Java Message Service)应用程序接口,是一个Java平台中关于面向消息中间件(MOM)的API,用于在两个应用程序之间,或分布式系统中发送消 ...
- UESTC - 1172 三句话题意
题目链接 记一个集合的gcd为该集合内所有数的最大公约数, 求一个给定集合的非空子集的gcd的k次方的期望~ Input 第一行有一个数t,表示数据组数 接下去每组数据两行,第一行两个数n,k(0 & ...
- SpringMVC学习笔记_01
1.JAVAEE体系结构 JAVAEE体系结构图如下所示: 2.什么是springmvc? 什么是mvc? Model1 Model2 SpringMVC是什么? SpringMVC是一个web层mv ...
- android get或post及HttpClient与服务器数据交互
1.Service package mydemo.mycom.demo2.service; import org.apache.http.HttpResponse; import org.apache ...
- elasticsearch 单机多实例
elasticsearch 配置单机器多实例 host: - - path data: /opt/elasticsearch/data/node1 /opt/elasticsearch/data/no ...