express+gulp构建项目(三)gulp任务

这次来看一看gulp是怎么工作的。
tasks/paths.js
paths.js文件里存放的是gulp任务中需要导入的文件的路径和导出的路径。
/**
* gulp.src 地址
* gulp.dest 地址
*/ var commonSrc = './public/**/*';
var widgetSrc = '!./public/widget/**/*.*';
var bowerSrc = '!./public/bower_components/**/*.*';
var fileOutput = '!./public/assets/**/*.*';
var nodeModulesSrc = '!./public/node_modules/**/*.*';
var nodeModulesEtcSrc = '!./public/etc/**/*.*';
var commonDest = './public/assets'; module.exports = {
js: {
src: [
commonSrc + '.js',
widgetSrc,
bowerSrc,
nodeModulesSrc,
nodeModulesEtcSrc,
fileOutput
],
dest: commonDest
},
node: {
src: ['./config/**/*.js', './controllers/**/*.js', './models/**/*.js', './swig/**/*.js']
},
img: {
src: [
commonSrc + '.png',
commonSrc + '.jpg',
commonSrc + '.jpeg',
commonSrc + '.gif',
commonSrc + '.bmp',
widgetSrc,
bowerSrc,
nodeModulesSrc,
nodeModulesEtcSrc,
fileOutput
],
dest: commonDest
},
css: {
src: [
commonSrc + '.css',
commonSrc + '.scss',
widgetSrc,
bowerSrc,
nodeModulesSrc,
nodeModulesEtcSrc,
fileOutput
],
dest: commonDest
},
cleanSrc: [
'./hash',
commonDest
]
};
这样子把gulp任务用到的路径集中到一个文件里面好管理,也方便。
gulp任务的路径书写方式是依照node-glob的语法。
glob语法基本如下:
*:匹配该路径段中0个或多个任意字符。
js/*.js
?:匹配该路径中1个任意字符。
js/?.js
[...]:匹配指定范围内的字符,与正则语法类似。
js/a[0-3].js
*(pattern|pattern|pattern):匹配括号中多个模型的0个或多个或任意个的组合,注意竖线|前后没有空格。
js/*(a|a1|b).js
!(pattern|pattern|pattern):匹配不包含任何模型。
js/!(a|b).js
?(pattern|pattern|pattern):匹配多个模型中的0个或任意1个,精确匹配,不可组合。
js/?(a|a2|b).js
+(pattern|pattern|pattern):匹配多个模型中的1个或多个。必须至少有一个,可以是这几个模型的组合。
js/+(a|a1|b).js
@(pattern|pattern|pattern):匹配多个模型中任意一个。精确匹配,不可组合,必须有一个,不可为空。
js/@(a|a1|b).js
**:可以匹配任何内容,但**不仅匹配路径中的某一段,而且可以匹配 'a/b/c' 这样带有'/'的内容,所以,它还可以匹配子文件夹下的文件。
**/@(a|a1|b).js
gulp_build.js
gulp_build文件是gulp任务主文件。
var gulp = require('gulp');//引入gulp自动化工具
var gulpif = require('gulp-if');//有条件的执行gulp任务
var runSequence = require('run-sequence');//运行多个gulp任务
var argv = require("yargs").argv;//接收用户在命令行输入的参数 例如:gulp build -s 中的 -s
var clean = require('gulp-clean');//删除文件或文件夹
var path = require('path'); //path模块用于处理和转换文件路径
var RevAll = require('gulp-rev-all-fixed');//静态资源版本迭代,解决文件缓存问题,其实就是给文件名后加上哈希值
var uglify = require('gulp-uglify');//js压缩
var cssmin = require('gulp-cssmin');//css压缩
var replace = require('gulp-replace');//字符串替换插件
var sass = require("gulp-sass");//预处理sass
var bower = require('gulp-bower');//下载bower包
var taskPaths = require('./tasks/paths'); //引入gulp任务所需要的路径
var jsonfile = require('jsonfile');//创建和解析json文件
var imgHashFile = {};
function getRevOpts(isDev) {//给gulp-rev-all-fixed模块生成选项的方法
var opt = {
hashLength: 12, //哈希值长度
transformFilename: function (file, hash) {
var ext = path.extname(file.path); //获取扩展名
var suffixes = isDev ? '' : '.' + hash.substr(0, 6); //获取哈希值前六位
if (ext.indexOf('scss') > 0) {//如果是scss文件,就变成css扩展名
ext = '.css';
}
return path.basename(file.path, ext) + suffixes + ext;
//path.basename(path, ext)返回路径的最后一段,例如./name/abc.html 则会返回abc.html
//path.extname(path, ext)返回路径最后一段的扩展名,例如./name/abc.html 则返回.html
//不带扩展名的基础文件名,后面连上哈希值,最后连上处理过的扩展名返回
}
};
return opt;
}
gulp.task('bower', function() {//bower任务,下载bower插件到以下目录
return bower({ directory: './public/bower_components' });
});
//clean任务,删除hash和assets文件夹下的文件,read: false参数使gulp不读取文件内容就进行删除,使操作变快
gulp.task('clean', function () {
return gulp.src(taskPaths.cleanSrc, { read: false })
.pipe(clean());
});
var temmoGulp = {
buildJs: function (src, dest, isDev, isWatching) {
var opts = {
fileNameManifest: 'js_hash.json',
hashLength: getRevOpts(isDev).hashLength,
transformFilename: getRevOpts(isDev).transformFilename
};
//gulp-rev-all-fixed选项,
//fileNameManifest,hash文件自定义命名
//transformFilename,自定义命名方法
var revAll = new RevAll(opts);//在将流导入revAll.revision()处理之前先要实例化RevAll对象
return gulp.src(src)
.pipe(uglify())
.pipe(revAll.revision())
.pipe(gulp.dest(dest))
.pipe(revAll.manifestFile()) //生成hash名单文件
.pipe(gulpif(!isWatching, gulp.dest('./hash'))); //将hash名单文件导入hash文件夹
},
buildImg: function (src, dest, isDev, isWatching) {
var opts = {
fileNameManifest: 'img_hash.json',
hashLength: getRevOpts(isDev).hashLength,
transformFilename: getRevOpts(isDev).transformFilename
};
var revAll = new RevAll(opts);
return gulp.src(src)
.pipe(revAll.revision())
.pipe(gulp.dest(dest))
.pipe(revAll.manifestFile())
.pipe(gulpif(!isWatching, gulp.dest('./hash')));
},
buildCss: function (src, dest, isDev, isWatching) {
var opts = {
fileNameManifest: 'css_hash.json',
hashLength: getRevOpts(isDev).hashLength,
transformFilename: getRevOpts(isDev).transformFilename
};
var revAll = new RevAll(opts);
var compassImporter = function(url, prev, done) {
if (!/^compass/.test(url)) {
return done({ file: url }); //如果url中不含compass
}
done({ file: 'compass-mixins/lib/' + url });
};
//url是当前导入的路径,prev是上一次处理的路径,done是可选回调函数
//outputStyle最终输出css的风格
//importer处理当遇到@import导入
//includePaths一个路径的数组让Libsass可以解决@import声明
//data传递给Libsass渲染
return gulp.src(src)
.pipe(sass({
outputStyle: 'compressed',
importer: compassImporter,
includePaths: [ 'node_modules' ],
data: '@import "compass"; .transition { @include transition(all); }'
}))
.pipe(sass().on('error', sass.logError))
.pipe(cssmin()) //压缩css
.pipe(replace(/(\.\.\/.*?(jpg|jpeg|gif|png|bmp){1})/mg, function ($1) {
//replace第二个参数如果是函数,每个匹配都调用该函数,它返回的字符串将作为替换文本使用
//$1参数是匹配到的字符串
var img = $1;
var imgRal = img.replace(/((\.\.\/){1,})(.*?(jpg|jpeg|gif|png|bmp){1})/g, '$1');
try {
imgHashFile = jsonfile.readFileSync('./hash/img_hash.json'); //读取img hash名单
} catch (e) {
imgHashFile = {};
}
for (var key in imgHashFile) {//循环img hash名单
if ((imgRal + key) === $1) {
img = '/static/assets/' + imgHashFile[key];
break;
}
}
return img; //将css中图片url文件名变为带hash的文件名
}))
.pipe(revAll.revision())
.pipe(gulp.dest(dest))
.pipe(revAll.manifestFile())
.pipe(gulpif(!isWatching, gulp.dest('./hash')));
}
};
var build = function () {
// 是否监听
var isDev = argv.dev === true;//若在命令行输入了dev参数,则isDev为真
gulp.task('js', function () {
return temmoGulp.buildJs(taskPaths.js.src, taskPaths.js.dest, isDev);
});
gulp.task('img', function () {
return temmoGulp.buildImg(taskPaths.img.src, taskPaths.img.dest, isDev);
});
gulp.task('css', function () {
return temmoGulp.buildCss(taskPaths.css.src, taskPaths.img.dest, isDev, false);
});
runSequence('clean', 'bower', 'js', 'img', 'css'); //顺序执行相关任务
};
gulp.task('build', function () { //build任务包含clean,下载bower包,处理js,处理img,处理css
return build();
});
var getFileName = function (str) {
var reg = /[^\\\/]*[\\\/]+/g; //获取路径里的文件名
str = str.replace(reg, '');
return str;
};
var getSinglePath = function (file) {
var filePath = file.path,
destPath,
fileName = getFileName(file.path);
filePath = filePath.replace(/(\\|\/)/img, '%');
//将路径里的/和\换成%
filePath = filePath.substring(filePath.indexOf('%public%'));
//从$public%开始截取
filePath = '.' + filePath.replace(/%/img, '/');
//将%替换回/,导入路径
destPath = filePath.replace(eval('/(' + fileName + ')$/mg'), '');
//去掉文件名
destPath = destPath.replace(eval('/\.\\/public\\//'), ('./public/assets/'));
//导出路径为./public/assets/下
return {
filePath: filePath,
destPath: destPath
};
};
gulp.task('watch', function () { //监控js,css,img的变动,如果变动,重新执行处理任务
gulp.watch(taskPaths.css.src, function (file) {
var paths = getSinglePath(file);
gulp.task('watchCss', function () {
return temmoGulp.buildCss(paths.filePath, paths.destPath, true, true);
});
runSequence('watchCss');
});
gulp.watch(taskPaths.js.src, function (file) {
var paths = getSinglePath(file);
gulp.task('watchJs', function () {
return temmoGulp.buildJs(paths.filePath, paths.destPath, true, true);
});
runSequence('watchJs');
});
gulp.watch(taskPaths.img.src, function(file) {
var paths = getSinglePath(file);
gulp.task('watchImg', function () {
return temmoGulp.buildImg(paths.filePath, paths.destPath, true, true);
});
runSequence('watchImg');
});
});
package.json里的scripts里的命令就是执行的gulp命令:
"scripts": {
"start": "node ./bin/www",
"dev": "./node_modules/.bin/gulp develop --gulpfile gulp_dev.js",
"ci": "./node_modules/.bin/gulp test --gulpfile gulp_test.js",
"assets": "./node_modules/.bin/gulp build --gulpfile gulp_build.js",
"assets_dev": "./node_modules/.bin/gulp build --dev --gulpfile gulp_build.js",
"watch": "./node_modules/.bin/gulp watch --gulpfile gulp_build.js"
}
npm run assets 是执行了gulp build,处理js,css,img等静态资源;npm run dev是执行了gulp develop,启动node服务;npm run watch执行gulp watch监听文件的变动。
gulp_dev.js
这个文件里是启动服务的gulp任务。
var gulp = require('gulp');
var nodemon = require('gulp-nodemon'); //监控文件变化,有变化则立即重启服务
gulp.task("develop", function () {
nodemon({
script: "bin/www",
ext: "js",
ignore: ['./public', './node_modules']
}).on("restart", function () {
console.log('restart');
});
});
gulp_test.js
执行eslint检测
var gulp = require('gulp');
var runSequence = require('run-sequence');
var eslint = require('gulp-eslint');
var taskPaths = require('./tasks/paths');
gulp.task('eslint_node', function () {
return gulp.src(taskPaths.node.src)
.pipe(eslint({
useEslintrc: './node.eslintrc'
}))
.pipe(eslint.format())
.pipe(eslint.failOnError());
});
gulp.task('test', function () {
runSequence('eslint_node')
});
express+gulp构建项目(三)gulp任务的更多相关文章
- express+gulp构建项目(二)启动项目和主文件
这一次整理的内容是项目主文件和如何启动项目. 启动项目 通过nodejs官网的例子https://nodejs.org/docs/latest-v4.x/doc/api/synopsis.html我们 ...
- 使用gulp构建项目
gulp.js作为一个前端构建工具,类似于webpack.Grountjs.rollupjs,不过相对于其他几种打包工具,gulp的使用更轻量,配置更简单,打包速度更快,今天不说他们几个的区别,也不说 ...
- express+gulp构建项目(一)项目目录结构
express是基于nodejs平台的web框架,它可以让我们快速开发出web引用.而gulp是一种自动构建工具,非常强大,有了它,能帮我们完成很多繁琐的工作,例如,静态文件的压缩,为静态文件加上哈希 ...
- express+gulp构建项目(五)swig模板
这里的文件负责配置swig模板引擎. index.js var jsonHash = require('./json_file'); var staticTag = require("./t ...
- express+gulp构建项目(四)env环境变量
这里的文件的作用是负责设置env环境变量和日志. index.js try { require('dotenv').load({silent: true}); //dotenv从一个.env文件中读取 ...
- express - 快速构建项目
1,cnpm i express -g 2, cnpm i express-generater -g 3, express - e 项目名
- 使用gulp搭建项目
项目源码地址 前期准备工作 首先确保本机安装了 node gulp中文文档 安装 gulp 命令行工具 npm install --global gulp-cli 在项目目录下创建 package.j ...
- 用gulp构建你的前端项目
前言 前端技术发展日新月异,随着模块化.组件化的提出,前端变得越来越复杂,静态资源越来越多,那么对静态资源的处理,如压缩,合并,去掉调试信息.. 如果还是人工去处理,效率非常之低且还容易出错,于是自动 ...
- gulp进阶构建项目由浅入深
gulp进阶构建项目由浅入深 阅读目录 gulp基本安装和使用 gulp API介绍 Gulp.src(globs[,options]) gulp.dest(path[,options]) gulp. ...
随机推荐
- mysql中使用 where 1=1和 0=1 的作用
操作mysql的时候,经常使用where语句进行查询.当where语句不存在的时候,经常在后面加一个where 1=1 where 1=1; 这个条件始终为True,在不定数量查询条件情况下,1=1可 ...
- “error LNK2019: 无法解析的外部符号”之分析
最近在用VS 2008开发,初学遇到不少问题,最头疼的问题之一就是:LNK2019. 百度一下讲的并不够全面,反正都没解决我的问题. error LNK2019问题在VC 6.0中是error LNK ...
- 6.Git内容修改之后的查看和提交
我们已经成功地添加并提交了一个readme.txt文件,现在,是时候继续工作了,于是,我们继续修改readme.txt文件,改成如下内容: Git is a distributed version c ...
- 初始化datetimepicker的时候就报了js异常
全栈眼中的http这一章分别从前端视角和后端视角来分析前后端所关注的侧重点.前端可以通过抓包工具或者chrome devtools 查看每个请求,同域下的资源请求数量等来找出优化点,更关注的是一个页面 ...
- MFC 修改各种控件的背景颜色、字颜色和字体
今天主要总结一下有关MFC 中静态编辑框(StaticEdit).编辑框(Edit)和按钮(Button)的背景颜色.字颜色和字体. 我的程序运行结果如下: 由上图我们知道修改的地方有:1.把Stat ...
- [转] 将DOS格式文本文件转换成UNIX格式
点击此处阅读原文 用途说明 dos2unix命令用来将DOS格式的文本文件转换成UNIX格式的(DOS/MAC to UNIX text file format converter).DOS下的文本文 ...
- windows系统nginx配置root绝对路径的问题
看了下logs下面的error.log文件,发现路径有问题,修改了conf配置,把root的路径反斜杠要用两个反斜杠进行转义,再次运行正常了.
- Cocos2dx3.11.1Android播放视频,后台 黑屏,无法记忆播放bug修改
/* * Copyright (C) 2006 The Android Open Source Project * Copyright (c) 2014 Chukong Technologies In ...
- javascript高级程序设计第5章,引用类型
object类型: 创建object实列的方式有两种,一种是new()方法,一种是对象字面量表示法: 第一种法方: var obj = new object(); obj.name = 'name' ...
- 据说年薪30万的Android程序员必须知道的帖子
Android中国开发精英 目前包括: Android开源项目第一篇--个性化控件(View)篇 包括ListView.ActionBar.Menu.ViewPager.Gallery.G ...