美文网首页
使用命令行创建前端多页面模版(制作脚手架)

使用命令行创建前端多页面模版(制作脚手架)

作者: 该帐号已被查封_才怪 | 来源:发表于2020-08-11 17:33 被阅读0次

创建模版的方法有plop.jsyeoman及node cli方式,本文将以node cli为示例进行讲解:

一、前置知识

1、chalk

可以使用它来实现各种命令行字体的颜色;

const chalk = require('chalk');
console.log(chalk.green('success'));
console.log(chalk.red('error'));

2、inquirer

可以使用它在命令行实现与用户的一些交互如让用户选择你预定的选项或者让用户输入文字;

const inquirer = require('inquirer');
inquirer
 .prompt([
 // 一些交互式的问题
 ])
 .then(answers => {
 // 回调函数,answers 就是用户输入的内容,是个对象
 });

3、ora

在命令行实现一个好看的loading效果;

const ora = require('ora')
let spinner = ora('downloading template ...')
spinner.start()

4、download-git-repo

用它可以下载或者clone远程仓库

const download = require('download-git-repo')
download(repository, destination, options, callback)
// repository  表示远程仓库地址
// destination  表示存放下载的文件路径
// options  一些选项 如 { clone:boolean } 表示是clone 还是http下载

5、commander

用来编写指令和处理命令行的

const program = require("commander");
// 定义指令
program
 .version('0.0.1')
 .command('init', 'Generate a new project from a template')
 .action(() => {
 // 回调函数
 })
// 解析命令行参数
program.parse(process.argv);

6、handlebars

根据指定版本用来生成html的
(ps:除此之外,还有Mustach/ejs等)

7、figlet

制作欢迎页的


image.png

8、runjs

运行脚本(新tasksfile

9、其他

清屏 https://www.npmjs.com/package/clear
打开文件 https://www.npmjs.com/package/open
进度 https://www.npmjs.com/package/ora
任务列表 https://www.npmjs.com/package/listr

image.png

二、创建步骤

1、npm init
2、在 package.json里写入如下依赖并安装

"dependencies": {
 "chalk": "^2.4.2",
 "commander": "^2.19.0",
 "download-git-repo": "^1.1.0",
 "inquirer": "^6.2.2",
 "ora": "^3.2.0",
 "handlebars": "~4.7.6",
}

3、在项目根目录下新建 bin文件夹,并在里面新建无后缀名的文件init

#!/usr/bin/env node

const Handlebars = require('handlebars')
// 交互式命令行
const inquirer = require('inquirer')
// 修改控制台字符串的样式
const chalk = require('chalk')
// node 内置文件模块
const fs = require('fs')
const { resolve } = require('path')
const path = require('path')
// 读取根目录下的 template.json
var htmlTemplet='';
var packagejsonTemplet='';
var cssTemplet='';
var jsTemplet='';
// console.log('resolve',resolve('./'))
// console.log('cwd       : ' + process.cwd())
// console.log('__dirname : ' + __dirname)

// 读取文件操作
fs.readFile(`${__dirname}/../template/doNotChangeIt.html`, null, function(error, data) {
    if(error){
        console.log('读取doNotChangeIt.html文件出错了'+error)
    }else {
        htmlTemplet=data.toString()
        // console.log('成功读取index.html'+htmlTemplet)



    }

})
fs.readFile(`${__dirname}/../template/package.json`, function(error, data) {
    if(error){
        console.log('读取package.json文件出错了'+error)
    }else {
        packagejsonTemplet=data.toString()



    }

})

fs.readFile(`${__dirname}/../template/doNotChangeIt.css`, null, function(error, data) {
    // console.log('读取doNotChangeIt.css __dirname:'+__dirname)


    if(error){
        console.log('读取doNotChangeIt.css文件出错了'+error)
    }else {
        cssTemplet=data.toString()
        // console.log('成功读取index.html'+htmlTemplet)
        // console.log('读取doNotChangeIt.css==cssTemplet:'+cssTemplet)



    }

})
fs.readFile(`${__dirname}/../template/doNotChangeIt.js`, null, function(error, data) {
    if(error){
        console.log('读取doNotChangeIt.js文件出错了'+error)
    }else {
        jsTemplet=data.toString()
        // console.log('成功读取index.html'+htmlTemplet)


    }

})

// 拷贝文件

// function copyFile(src, dist) {
//     fs.writeFileSync(dist, fs.readFileSync(src));
// }
function copyFile(src, dist) {
    fs.createReadStream(src).pipe(fs.createWriteStream(dist));
}

// 检测目录并创建目录
// checkDirAndMkDir=(folderpath)=>{
//     const pathArr=folderpath.split('/');
//     let _path='';
//     for(let i=0;i<pathArr.length;i++){
//         if(pathArr[i]){
//             _path +=`/${pathArr[i]}`;
//             if (!fs.existsSync(_path)) {
//                 fs.mkdirSync(_path);
//             }
//         }
//     }
// }


/**
 * 读取路径信息
 * @param {string} path 路径
 */
function getStat(path){
    return new Promise((resolve, reject) => {
        fs.stat(path, (err, stats) => {
            if(err){
                resolve(false);
            }else{
                resolve(stats);
            }
        })
    })
}

/**
 * 创建路径
 * @param {string} dir 路径
 */
function mkdir(dir){
    return new Promise((resolve, reject) => {
        fs.mkdir(dir, err => {
            if(err){
                resolve(false);
            }else{
                resolve(true);
            }
        })
    })
}

/**
 * 路径是否存在,不存在则创建
 * @param {string} dir 路径
 */
async function dirExists(dir){
    let isExists = await getStat(dir);
    //如果该路径且不是文件,返回true
    if(isExists && isExists.isDirectory()){
        return true;
    }else if(isExists){     //如果该路径存在但是文件,返回false
        return false;
    }
    //如果该路径不存在
    let tempDir = path.parse(dir).dir;      //拿到上级路径
    //递归判断,如果上级目录也不存在,则会代码会在此处继续循环执行,直到目录存在
    let status = await dirExists(tempDir);
    let mkdirStatus;
    if(status){
        mkdirStatus = await mkdir(dir);
    }
    return mkdirStatus;
}





// var data = { "name": "Alan",
//     "hometown": "Somewhere, TX",
//     "kids": [{"name": "Jimmy", "age": "12"}, {"name": "Sally", "age": "4"}]
// };
//
// var result = htmlHandlebarsComp(data);





// 自定义交互式命令行的问题及简单的校验
let question = [
    {
        name: "projectName",
        type: 'input',
        message: "请输入项目名称(要求英文),方便错误监控识别",
        validate (val) {
            if (val === '') return '项目名称为必填项!'
            if (/\W/.test(val)) return '项目名称要求为英文!'
            return true
        }
    },
    {
        name: "fileName",
        type: 'input',
        message: "请输入html/css/js的文件名",
        validate (val) {
            if (val === '') return 'html/css/js文件名为必填项!'
            return true
        }
    },
    {
        name: "title",
        type: 'input',
        message: "请输入页面标题title",
        validate (val) {
            if (val === '') return '页面标题title为必填项!'
            return true
        }
    },
    {
        name: "apikey",
        type: 'input',
        message: "请输入您的fundebug的apikey",
        validate (apikey) {
            if (apikey === '') {
                return 'apikey为必填项!'
            }else if (apikey.length<64){
                return 'apikey位数小于64!'
            }else {
                return true
            }
        }
    },
    {
        name: "isInjectJQ",
        type: 'confirm',
        message: "是否引入jQuery?"
        // validate (apikey) {
        //     if (apikey === '') {
        //         return 'apikey为必填项!'
        //     }else if (apikey.length<64){
        //         return 'apikey位数小于64!'
        //     }else {
        //         return true
        //     }
        // }
    }

]

inquirer
    .prompt(question).then(answers => {
    // answers 就是用户输入的内容,是个对象
    let {projectName,fileName,title,apikey,isInjectJQ } = answers;

var htmlHandlebarsComp = Handlebars.compile(htmlTemplet);
var packagejsonHandlebarsComp = Handlebars.compile(packagejsonTemplet);

var htmlResult = htmlHandlebarsComp(answers);
var packagejsonResult = packagejsonHandlebarsComp(answers);

// console.log('answers====',answers)
// console.log('htmlTemplet====',htmlTemplet)
// // 过滤 unicode 字符
// htmlTemplet[name] = url.replace(/[\u0000-\u0019]/g, '')
// // 把模板信息写入 template.json 文件中
// fs.writeFile(`${__dirname}/../template.json`, JSON.stringify(htmlTemplet), 'utf-8', err => {
//     if (err) console.log(err)
// console.log('\n')
// console.log(chalk.green('Added successfully!\n'))
// console.log(chalk.grey('The latest template list is: \n'))
// console.log(htmlTemplet)
// console.log('\n')
// })

    async function creatHTMLCssJs(){



// 生成html文件
    async function creatHtml(){
        await dirExists(`${resolve('./')}/workspace`);
        fs.writeFile(`${resolve('./')}/workspace/${fileName}.html`, htmlResult, 'utf8', err => {
            if (err) console.log(err)
            console.log('\n')
            console.log(chalk.green(`在${resolve('./')}/workspace下成功新建了${fileName}.html文件\n`))
// console.log(chalk.grey('请不要移动项目文件哦\n'))

// console.log('\n')
        })

    }
    await creatHtml()
// 生成package.json文件

        async function creatPackagejson(){
            fs.access(`${resolve('./')}/package.json`, fs.constants.F_OK, async (err) => {
                if(err && err.code==='ENOENT'){
                    fs.writeFile(`${resolve('./')}/package.json`, packagejsonResult, 'utf8', err => {
                        if (err) console.log('创建package.json出错了',err)
                        console.log('\n')
                        console.log(chalk.green(`在${resolve('./')}下成功新建了package.json文件\n`))
                    })

                }else if(err){
                    if (err) console.log('访问package.json出错了',err)

                }
            });
        }
        await creatPackagejson()



// 生成css文件
    async function creatCss(){
        await dirExists(`${resolve('./')}/workspace/css`);
        fs.writeFile(`${resolve('./')}/workspace/css/${fileName}.css`,cssTemplet , 'utf8', err => {
            if (err) console.log(err)
            console.log('\n')
            console.log(chalk.green(`在${resolve('./')}/workspace/css下成功新建了${fileName}.css文件\n`))
            // console.log('写入cssTemplet中的__dirname',__dirname)

            // console.log('写入cssTemplet',cssTemplet)
// console.log(chalk.grey('请不要移动项目文件哦\n'))
// console.log('htmlResult')
// console.log('\n')
        })

    }
        await creatCss()




// 生成js文件
    async function creatJS(){
        await dirExists(`${resolve('./')}/workspace/js`);
        fs.writeFile(`${resolve('./')}/workspace/js/${fileName}.js`,jsTemplet , 'utf8', err => {
            if (err) console.log(err)
            console.log('\n')
            console.log(chalk.green(`在${resolve('./')}/workspace/js下成功新建了${fileName}.js文件\n`))
            console.log(chalk.red(`所有文件均在${resolve('./')}下创建完毕,请执行 npm install`))

// console.log(chalk.grey('请不要移动项目文件哦\n'))
// console.log('htmlResult')
// console.log('\n')
        })
    }
        await  creatJS()
        // 具备检测是否存在的拷贝文件功能

        async function copyFileBeforeCheck(fromFileSrc,toFileSrc,toFileSrcSubDir){
            fs.access(toFileSrc, fs.constants.F_OK, async (err) => {
                if(err && err.code==='ENOENT'){
                    // copyFile(`${__dirname}/../gulpfile.js`,`${resolve('./')}/gulpfile.js`)
                    if(toFileSrcSubDir){
                        await dirExists(toFileSrcSubDir);
                    }
                    copyFile(fromFileSrc,toFileSrc)
                }else if (err) {
                    console.log('拷贝'+fromFileSrc+'出错了:',err)
                }
            });



        }
        copyFileBeforeCheck(`${__dirname}/../template/jQuery.v1.12.4.js`,`${resolve('./')}/workspace/js/notchange/jQuery.v1.12.4.js`,`${resolve('./')}/workspace/js/notchange/`)
        copyFileBeforeCheck(`${__dirname}/../gulpfile.js`,`${resolve('./')}/gulpfile.js`)
        // copyFileBeforeCheck(`${__dirname}/../template/package.json`,`${resolve('./')}/package.json`)



        // async function copyJQFile(){
        //     fs.access(`${resolve('./')}/workspace/js/notchange/jQuery.v1.12.4.js`, fs.constants.F_OK, async (err) => {
        //         console.log('拷贝jQuery出错了:',err)
        //         if(err.code==='ENOENT'){
        //             // copyFile(`${__dirname}/../gulpfile.js`,`${resolve('./')}/gulpfile.js`)
        //             await dirExists(`${resolve('./')}/workspace/js/notchange/`);
        //             copyFile(`${__dirname}/../template/jQuery.v1.12.4.js`,`${resolve('./')}/workspace/js/notchange/jQuery.v1.12.4.js`)
        //         }
        //     });
        //
        //
        //
        // }
        // copyJQFile()

        // async function copyGulpFile(){
        //     // copyFile(`${__dirname}/../gulpfile.js`,`${resolve('./')}/gulpfile.js`)
        //     // 检查文件是否存在于当前目录中。
        //     fs.access(`${resolve('./')}/gulpfile.js`, fs.constants.F_OK, (err) => {
        //         // console.log('拷贝gulp出错了:',err)
        //       if(err.code==='ENOENT'){
        //           copyFile(`${__dirname}/../gulpfile.js`,`${resolve('./')}/gulpfile.js`)
        //       }
        //     });
        //
        // }
        // copyGulpFile()



    }

    creatHTMLCssJs()




// // 写入文件
// var  writerStream = fs.createWriteStream(`${fileName}.html`);
// writerStream.write(htmlResult.toString(), 'UTF8');
// writerStream.end();

})

// 输入fundebug apikey
// 输入项目名称
// 输入文件名名称
// 输入网页标题




4、在package.json里写入如下信息,以便使用npm link后再运行 init 即可代替运行node ./bin/xr指令;

  "bin": {
    "init": "bin/init"
  }

或者使用 npm install -g也是一样的效果

5、新建如下目录或文件


image.png

6、doNotChangeIt.html 如下:

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0,user-scalable=no"
          name="viewport">
    <title>{{title}}</title>
     <!--引入fundebug。  -->
    <script src="https://js.fundebug.cn/fundebug.2.0.0.min.js" crossorigin="anonymous"></script>
    <script>
        // fundebug初始化
        if ("fundebug" in window) {
            fundebug.init({
                apikey: '{{apikey}}',
                // silentDev: true, // 开发环境不启用fundebug
                silentConsole: true,
                setHttpBody: true
            });
        }
    </script>
     <!--引入资源及api异常监控插件-->
    <script src="https://h5.4417.com/ios/js/notchange/monitorFileApiError.js?name={{projectName}}&allWrapId=allWrap"
            data-attr="monitor-file-api-error"></script>
    <!--说明
    1、data-attr="monitor-file-api-error" 请勿修改或遗漏 --必填
    2、name表示你的项目名称 --必填
    3、allWrapId表示如果接口异常(网络错误)或资源加载失败时,你想在html上的id为allWrap的原生元素上 innerText 相应的文字 --非必填
    4、notMonitorDiffSiteResource 有值时表示 不监听跨域资源异常并刷新页面,默认监听跨域资源异常并刷新页面 --非必填
    5、notMonitorResource 有值时 表示监听到资源异常时(含同源及跨域资源)也不会刷新页面,默认监听到资源异常时会进行刷新页面 ---非必填
    5、errNoNotAlert 表示errno 不为0时是否进行alert弹窗,默认弹窗---非必填
    -->

    <!--引入反馈网页问题样式-->
    <link rel="stylesheet" href="https://h5.4417.com/ios/css/reprot-response-width-lr-771a85cc8d.css">
    <!--在以下的css文件中编写你的css-->
    <link rel="stylesheet" href="./css/{{fileName}}.css">


</head>
<body>
<div id="allWrap">
    欢迎使用模板
</div>
</body>
{{#if isInjectJQ}}
<script src="js/notchange/jQuery.v1.12.4.js"></script>
{{/if}}
<!--在以下的js文件中编写你的js-->
<script src="./js/{{fileName}}.js"></script>
 <!--引入反馈网页问题js-->
<script src="https://h5.4417.com/ios/js/notchange/reportself.js?name={{projectName}}&apikey={{apikey}}&notStorageKeys=storage1,storage1&right=true"  data-attr="licai-reportself"></script>
<!--说明
1、data-attr="licai-reportself" 请勿修改或遗漏 --必填
2、name表示你的项目名称 --必填
3、notStorageKeys表示你不想传递给后台的localStorage及sessionStorage的key值,支持多个,中间用英文, 分隔 --非必填
4、right 有值表示 网页问题反馈标志在右边 否则在左边 -- 非必填
-->



</html>


7、package.json文件如下

{
  "name": "{{projectName}}",
  "version": "1.0.0",
  "main": "gulpfile.js",
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1",
    "dev": "gulp"
  },
  "repository": {
    "type": "git",
    "url": ""
  },
  "author": "",
  "license": "ISC",
  "dependencies": {
    "@babel/runtime": "~7.4.5",
    "babel-core": "~6.26.3",
    "babel-plugin-transform-runtime": "~6.23.0",
    "babel-polyfill": "~6.26.0",
    "babel-preset-env": "~1.7.0",
    "babel-preset-es2015": "~6.24.1",
    "browser-sync": "~2.26.7",
    "fix-orientation": "0.0.2",
    "gulp": "~3.9.1",
    "gulp-htmlmin": "~5.0.1",
    "gulp-jshint": "~2.0.2",
    "gulp-livereload": "~3.8.1",
    "gulp-minify-css": "~1.2.4",
    "gulp-notify": "~2.2.0",
    "gulp-open": "~3.0.1",
    "gulp-postcss": "~7.0.1",
    "gulp-rename": "~1.2.2",
    "gulp-sourcemaps": "~2.6.5",
    "gulp-uglify": "~3.0.1",
    "postcss-px2rem": "~0.3.0",
    "pump": "^3.0.0"
  },
  "devDependencies": {
    "@babel/core": "~7.4.5",
    "@babel/plugin-transform-runtime": "~7.4.4",
    "@babel/preset-env": "~7.4.5",
    "gulp-asset-rev": "0.0.15",
    "gulp-autoprefixer": "~5.0.0",
    "gulp-babel": "~8.0.0",
    "gulp-clean": "~0.3.2",
    "gulp-clean-css": "~2.0.13",
    "gulp-concat": "~2.6.0",
    "gulp-connect": "~5.7.0",
    "gulp-ejs": "~3.2.0",
    "gulp-file": "~0.4.0",
    "gulp-file-include": "~2.0.1",
    "gulp-imagemin": "~4.1.0",
    "gulp-inject": "~5.0.2",
    "gulp-jshint": "~2.0.2",
    "gulp-livereload": "~3.8.1",
    "gulp-minify-css": "~1.2.4",
    "gulp-notify": "~2.2.0",
    "gulp-plumber": "~1.2.1",
    "gulp-rename": "~1.2.2",
    "gulp-rev": "~8.1.0",
    "gulp-rev-collector": "~1.2.3",
    "gulp-watch": "~5.0.1",
    "jshint": "~2.9.4",
    "run-sequence": "~2.2.0"
  },
  "description": ""
}

8、gulpfile文件如下

var gulp = require('gulp');
var plumber = require('gulp-plumber'); // 实时更新错误不会导致终端gulp运行开启的服务断开
var connect = require('gulp-connect'); // 在本地开启一个websocket服务,使用liveReload实现实时更新
var watch = require('gulp-watch'); // 监听文件的变化,配合connect实现服务自动刷新
var gulpOpen = require('gulp-open');//用指定软件打开文件


var browserSync = require('browser-sync').create();
var postcss = require('gulp-postcss');
var px2rem = require('postcss-px2rem');

var autoprefixer = require('gulp-autoprefixer');
var fileinclude = require('gulp-file-include');
var ejs = require('gulp-ejs');

var os = require('os');
// var path = require('path');

var ipv4IpLocal = function getLoaclIP() {
    var ipObj = os.networkInterfaces();
    var IPv4 = [];
    Object.keys(ipObj).forEach(function (ele) {
        ipObj[ele].forEach(function (ip) {
            if (ip.family === 'IPv4') {
                IPv4.push(ip.address);
            }
        });
    });
    return IPv4[0];
};



// 实现静态资源版本更新与缓存==================================================
var runSequence = require('run-sequence'),
    rev = require('gulp-rev'),
    revCollector = require('gulp-rev-collector');
// assetRev = require('gulp-asset-rev');


//  压缩相关插件 ========================================
var cleanCSS = require('gulp-clean-css'),//css压缩
    uglify = require('gulp-uglify'),//js压缩
    concat = require("gulp-concat"), //合并文件
    htmlmin = require('gulp-htmlmin'),//html压缩
    // jshint=require("gulp-jshint"), //代码规范检查
    imagemin = require("gulp-imagemin"), //图片压缩
    rename = require("gulp-rename"),//重命名
    clean = require("gulp-clean");//清空文件夹
var pump = require('pump');


var babel = require("gulp-babel");
var sourcemaps = require('gulp-sourcemaps');
var file = require('gulp-file');
var inject = require('gulp-inject');

var workSpaceDir='./workspace';

var distDir='./workspace/dist';


// function getDateAndTime(dateObj) {
//     var nowDate = dateObj;
//     var y = (nowDate.getFullYear()).toString().slice(2,4);
//     var m = nowDate.getMonth() + 1;
//     m = m < 10 ?  ('0'+m) : m;
//     var d = nowDate.getDate();
//     d = d < 10 ? ('0'+d) : d;
//     var h = nowDate.getHours();
//     h = h < 10 ? ('0'+h) : h;
//     var min = nowDate.getMinutes();
//     min = min < 10 ? ('0'+min) : min;
//     var s = nowDate.getSeconds();
//     s = s < 10 ? ('0'+s) : s;
//     return  y+m+d+h+min+s
//     // return `${y}${m}${d}${h}${min}${s}`;
//
// }

// var curDateAndTime=getDateAndTime( new Date());


gulp.task('cssREM', function () {
    var processors = [px2rem({remUnit: 90.4})];
    var stream = gulp.src(workSpaceDir+'/css/*.css')
        .pipe(postcss(processors))
        .on('error', function (err) {
            console.error(' css rem处理出错了'+err)
        }).pipe(gulp.dest(workSpaceDir+'/css/rem'));
    console.log(' css rem在执行中 stream ' + stream)

    return stream;
});
gulp.task('fileinclude', function () {
    // 适配page中所有文件夹下的所有html,排除page下的include文件夹中html
    gulp.src(['page/**/*.html', '!page/include/**.html'])
        .pipe(fileinclude({
            prefix: '@@',
            basepath: '@file'
        }))
        .pipe(gulp.dest('dist0'));
});
gulp.task('ejs', function () {
    gulp.src('ejs/**.ejs')
        .pipe(ejs())
        .pipe(gulp.dest('dist_ejs'));
});








// gulp.task('html', function () {
//     return gulp.src("./workspace/*.html")
//         .pipe(htmlmin({collapseWhitespace: true}))
//         .pipe(gulp.dest("./dist/"));
// });
//
// gulp.task('cssjr', function () {
//     console.log('开始执行css兼容程序')
//     // https://www.gulpjs.com.cn/docs/api/#gulp.task
//     // cssjr 是css 任务的依赖 因此它要通知css它什么时候执行完了
//     // 有三种方式 通知它执行完了 1、回调 2、返回数据流stream  3、返回promise
//     var stream = gulp.src('./workspace/css/*.css')
//         .pipe(autoprefixer({
//             browsers: ['since 2010'],
//             cascade: false
//         }))
//         // .pipe(rename({
//         //     suffix: ".jr"
//         // }))
//         .pipe(gulp.dest('./dist/css/jianrong'));
//     console.log(' css兼容程序在执行中 stream ' + stream)
//     return stream;
//
//
// });
//
// gulp.task('css', ['cssjr'], function () {
//     // 执行该css任务的前提是 执行完了cssjr (cssjr 是css 的依赖)
//     //
//     console.log('开始执行css压缩')
//     return gulp.src("./dist/css/jianrong/*.css")
//         .pipe(cleanCSS({compatibility: 'ie8'}))
//         // .pipe(rename({
//         //     suffix: ".min."+curDateAndTime
//         // }))
//         .pipe(gulp.dest('./dist/css/'));
// });
//
//
// gulp.task('js', function (cb) {
//     console.log('开始压缩js');
//
//     pump([
//             gulp.src('./workspace/js/*.js'),
//             uglify(),
//             // rename({suffix: '.min.'+curDateAndTime}),
//             gulp.dest('./dist/js')
//         ],
//         cb
//     );
// });
//
// gulp.task('img', function () {
//     console.log('开始压缩图片');
//     return gulp.src('./workspace/img/**/*')
//             .pipe(imagemin())
//             .pipe(gulp.dest('./dist/img'))
//     }
// );
//
//
// gulp.task("clean", function(){
//     console.log('开始清空dist文件夹 --不清空img ');
//     return gulp.src(['./dist/*.html','./dist/css/*','./dist/js/*.js'])
//         .pipe(clean());
// });
//
// gulp.task("cleanIncludeImg", function(){
//     console.log('开始清空dist文件夹 --清空img');
//     return gulp.src(['./dist/*.html','./dist/css/*','./dist/js/*.js','./dist/img/*'])
//         .pipe(clean());
// });
//
//
// // 压缩(不压缩图片)但不清空
// gulp.task('jrmin', ['html', 'css', 'js']);
//
// // 清空并压缩 (不含图片)
// gulp.task('cleanAndMin', ['clean'],function(){
//     gulp.start('html', 'css', 'js')
// });
//
//
// // 清空并压缩 (含图片)
// gulp.task('min', ['cleanIncludeImg'],function(){
//     gulp.start('html', 'css', 'js','img')
// });






// //定义css、js源文件路径
// var cssSrc = './dist/css/*.css',  //dist下的所有css文件
//     jsSrc = './dist/js/*.js';  //dist下的所有js文件
//
//
// //为css中引入的图片/字体等添加hash编码
// // gulp.task('assetRev', function(){
// //     return gulp.src(cssSrc)   //该任务针对的文件
// //         .pipe(assetRev())  //该任务调用的模块
// //         .pipe(gulp.dest('src/css')); //编译后的路径
// // });
//
// //CSS生成文件hash编码并生成 rev-manifest.json文件名对照映射
// gulp.task('revCss', function(){
//     console.log('revCss执行了')
//     return gulp.src(cssSrc)
//         .pipe(rev())
//         .pipe(gulp.dest('revDist/css'))
//         .pipe(rev.manifest())
//         .pipe(gulp.dest('revDist/css'));
//
// });
//
// //js生成文件hash编码并生成 rev-manifest.json文件名对照映射
// gulp.task('revJs', function(){
//     return gulp.src(jsSrc)
//         .pipe(rev())
//         .pipe(gulp.dest('revDist/js'))
//         .pipe(rev.manifest())
//         .pipe(gulp.dest('revDist/js'));
// });
//
// //Html替换css、js文件版本
// gulp.task('revHtml', function () {
//     return gulp.src(['revDist/**/*.json', 'dist/*.html'])
//         .pipe(revCollector())
//         .pipe(gulp.dest('revDist'));
// });
//
// //开发构建  代码压缩后 加hash  注意img文件夹和js/notchange 文件夹中的文件均不会进行更新!!!
// gulp.task('cacheCtr', function (done) {
//     condition = false;
//     runSequence(
//         // ['assetRev'],
//         ['revCss'],
//         ['revJs'],
//         ['revHtml'],
//         done);
// });


// gulp gulp1 执行前应该先执行 default
// gulp.task('gulp1', ['cacheCtr']);


// // E6语法转换相关代码
//
// var babel = require("gulp-babel");
// var sourcemaps = require('gulp-sourcemaps');
//
// gulp.task('babelTest', function () {
//     // 1、插入polyfill文件到html并测试 2、 实现类似于 browser-sync 功能 todo
//     // var polyfill = './node_modules/babel-polyfill/dist/polyfill.js';
//     // return gulp.src([polyfill,'es6/*.js'])
//     //     .pipe(sourcemaps.init())
//     return gulp.src('es6/*.js')
//            .pipe(sourcemaps.init())
//         // .pipe(babel({presets: ['es2015']}))
//         // .pipe(concat('all.min.js'))
//         // .pipe(babel())
//         .on('error', function (err) {
//            console.error('sourcemaps相关程序出错了'+err)
//         })
//         // .pipe(babel())
//         .pipe(babel({
//             presets: ['@babel/env']
//             // ,plugins: ['@babel/transform-runtime'] // 如果启用这个 则会在js中 采用require语法引用的方式 因此不适合
//             // 另外 babel-runtime 可以部分取代babel-polyfill功能 但是像 [1,2,3].includes(1)实例方法 则不可替代
//         }))
//         .on('error', function (err) {
//            console.error('babel({\n' +
//                '            presets: [\'@babel/env\']\n' +
//                '        })出错了'+err)
//         })
//         .pipe(sourcemaps.write('.'))
//         // .pipe(uglify())
//         .pipe(gulp.dest('es6/dist/'))
//         .on('error', function (err) {
//             console.error('babelTest出错了'+err)
//         })
// });


// 一、gulp 启动服务并监测资源变化并自动刷新

// (一)、方法一:不使用browserSync的方法 缺点 不能启用两个
// var workSpaceDir='./es6';
// gulp.task('connectAllFile', function(){
//     gulp.src(workSpaceDir+'/**/*')
//         .pipe(plumber({
//             errorHandler: function(error) {
//                 this.error('end');
//                 console.error('监听所有文件出错了'+error)
//             }
//         }))
//         // .pipe(gulp.dest('dist/js'))
//         .pipe(connect.reload())
//     console.log('connectAllFile')
//
// })
// // gulp.task('ant_html', function(){
// //     gulp.src('index.html')
// //         .pipe(rev())
// //         .pipe(gulp.dest('dist/html'))
// //         .pipe(connect.reload())
// // })
// //选取谷歌浏览器
// var browser = os.platform() === 'linux' ? 'google-chrome' : (
//     os.platform() === 'darwin' ? 'google chrome' : (
//         os.platform() === 'win32' ? 'chrome' : 'firefox'));
//
// gulp.task('startServerAndOpenBrower', function() {
//     connect.server({
//         livereload: true,
//         port: 8888,
//         host:ipv4IpLocal()
//     })
//     gulp.src(__filename)
//         .pipe(gulpOpen({
//             uri: ipv4IpLocal()+':8888/'+workSpaceDir.slice(2),
//             app: browser
//         }));
//     console.log('startServerAndOpenBrower执行了')
// })
// gulp.task('watchAllFile', function() {
//     gulp.watch(workSpaceDir+'/**/*', ['connectAllFile'])
//     // gulp.watch('src/js/*.js', ['js'])
//     console.log('watchAllFile执行了')
//
// })
//
// gulp.task('default', [ 'connectAllFile', 'watchAllFile', 'startServerAndOpenBrower'])

// (二)、方法二:使用browserSync的方式

gulp.task('default', function() {
    browserSync.init({
        server: {
            baseDir: workSpaceDir
        },
        open:'external',
        files:[workSpaceDir+'/*.html',workSpaceDir+'/css/*.css',workSpaceDir+'/js/*.js'],
        ghostMode: false // 禁止 点击,滚动和表单在任何设备上输入将被镜像到所有设备里
    });
});



// 二、gulp build  依次实现 1、babel语法转换 2、加hash 3、压缩混淆 4、启动build后服务



// js sourcemap和babel及hash值
gulp.task('sourcemapBabelHashJs', function () {
    console.log('开始压缩js');
    // uglify 暂时不加
    return  gulp.src(workSpaceDir+'/js/*.js')
        .pipe(sourcemaps.init())
        .on('error', function (err) {
            console.error('sourcemaps相关程序出错了'+err)
        })
        .pipe(babel({
            presets: ['@babel/env']
            // ,plugins: ['@babel/transform-runtime'] // 如果启用这个 则会在js中 采用require语法引用的方式 因此不适合
            // 另外 babel-runtime 可以部分取代babel-polyfill功能 但是像 [1,2,3].includes(1)实例方法 则不可替代
        }))
        .on('error', function (err) {
            console.error('babel({\n' +
                '            presets: [\'@babel/env\']\n' +
                '        })出错了'+err)
        })
        .pipe(sourcemaps.write('.'))
        .on('error', function (err) {
            console.error('sourcemaps.write出错了'+err)
        })
        .pipe(rev())
        .on('error', function (err) {
            console.error(' js加hash值出错了 pipe(rev())'+err)
        })
        .pipe(gulp.dest(distDir+'/js'))
        .pipe(rev.manifest())
        .on('error', function (err) {
            console.error(' rev.manifest出错了'+err)
        })
        // .pipe(uglify())
        // .on('error', function (err) {
        //     console.error(' js 压缩混淆出错了 pipe(uglify())'+err)
        // })
        .pipe(gulp.dest(distDir+'/js'));


    // pump([
    //         gulp.src(workSpaceDir+'/js/*.js'),
    //          sourcemaps.init(),
    //         uglify(),
    //         // rename({suffix: '.min.'+curDateAndTime}),
    //         gulp.dest('./dist/js')
    //     ],
    //     cb
    // );
});

// js 压缩
// 其执行的前提是sourcemapBabelHashJs先执行完了
gulp.task('sourcemapBabelHashJsThenUglifyJs', ['sourcemapBabelHashJs'], function () {
    console.log('开始执行js压缩,不过执行前 要先执行sourcemapBabelHashJs')
    return gulp.src(distDir+"/js/*.js")
        .pipe(uglify())
        .on('error', function (err) {
            console.error(' js 压缩混淆出错了 pipe(uglify())'+err)
        })
        .pipe(gulp.dest(distDir+'/js/'));

})

gulp.task('remThenJrAndHashCss', ['cssREM'], function () {

    console.log('开始执行css兼容程序')
    // https://www.gulpjs.com.cn/docs/api/#gulp.task
    // cssjr 是css 任务的依赖 因此它要通知css它什么时候执行完了
    // 有三种方式 通知它执行完了 1、回调 2、返回数据流stream  3、返回promise
    var stream = gulp.src(workSpaceDir+'/css/rem/*.css')
        .pipe(autoprefixer({
            browsers: ['since 2010'],
            cascade: false
        }))
        .on('error', function (err) {
            console.error(' css兼容处理出错了 autoprefixer'+err)
        })
        .pipe(rev())
        .on('error', function (err) {
            console.error(' css hash处理出错了 .pipe(rev())'+err)
        })
        .pipe(gulp.dest(distDir+'/css/jianrong'))
        .pipe(rev.manifest())
        .on('error', function (err) {
            console.error(' css hash manifest.json处理出错了 rev.manifest'+err)
        })

        // .pipe(rename({
        //     suffix: ".jr"
        // }))
        .pipe(gulp.dest(distDir+'/css/'));
    console.log(' css兼容程序在执行中 stream ' + stream)
    return stream;

})

// css 加前缀及hash
gulp.task('jrAndHashCss', function () {
    console.log('开始执行css兼容程序')
    // https://www.gulpjs.com.cn/docs/api/#gulp.task
    // cssjr 是css 任务的依赖 因此它要通知css它什么时候执行完了
    // 有三种方式 通知它执行完了 1、回调 2、返回数据流stream  3、返回promise
    var stream = gulp.src(workSpaceDir+'/css/*.css')
        .pipe(autoprefixer({
            browsers: ['since 2010'],
            cascade: false
        }))
        .on('error', function (err) {
            console.error(' css兼容处理出错了 autoprefixer'+err)
        })
        .pipe(rev())
        .on('error', function (err) {
            console.error(' css hash处理出错了 .pipe(rev())'+err)
        })
        .pipe(gulp.dest(distDir+'/css/jianrong'))
        .pipe(rev.manifest())
        .on('error', function (err) {
            console.error(' css hash manifest.json处理出错了 rev.manifest'+err)
        })

        // .pipe(rename({
        //     suffix: ".jr"
        // }))
        .pipe(gulp.dest(distDir+'/css/'));
    console.log(' css兼容程序在执行中 stream ' + stream)
    return stream;


});


// css 压缩
gulp.task('jrAndHashCssThenCompressionCss', ['jrAndHashCss'], function () {
    // 执行该css任务的前提是 执行完了jrAndHashCss (jrAndHashCss是css 的依赖)
    //
    console.log('开始执行css压缩')
    return gulp.src(distDir+"/css/jianrong/*.css")
        .pipe(cleanCSS({compatibility: 'ie8'}))
        .on('error', function (err) {
            console.error(' css压缩处理出错了 cleanCSS'+err)
        })
        // .pipe(rename({
        //     suffix: ".min."+curDateAndTime
        // }))
        .pipe(gulp.dest(distDir+'/css/'));
});
gulp.task('jrAndHashCssThenCompressionCssREM', ['remThenJrAndHashCss'], function () {
    // 执行该css任务的前提是 执行完了jrAndHashCss (jrAndHashCss是css 的依赖)
    //
    console.log('开始执行css压缩')
    return gulp.src(distDir+"/css/jianrong/*.css")
        .pipe(cleanCSS({compatibility: 'ie8'}))
        .on('error', function (err) {
            console.error(' css压缩处理出错了 cleanCSS'+err)
        })
        // .pipe(rename({
        //     suffix: ".min."+curDateAndTime
        // }))
        .pipe(gulp.dest(distDir+'/css/'));
});


// html hash
gulp.task('compressionHtmlThenHashHtml',['compressionHtml'], function () {
    return gulp.src([ distDir+'/**/*.json', distDir+'/*.html'])
        .pipe(revCollector())
        .pipe(gulp.dest(distDir));
});

// html 压缩
gulp.task('compressionHtml', function () {
    return gulp.src(workSpaceDir+"/*.html")
        .pipe(htmlmin({collapseWhitespace: true}))
        .pipe(gulp.dest(distDir));
});

// 清空文件夹
gulp.task("cleanHTMLCssJsImg", function(){
    console.log('开始清空dist文件夹');
    return gulp.src([distDir+'/*.html',distDir+'/css/*',distDir+'/js/*',distDir+'/img/*'])
        .pipe(clean());
});

// 在<head><meta 之间 注入 polyfill.min.js

gulp.task('injectJS',['copyNotchangeJsAndBabelPolyfill'], function () {
    // var target = gulp.src(distDir+'/*.html');
    // // It's not necessary to read the files (will speed up things), we're only after their paths:
    // // var sources = gulp.src(['./src/**/*.js', './src/**/*.css'], {read: false});
    // var sources = gulp.src([distDir+'/js/notchange/polyfill.min.js'], {read: false},{relative: true});
    // // console.log('sources',sources)
    //
    // return target.pipe(inject(sources))
    //         .on('error', function (err) {
    //            console.error('injectJS相关程序出错了--target.pipe(inject(sources))'+err)
    //         })
    //     .pipe(gulp.dest(distDir));

    // 1、注入所有js最前面 2、src不需要es6/dist todo
    return  gulp.src(distDir+'/*.html')
        // .pipe(file('es6/dist/index.html', '<html><head></head></html>'))
        //    .on('error', function (err) {
        //       console.error('injectJS相关程序出错了--file(distDir+\'/index.html\''+err)
        //    })
        .pipe(inject(gulp.src([distDir+'/js/notchange/polyfill.min.js']), {
            starttag: '<head>',
            endtag: '<meta',
            relative: true
        })).on('error', function (err) {
            console.error('injectJS相关程序出错了--pipe(inject(gulp.src([distDir'+err)
        })
        .pipe(gulp.dest(distDir));
});


// 1、从node_modules/babel-polyfill/dist中拷贝出 babel-polyfill 文件
// 2、拷贝出workspace 中js/notchange 中所有js 到 dist js/notchange中
gulp.task('copyNotchangeJsAndBabelPolyfill',['cleanNotchangeJs'], function () {
        console.log('开始导出js/notchange 中的所有文件');
        var polyfill = './node_modules/babel-polyfill/dist/polyfill.min.js';

        return gulp.src([workSpaceDir+'/js/notchange/*.js',polyfill])
            .pipe(gulp.dest(distDir+'/js/notchange'))
    }
);

// gulp.task('exportImg', function () {
//         console.log('开始导出img 中的所有文件');
//         return gulp.src(workSpaceDir+'/img/**/*')
//             .pipe(gulp.dest(distDir+'/img'))
//     }
// );

// 清空 dist/img 文件夹
gulp.task("cleanNotchangeImg", function(){
    console.log('开始清空img文件夹 ');
    return gulp.src([distDir+'/img'])
        .pipe(clean());
});

// 清空 dist/js/notchange
gulp.task("cleanNotchangeJs", function(){
    console.log('开始清空js/notchange文件夹');
    return gulp.src([distDir+'/js/notchange'])
        .pipe(clean());
});

// 拷贝  workSpace img文件夹 中的文件 至 dist img文件夹下
gulp.task('copyNotchangeImg', ['cleanNotchangeImg'],function(){
    // gulp.start('exportImg')
    console.log('开始导出img 中的所有文件');
    return gulp.src(workSpaceDir+'/img/**/*')
        .pipe(gulp.dest(distDir+'/img'))
});

// 压缩  workSpace img文件夹 中的文件 至 dist img文件夹下
gulp.task('minImg', ['cleanNotchangeImg'],function(){
    // gulp.start('exportImg')
    console.log('开始压缩图片');
    return gulp.src(workSpaceDir+'/img/**/*')
        .pipe(imagemin())
        .pipe(gulp.dest(distDir+'/img'))
});


// 在dist上启动服务
gulp.task('showDistAllFile', function() {
    browserSync.init({
        server: {
            baseDir: distDir
        },
        open:'external',
        files:['*','./**/*'],
        ghostMode: false // 禁止 点击,滚动和表单在任何设备上输入将被镜像到所有设备里

    });
});


// 完成一系列流程
// 不压缩图片 不注入Polyfill

gulp.task('build', ['cleanHTMLCssJsImg'],function (done) {
    condition = false;
    runSequence(
        // ['assetRev'],
        ['sourcemapBabelHashJsThenUglifyJs'],
        ['jrAndHashCssThenCompressionCss'],
        ['compressionHtmlThenHashHtml'],
        ['copyNotchangeImg'],
        ['copyNotchangeJsAndBabelPolyfill'],
        ['showDistAllFile'],
        done);
});
// 不压缩图片 注入Polyfill
gulp.task('buildES', ['cleanHTMLCssJsImg'],function (done) {
    condition = false;
    runSequence(
        // ['assetRev'],
        ['sourcemapBabelHashJsThenUglifyJs'],
        ['jrAndHashCssThenCompressionCss'],
        ['compressionHtmlThenHashHtml'],
        ['copyNotchangeImg'],
        ['injectJS'],
        ['showDistAllFile'],
        done);
});

// 不压缩图片 不注入Polyfill 不打开browser-sync  jenkins专用

gulp.task('buildJK', ['cleanHTMLCssJsImg'],function (done) {
    condition = false;
    runSequence(
        // ['assetRev'],
        ['sourcemapBabelHashJsThenUglifyJs'],
        ['jrAndHashCssThenCompressionCss'],
        ['compressionHtmlThenHashHtml'],
        ['copyNotchangeImg'],
        ['copyNotchangeJsAndBabelPolyfill'],
        done);
});

// 完成一系列流程
// 压缩图片 注入Polyfill 打开browser-sync

gulp.task('build0', ['cleanHTMLCssJsImg'],function (done) {
    condition = false;
    runSequence(
        // ['assetRev'],
        ['sourcemapBabelHashJsThenUglifyJs'],
        ['jrAndHashCssThenCompressionCss'],
        ['compressionHtmlThenHashHtml'],
        ['minImg'],
        ['injectJS'],
        ['showDistAllFile'],
        done);
});

// rem适配方案 不压缩图片 不注入Polyfill
gulp.task('buildrem', ['cleanHTMLCssJsImg'],function (done) {
    condition = false;
    runSequence(
        // ['assetRev'],
        ['sourcemapBabelHashJsThenUglifyJs'],
        ['jrAndHashCssThenCompressionCssREM'],
        ['compressionHtmlThenHashHtml'],
        ['copyNotchangeImg'],
        ['copyNotchangeJsAndBabelPolyfill'],
        ['showDistAllFile'],
        done);
});

// rem适配方案 不压缩图片 不注入Polyfill 不打开browser-sync  jenkins专用

gulp.task('buildremJK', ['cleanHTMLCssJsImg'],function (done) {
    condition = false;
    runSequence(
        // ['assetRev'],
        ['sourcemapBabelHashJsThenUglifyJs'],
        ['jrAndHashCssThenCompressionCssREM'],
        ['compressionHtmlThenHashHtml'],
        ['copyNotchangeImg'],
        ['copyNotchangeJsAndBabelPolyfill'],
        done);
});

// // 压缩所有图片
// gulp.task('imgMin', function () {
//         console.log('开始压缩图片');
//         return gulp.src(workSpaceDir+'/img/**/*')
//             .pipe(imagemin())
//             .pipe(gulp.dest(distDir+'/img'))
//     }
// );

补充

模版管理
一、和脚手架耦合(@umijs/create-umi-app)
二、基于github或者gitlab(cra、vue-cliinit)
三、基于npm(cra、react-native、tuyaworks)

模版解析
一、metalsmith、handlebars、consolidate实现
二、node-plob小而美的脚手架工具
三、基于handlebars、recursive-readdir和glob自己实现

参考资料:

1、仿vue-cli搭建属于自己的脚手架的方法步骤
2、前端工程化之plop的使用
3、https://my.oschina.net/u/3347851/blog/5146718

相关文章

  • 使用命令行创建前端多页面模版(制作脚手架)

    一、前置知识 1、chalk 可以使用它来实现各种命令行字体的颜色; 2、inquirer 可以使用它在命令行实现...

  • Vue.js项目的搭建

    使用Vue框架编写前端页面步骤 1.安装vue-cli脚手架工具 打开命令行,输入指令 npm install -...

  • Vue.js项目的搭建

    使用Vue框架编写前端页面步骤 1.安装vue-cli脚手架工具 打开命令行,输入指令 npm install -...

  • vue多页应用配置

    vue脚手架创建的项目都是单页面应用但是有一些场景下我们需要使用多页面应用(比如一个系统的后端和前端要分为不同的页...

  • Vue CLI3开发多页面应用

    简要说明 使用vue脚手架创建的vue项目均为单页面应用。但是有时候我们也需要多页面应用,那该如何使用cli来配置...

  • vue iview 使用经验

    入门 Vue 是响应式前端框架,所谓响应式指的是变量值有变化时,页面跟随变化。可以使用vue 脚手架创建一个vue...

  • 1-4 使用命令行创建项目页面

    使用命令行创建项目页面 如果你熟悉命令行使用Git,这是简单的手工创建项目页面的网站。 作出新的克隆要建立一个项目...

  • MK前端架构师-init模块技术方案

    脚手架项目创建功能架构设计图 准备阶段 模版的下载(复杂) 模版的安装

  • egg.js+html+上传图片

    前端页面 前端页面如果需要预览与裁剪或更多的功能可自行添加 后端准备工作 我们推荐直接使用脚手架,只需几条简单指令...

  • 使用React脚手架创建一个react项目

    使用脚手架创建项目 打开命令行窗口 执行命令 npx create-react-app react-basic 说...

网友评论

      本文标题:使用命令行创建前端多页面模版(制作脚手架)

      本文链接:https://www.haomeiwen.com/subject/nfpedktx.html