laravel 基础教程 —— 炼金药

Laravel Elixir(炼金药)

简介

Laravel Elixir 为你的应用定义基础的 Gulp 任务提供了简单流利的 API。Elixir 提供了几种常用的 CSS 和 JavaScript 预处理器和测试工具。Elixir 允许你通过链式调用来对你的资源管道进行流利的操作。比如:

elixir(function (mix) {
  mix.sass('app.scss')
     .coffee('app.coffee');
})

如果你曾对如何使用 Gulp 和资源预编译有所疑惑,那么你肯定会爱上 Laravel Elixir。事实上,你也可以在开发应用的时候不使用它。你可以自由的使用任何的资源管道工具,或者一点也不用。

安装 & 起步

安装 Node

在触碰到 Elixir 之前,你首先需要确定你的机器中已经安装了 Node:

node -v

默认的,Laravel Homestead 包含了所有你所需要的;事实上,如果你使用 Vagrant,你也是可以非常简单的通过 这里 来进行安装 Node。

Gulp

接着,你需要通过 NPM 中安装 Gulp 到全局:

npm install --global gulp

如果你使用了版本控制器,你可能希望去运行 npm shrinkwrap 来锁住你的 NPM 依赖:

npm shrinkwrap

一旦你运行了这条命令。你可以自由的提交 npm-shrinkwrap.json 文件到源码控制器中去。

Laravel Elixir

最后剩下的就是安装 Elixir 了!在一个新的 laravel 应用中,你会在根目录中发现 package.json 文件。你可以把它想象成 composer.json 文件。它的不同之处就是它定义的是 Node 依赖,而不是 PHP。你可以通过下面的命令来安装这些依赖:

npm install

如果你是在 Windows 系统中进行开发,或者运行在虚拟机中的 Windows 系统,你需要运行 npm install 命令的同时添加 --no-bin-links 选项:

npm install --no-bin-links

运行 Elixir

Elixir 是建立于 Gulp 之上的,所以运行 Elixir 任务你只需要在终端运行 gulp 命令就行了。添加 --production 标识到命令会指导 Elixir 去压缩你的 CSS 和 JavaScript 文件:

// Run all tasks...
gulp

// Run all tasks and minify all CSS and JavaScript...

监控资源文件的变动

为了不让你每次文件变动之后还要重新运行 gulp 命令,你应该使用 gulp watch 命令来监控资源文件的变动。这个命令会持续的在你的终端中运行。当检测到资源文件的变动,新的文件将自动编译完成:

gulp watch

与样式文件合作

在你项目的根目录中有一个 gulpfile.js 文件,该文件包含了所有的 Elixir 任务。Elixir 任务可以被链式的调用,会通过有序的传递来对你的资源文件进行编译。

Less

你可以使用 less 方法来将 less 文件编译为 CSS。less 方法假设你的 less 文件存放在 resources/assets/less 目录中。默认的,在下面的示例中,任务运行的结果将编译的 CSS 文件存放在 public/css/app.css:

elixir(function (mix) {
 mix.less('app.less');
});

你也可以合并多个 less 文件到一个单独的 CSS 文件中。默认的,他们将被编译为 public/css/app.css 文件:

elixir(function (mix) {
mix.less([
  'app.less',
  'controllers.less'
]); 
});

如果你希望将编译的文件存放到自定义的位置,那么你可以传递第二个参数到 less 方法:

elixir(function (mix) {
mix.less('app.less', 'public/stylesheets');
});

// Specifying a specific output filename...
elixir(function (mix) {
mix.less('app.less', 'public/stylesheets/style.css');
});

sass

sass 方法允许你讲 sass 文件编译为 CSS。它假定你的 Sass 文件存放在 resources/assets/sass,你可以像下面的方式来使用该方法:

elixir(function (mix) {
  mix.sass('app.scss'); 
});

就像 less 方法一样,你可以编译多个 Sass 文件到一个 CSS 文件中,还可以将编译的结果存放到指定的位置:

elixir(function (mix) {
  mix.sass([
    'app.scss',
    'controllers.scss'
  ], 'public/assets/css'); 
});

原生 CSS

如果你希望合并多个 CSS 文件到一个文件中,你可以使用 styles 方法。你需要传递文件的路径是相对于 resources/assets/css 目录的,并且默认的合并的结果将会被存放到 public/css/all.css:

elixir(function (mix) {
  mix.styles([
    'normalize.css',
    'main.css'
  ]);
});

当然,你也是可以自定义输出结果的路径:

elixir(function (mix) {
  mix.styles([
    'normalize.css',
    'main.css'
  ], 'public/assets/css'); 
});

编译地图

编译地图是开箱即用的。所以,对于所有的被编译后的文件你都可以在相同的目录下发现 *.css.map 文件。这个地图文件可以使你在浏览器中追踪到编译前代码的位置,这样方便于调试。

如果你不希望生成地图,你可以在配置中进行关闭:

elixir.config.sourcemaps = false;

elixir(function (mix) {
  mix.sass('app.scss');
});

与脚本合作

Elixr 也提供了多种方法来帮助你协同 JavaScript 的工作,例如 ECMAPScript 6,CoffeeScript 的编译,Browserify 模块的加载,脚本的压缩,或者是简单的原生 JavaScript 文件的合并,这都不是问题!

CoffeeScript

coffee 方法可以用来编译 CoffeeScript 到原生 JavaScript。coffee 方法可以接收一个字符串或者一个 CoffeeScript 文件的数组来进行文件的编译,它假设你的 CoffeeScript 文件存放在 resources/assets/coffee 目录,并且合并生成的 JavaScript 文件到 public/js/app.js

elixir(function (mix) {
  mix.coffee(['app.coffee', 'controllers.coffee']);
});

Browserify

Elixir 也附带了 browserify 方法,该方法可以在浏览器中给你提供你所需要模块的加载,并且允许你使用 ECMAScript 6 和 JSX。

这个任务假设你的脚本存放在 resources/assets/js 并且会将结果文件输出到 public/js/main.js。你也可以传递第二个参数来指定输出的位置:

elixir(function (mix) {
  mix.browserify('main.js');
});

// Specifying a specific output filename...
elixir(function(mix) {
  mix.browserify('main.js', 'public/javascripts/main.js');
});

而 Browserify 附带了 Partialify 和 Babelify 转换器,你可以自由的安装你所希望的:

npm install aliasify --save-dev
elixir.config.js.browserify.transformers.push({
  name: 'aliasify',
  options: {} 
});

elixir(function (mix) {
  mix.browserify('main.js');
});

Babel

babel 方法可以使你编译 ECMAScript 6 和 7JSX 到原生的 JavaScript。该方法接收一个文件列表相对于 resources/assets/js 目录的数组,并且在 public/js 目录中生成 all.js 文件:

elixir(function (mix) {
  mix.babel([
    'order.js',
    'product.js',
    'react-component.jsx'
  ]); 
});

你可以传递第二个参数来指定不同的输出路径。除了进行 Babel 编译,其方法的功能和 mix.scripts 一样。

Scripts

如果你想要将多个 JavaScript 文件合并到一个文件中,你可以使用 scripts 方法。

scripts 方法假设你所有的文件都相对于 resouces/assets/js 目录,并且会默认的将结果编译到 public/js/all.js 文件中:

elixir(function (mix) {
  mix.scripts([
    'jquery.js',
    'app.js'
  ]);
})

如果你需要合并多个文件到多个不同的路径,你可以通过多次链式调用并传递第二个参数作为指定输出的路径:

elixir(function (mix) {
  mix.scripts(['app.js', 'controllers.js'], 'public/js/app.js')
     .scripts(['forum.js', 'threads.js'], 'public/js/forum.js');
})

如果你需要合并指定目录下的所有脚本文件,你可以使用 scriptIn 方法。合并的结果将存放到 public/js/all.js:

elixir(function (mix) {
  mix.scriptsIn('public/js/some/directory')
});

复制文件 & 目录

copy 方法可以用来复制文件和目录到一个新的位置。所有的操作都是相对于项目的根目录:

elixir(function (mix) {
  mix.copy('vendor/foo/bar.css', 'public/css/bar.css')
})

elixir(function(mix) {
  mix.copy('vendor/package/views', 'resources/views')
})

版本 / 缓存移除

对于许多开发者比较痛苦的事就是手动的对资源文件增加时间戳或者唯一的 token 标识来强迫浏览器重新加载新的资源文件。Elixir 可以通过 version 方法来帮你自动的完成这些。

version 方法接收文件名称相对于 public 目录,并且它会自动的为文件名增加一个独特的 hash,这样就可以自动的进行缓存清除了。比如,新生成的文件名看上去像这样:all-16d570a7.css:

elixir(function (mix) {
  mix.version('css/all.css') 
});

在生成版本化的文件之后,你可以使用 laravel 的全局帮助函数 elixir 在你的视图文件中进行加载适当的 hashed 资源。elixir 方法会自动的判断文件的名称:

<link rel="stylesheet" href="{{ elixir('css/all.css') }}">

对多个文件进行版本化

你可以传递一个数组到 version 方法来进行多个文件的版本化:

elixir(function (mix) {
  mix.version(['css/all.css', 'js/app.js']);
});

一旦文件本版本化,你就可以使用 laravel 的 elixir 方法去生成版本化的 link。记住,你只需要向 elixir 帮助方法中传递文件名的前缀就可以了,并不需要填写 hash 后的文件名。帮助方法会自动的识别 hash 后的文件名:

<link rel="stylesheet" href="{{ elixir('css/all.css') }}">

<script src="{{ elixir('js/app.js') }}"></script>

BrowserSync

BrowserSync 可以在你的前端资源文件变更之后自动的刷新的你浏览器。你可以使用 browserSync 方法来指导 Elixir 当你运行 gulp watch 命令时启动 BrowserSync 服务:

elixir(function (mix) {
  mix.browserSync(); 
});

一旦你运行 gulp watch,你可以通过访问你的应用使用 3000 端口来启用 browsersyncing: http://homestead.app:3000。如果你使用了 homestead.app 之外的域名作为本地开发的支持,你可以传递一个 options 数组到第一个参数到 browserSync 方法:

elixir(function (mix) {
  mix.browserSync({
    proxy: 'project.app'
  }) 
});

调用已存在的 Gulp 任务

如果你需要从 Elixir 中调用已经存在的 Gulp 任务。你可以使用 task 方法。你可以想象一下你有一个 Gulp 任务是用来简单的发送一个文本:

gulp.task('speak', function () {
  var message = 'Tea...Earl Grey...Hot';

  gulp.src('').pipe(shell('say ' + message))
})

如果你想通过 Elixir 来调用这个任务,你可以使用 mix.task 方法并且传递任务的名称到该方法:

elixir(function (mix) {
  mix.task('speak')
})

自定义监控

如果你需要注册一个监控者在你的自定义任务中监控每次文件的变更。你可以传递一个正则表达式作为第二个参数到 task 方法:

elixir(function (mix) {
  mix.task('speak', 'app/**/*.php') 
})

编写 Elixir 扩展

如果你需要比 Elixir task 方法所提供的更高的灵活性,你可以创建自己的 Elixir 扩展。Elixir 扩展允许你传递参数到你的自定义任务中。比如,你可以编写一个下面类似的扩展:

// File: elixir-extensions.js

var gulp = require('gulp')
var shell = require('gulp-shell')
var Elixir = require('laravel-elixir')

var Task = Elixir.Task

Elixi.extends('speak', function (message) {
  new Task('speak', function () {
    return gulp.src('').pipe(shell('say ' + message))
  })  
})

// mix.speak('Hello World')

就是这么简单。你应该注意,任务的专有逻辑应该编写在方法中传递给 Tast 构造函数作为第二个参数。你也可以将其存放在 Gulpfile 的顶部或者提取到独立的扩展文件中。比如,你可以存放你的扩展到 elixir-extendsions.js,然后在你的 Gulpfile 文件中进行载入:

// File: Gulpfile.js

var elixir = require('laravel-elixir')

require('./elixir-extendsions')

elixir(function (mix) {
  mix.speak('Tea, Earl Grey, Hot')
})

自定义监控者
如果你希望在运行 gulp watch 时,你的自定义任务可以在文件变更时自动的触发,你可以注册一个监控者:

new Task('speak', function () {
  return gulp.src('').pipe(shell('say ' + mesage))
})
.watch('./app/**')
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容

  • gulpjs是一个前端构建工具,与gruntjs相比,gulpjs无需写一大堆繁杂的配置参数,API也非常简单,学...
    build1024阅读 527评论 0 0
  • 前言 本文默认你已经安装好node环境,并且熟悉node命令,和window cd命令。 gulp简介 基于nod...
    9I阅读 1,979评论 4 50
  • 对网站资源进行优化,并使用不同浏览器测试并不是网站设计过程中最有意思的部分,但是这个过程中的很多重复的任务能够使用...
    懵逼js阅读 1,061评论 0 8
  • gulpjs是一个前端构建工具,与gruntjs相比,gulpjs无需写一大堆繁杂的配置参数,API也非常简单,学...
    井皮皮阅读 1,293评论 0 10
  • gulpjs是一个前端构建工具,与gruntjs相比,gulpjs无需写一大堆繁杂的配置参数,API也非常简单,学...
    依依玖玥阅读 3,150评论 7 55