在实际工作中我们已经下下来不下于一万个npm包了,像我们熟悉的 vue-clireact-native-cli 等,只需要输入简单的命令 vue init webpack project,即可快速帮我们生成一个初始项目。在实际开发项目中,我们也可以定制一个属于自己的npm包,来提高自己的工作效率。

为什么要开发一个工具包?

  • 减少重复性的工作,不再需要复制其他项目再删除无关代码,或者从零创建一个项目和文件。

  • 根据交互动态生成项目结构和所需要的文件等。

  • 减少人工检查的成本。

  • 提高工作效率,解放生产力。

这次以帧动画工具为例,来一步一步解析如何开发一个npm包。

开始前的准备

以我们这次为例。由于目前在做一些活动页相关的工作,其中动画部分全都采用CSS3中的animation来完成,但是这样每次开发都要计算百分比,手动判断动画的一些属性值,十分耗时又很容易出错,就想能不能写个脚本,直接一行命令就可以搞定了呢?!答案当然是肯定的。

理清思路

我们要做一个可以通过读取图片就可以自动生成包含CSS animationHTML页面,以后需要生成相应的CSS片段,直接执行命令就可以了。

初始化

既然是npm包,那我们就需要在npmjs上注册一个账号,注册完成之后回到本地新建一个文件目录fbf,进入fbf目录下执行npm init -y。

  1. {
  2. "name": "fbf",
  3. "version": "1.0.0",
  4. "description": "",
  5. "main": "index.js",
  6. "scripts": {
  7. "test": "echo \"Error: no test specified\" && exit 1"
  8. },
  9. "bin": {
  10. "test": "index.js"
  11. },
  12. "keywords": [],
  13. "author": "",
  14. "license": "ISC"
  15. }

  这样,你的package.json就建好了。

依赖的库

来看看会用到哪些库。

  • commander.js,可以自动的解析命令和参数,用于处理用户输入的命令。

  • chalk,可以给终端的字体加上颜色。

  • create-html,创建HTML模版,用于生成HTML

  • image-size,获取图片大小。

npm install commander chalk create-html image-size -S

命令行操作

node.js 内置了对命令行操作的支持,在 package.json 中的 bin 字段可以定义命令名和关联的执行文件。所以现在 package.json 中加上 bin 的内容:

  1. {
  2. "name": "fbf",
  3. "version": "1.0.0",
  4. "description": "",
  5. "bin": {
  6. "fbf": "index.js"
  7. },
  8. ...
  9. }

然后在 index.js 中来定义 start 命令:

  1. #!/usr/bin/env node
  2. const program = require('commander');
  3.  
  4. program.version('1.0.0', '-v, --version')
  5. .command('start <name>')
  6. .action((name) => {
  7. console.log(name);
  8. });
  9. program.parse(process.argv);

调用 version('1.0.0', '-v, --version') 会将 -v 和 --version 添加到命令中,可以通过这些选项打印出版本号。

调用 command('start <name>') 定义 start 命令,name 则是必传的参数,为文件名。

action() 则是执行 start 命令会发生的行为,要生成项目的过程就是在这里面执行的,这里暂时只打印出 name

其实到这里,已经可以执行 start 命令了。我们来测试一下,在 fbf 的同级目录下执行:

node ./test/index.js start HelloWorld

可以看到命令行工具也打印出了 HelloWorld,那么很清楚, action((name) => {}) 这里的参数 name,就是我们执行 start 命令时输入的项目名称。

命令已经完成,接下来就要针对图片的操作了。

获取图片信息

这里我们默认根据第一张图片的尺寸信息作为外层DIV的默认尺寸。

  1. #!/usr/bin/env node
  2. const program = require('commander');
  3. const fs = require('fs');
  4. const path = require('path');
  5. const createHTML = require('create-html');
  6. const sizeOf = require('image-size');
  7. const chalk = require('chalk');
  8.  
  9. program.version('1.0.0', '-v, --version')
  10. .command('start <dir>')
  11. .action((dir) => {
  12.  
  13. //获取图片路径
  14. const imgPath = path.resolve(dir)
  15.  
  16. let imgSize = null;
  17. fs.readdir(imgPath, (err, file) => {
  18. imgSize = sizeOf(dir + '/' +file[0]);
  19. //取第一个图片的尺寸作为框尺寸
  20. let cssString = `
  21. .fbf-animation{
  22. width: ${imgSize.width}px;
  23. height: ${imgSize.height}px;
  24. margin:auto;
  25. background-image: url(./${dir}/${file[0]});
  26. background-size: ${imgSize.width}px ${imgSize.height}px;
  27. background-repeat: no-repeat;
  28. animation-name: keyframes-img;
  29. animation-duration: 0.36s;
  30. animation-delay: 0s;
  31. animation-iteration-count: infinite;
  32. animation-fill-mode: forwards;
  33. animation-timing-function: steps(1);
  34. }
  35. `
  36. })
  37. })

生成CSS代码

然后根据图片数量生成相应的keyframes代码

  1. function toCss(dir, fileList){
  2. let _css = '';
  3. let start = 0;
  4. const per = Math.floor(100/fileList.length);
  5. fileList.map((path, i) => {
  6. if(i === fileList.length - 1){
  7. _css += `
  8. ${start + i*per}%, 100% {
  9. background:url(./${dir}/${path}) center center no-repeat;
  10. background-size:100% auto;
  11. }
  12. `
  13. }else{
  14. _css += `
  15. ${start + i*per}% {
  16. background:url(./${dir}/${path}) center center no-repeat;
  17. background-size:100% auto;
  18. }
  19. `
  20. }
  21. })
  22.  
  23. return _css;
  24. }
  25.  
  26. let frameCss = toCss(dir, newFile)
  27.  
  28. //取第一个图片的尺寸作为框尺寸
  29. let cssString = `
  30. .fbf-animation{
  31. width: ${imgSize.width}px;
  32. height: ${imgSize.height}px;
  33. margin:auto;
  34. background-image: url(./${dir}/${file[0]});
  35. background-size: ${imgSize.width}px ${imgSize.height}px;
  36. background-repeat: no-repeat;
  37. animation-name: keyframes-img;
  38. animation-duration: 0.36s;
  39. animation-delay: 0s;
  40. animation-iteration-count: infinite;
  41. animation-fill-mode: forwards;
  42. animation-timing-function: steps(1);
  43. }
  44. @keyframes keyframes-img {
  45. ${frameCss}
  46. }

生成html文件

最后我们把生成的CSS放到HTML里。

  1. //生成html
  2. let html = createHTML({
  3. title: '逐帧动画',
  4. scriptAsync: true,
  5. lang: 'en',
  6. dir: 'rtl',
  7. head: '<meta name="description" content="example">',
  8. body: '<div class="fbf-animation"></div>' + css,
  9. favicon: 'favicon.png'
  10. })
  11. fs.writeFile('fbf.html', html, function (err) {
  12. if (err) console.log(err)
  13. })

视觉美化

通过 chalk 来为打印信息加上样式,比如成功信息为绿色,失败信息为红色,这样子会让用户更加容易分辨,同时也让终端的显示更加的好看。

const chalk = require('chalk'); console.log(chalk.green('生成代码成功!')); console.log(chalk.red('生成代码失败'));

完整示例

  1. #!/usr/bin/env node
  2. const program = require('commander');
  3. const fs = require('fs');
  4. const path = require('path');
  5. const createHTML = require('create-html');
  6. const sizeOf = require('image-size');
  7. const chalk = require('chalk');
  8.  
  9. //排序
  10. const sortByFileName = files => {
  11. const reg = /[0-9]+/g;
  12. return files.sort((a, b) => {
  13. let imga = (a.match(reg) || []).slice(-1),
  14. imgb = (b.match(reg) || []).slice(-1)
  15. return imga - imgb
  16. });
  17. }
  18.  
  19. //删除.DS_Store
  20. function deleteDS(file) {
  21. file.map((v, i) => {
  22. if(v === '.DS_Store'){
  23. fs.unlink('img/.DS_Store', err => {})
  24. }
  25. })
  26. }
  27.  
  28. // 生成keyframe
  29. function toCss(dir, fileList){
  30. let _css = '';
  31. let start = 0;
  32. const per = Math.floor(100/fileList.length);
  33. fileList.map((path, i) => {
  34. if(i === fileList.length - 1){
  35. _css += `
  36. ${start + i*per}%, 100% {
  37. background:url(./${dir}/${path}) center center no-repeat;
  38. background-size:100% auto;
  39. }
  40. `
  41. }else{
  42. _css += `
  43. ${start + i*per}% {
  44. background:url(./${dir}/${path}) center center no-repeat;
  45. background-size:100% auto;
  46. }
  47. `
  48. }
  49. })
  50. console.log(chalk.green('css successed!'))
  51. return _css;
  52. }
  53.  
  54. program.version('1.0.0', '-v, --version')
  55. .command('start <dir>')
  56. .action((dir) => {
  57.  
  58. const imgPath = path.resolve(dir)
  59.  
  60. let imgSize = null;
  61. fs.readdir(imgPath, (err, file) => {
  62. const newFile = sortByFileName(file)
  63. deleteDS(newFile)
  64. imgSize = sizeOf(dir + '/' +file[0]);
  65. let frameCss = toCss(dir, newFile)
  66.  
  67. //取第一个图片的尺寸作为框尺寸
  68. let cssString = `
  69. .fbf-animation{
  70. width: ${imgSize.width}px;
  71. height: ${imgSize.height}px;
  72. margin:auto;
  73. background-image: url(./${dir}/${file[0]});
  74. background-size: ${imgSize.width}px ${imgSize.height}px;
  75. background-repeat: no-repeat;
  76. animation-name: keyframes-img;
  77. animation-duration: 0.36s;
  78. animation-delay: 0s;
  79. animation-iteration-count: infinite;
  80. animation-fill-mode: forwards;
  81. animation-timing-function: steps(1);
  82. }
  83. @keyframes keyframes-img {
  84. ${frameCss}
  85. }
  86. `
  87. let css = `
  88. <style>${cssString}</style>
  89. `
  90. //生成html
  91. let html = createHTML({
  92. title: '逐帧动画',
  93. scriptAsync: true,
  94. lang: 'en',
  95. dir: 'rtl',
  96. head: '<meta name="description" content="example">',
  97. body: '<div class="fbf-animation"></div>' + css,
  98. favicon: 'favicon.png'
  99. })
  100. fs.writeFile('fbf.html', html, function (err) {
  101. console.log(chalk.green('html successed!'))
  102. if (err) console.log(err)
  103. })
  104. })
  105. });
  106. program.parse(process.argv);

代码一共100行左右,可以说非常简单明了,有兴趣的同学可以试试。

最后

完成之后,使用npm publish fbf就可以把脚手架发布到 npm 上面,通过 -g 进行全局安装,就可以在自己本机上执行 fbf start [dir]来生成一个fbf.html文件,这样便完成了一个简单的node工具了。

使用node.js开发一个生成逐帧动画小工具的更多相关文章

  1. 《Node入门》读书笔记——用Node.js开发一个小应用

    Android APP的开发告一段落,一个稳定的.实现了基本功能的APP已经交付用户使用了!我和老板交流了下,接下来准备转战Node.js了,而且一部分前端的功能也要做进去!哈哈哈~~~接下来要朝一个 ...

  2. 腾讯高级工程师带你完整体验Node.js开发实战

    Node.js拥有广大的 JavaScript程序员基础并且完全开源,它被广泛地用在 Web服务.开发工作流.客户端应用等诸多领域.在 Web 服务开发这个领域,业界对 Node.js 的接受程度最高 ...

  3. 浅试WebStorm配置Node.js开发环境

    web前端开发IDE一直喜欢用WebStorm,这里简单介绍如何用WebStorm搭建一个Node.js开发环境. 首先,需要在本地安装好node.js,以及npm包管理工具.你也可以吧node.js ...

  4. Node.js开发Web后台服务

    一.简介 Node.js 是一个基于Google Chrome V8 引擎的 JavaScript 运行环境.Node.js 使用了一个事件驱动.非阻塞式 I/O 的模型,使其轻量又高效.Node.j ...

  5. Koa与Node.js开发实战(1)——Koa安装搭建(视频演示)

    学习架构: 由于Koa2已经支持ES6及更高版本,包括支持async方法,所以请读者保证Node.js版本在7.6.0以上.如果需要在低于7.6的版本中应用Koa的async方法,建议使用Babel ...

  6. Node.js 开发

    Node.js不必介绍,已经太火爆了.简单说是用Javascript开发Web服务端,基于Google V8引擎,单线程.不多说从零开始Windows平台下的Node.js的开发之旅. 环境工具为先 ...

  7. Node.js 开发指南

    1.Node.js 简介 Node.js 其实就是借助谷歌的 V8 引擎,将桌面端的 js 带到了服务器端,它的出现我将其归结为两点: V8 引擎的出色: js 异步 io 与事件驱动给服务器带来极高 ...

  8. node.js 开发简易的小爬虫

    node.js  开发简易的小爬虫 最近公司开发一款医药类的软件,所以需要一些药品的基础数据,所以本人就用node.js写一个简易的小爬虫,并写记录这个Demo以供大家参考. 一.开发前的准备: 1, ...

  9. Node.js开发入门—使用cookie保持登录

    这次来做一个站点登录的小样例,后面会用到. 这个演示样例会用到Cookie.HTML表单.POST数据体(body)解析. 第一个版本号,我们的用户数据就写死在js文件中. 第二个版本号会引入Mong ...

随机推荐

  1. Java基础(二十五)Java IO(2)文件File类

    File类是一个与流无关的类.File类的对象可以获取文件及其文件所在的目录.文件的长度等信息. 1.File对象的常用构造方法. (1)File(String pathname) File file ...

  2. Java基础(十九)集合(1)集合中主要接口和实现类

    1.Java集合框架为不同类型的集合定义了大量接口 其中,集合有两个基本接口:Collection和Map. 2.各接口的主要特征如下 (1)Collection接口:是List接口.Set接口和Qu ...

  3. WebApp 滚动列表的实现

    实现效果: 实现技术:overflow,flex,element::-webkit-scrollbar 实现步骤: //html:代码<div id="slider"> ...

  4. Https 与 iOS 信息安全

    转载自:swift-cafe 什么是 Https 咱们从最直观的说起. 我们平时在用电脑访问网页的时候,有时候会在地址栏的左边多出一个小锁的图标,就像这样: 这是大多数主流浏览器的一个通用做法,当我们 ...

  5. (23)ASP.NET Core EF关系数据库建模

    1.简介 一般而言,本部分中的配置适用于关系数据库.安装关系数据库提供程序时,此处显示的变为可用扩展方法(原因在于共享的Microsoft.EntityFrameworkCore.Relational ...

  6. Flask:网页路由及请求方式的设定

    1.Flask路由的实现 Flask的路由是由route装饰器来实现的 @app.route("/index/") def index(): return "hello ...

  7. 从《国产凌凌漆》看到《头号玩家》,你就能全面了解5G

    2019 年 9 月,移动.联通.电信5G套餐预约总和已突破 1000 万.2019 年 11 月,三大电信运营商将在全国范围内提供携号转网服务.2019 年内,移动将建立 5 万个 5G 基站,联通 ...

  8. 解决SAP740 GUI 搜索帮助(F4)回填值乱码的问题

    SAP 740客户端引入了搜索帮助增强功能,并且默认是开启该功能的,在带有F4搜索帮助的字段输入框中输入字段的前两个字符,可以自动以下拉框的方式带出包含包含所输入字符的条目,从而实现快速的输入帮助,如 ...

  9. Python3爬虫(2)_利用urllib.urlopen发送数据获得反馈信息

    一.urlopen的url参数 Agent url不仅可以是一个字符串,例如:https://baike.baidu.com/.url也可以是一个Request对象,这就需要我们先定义一个Reques ...

  10. 前端组件用 Scope 发布 npm 包的方法

    1.引言 多人.多组织或多组件发布 npm 包到同一个仓库时,可能出现命名冲突问题. 为了解决这个问题,npm 引入了“scope”(范围)概念. 在 Angular 项目中,我们通常可以看到“@an ...