如何搭建一个vue组件,并发布到npm

我将以一个vue2.0的组件为例,为大家详细的讲述下组件创建以及发布的过程

准备工作

  • 安装nodejs
  • 全局安装vue脚手架

npm install -g vue-cli

创建一个vue项目

vue create vue-loading-icon

按照脚手架提示进行安装


image.png

显示如下信息即为安装成功


250aebb0dc074544a3e9e0a5b76be108.png

修改项目目录

在项目根目录下创建packages文件夹
将src文件夹改名为examples
修改vue.config.js文件,若根目录下没有此文件,请手动添加

const { defineConfig } = require('@vue/cli-service');
const path = require('path');

function resolve(dir) {
  return path.join(__dirname, dir)
}

module.exports = defineConfig({
  transpileDependencies: true,
  pages: {
    index: {
      entry: 'examples/main.js',
      template: 'public/index.html',
      fileName: 'index.html'
    }
  },
  chainWebpack: config => {
    config.module
    .rule('js')
    .include.add(resolve('packages')).end()
    .use('babel').loader('babel-loader').tap(options=>{return options});
  },
  // css: { extract: false }
})

开始开发vue组件

在packages文件夹下写你想要开发的组件源码
我的packages的目录结构是这样的


ed93833cda134f33af9118a2b5b6bd81.png

LoadingA/main.vue如下:

<style scoped lang="less">
.loader01 {
  box-sizing: border-box;
  width: 56px;
  height: 56px;
  border: 8px solid var(--color);
  border-right-color: transparent;
  border-radius: 50%;
  animation: loader-rotate 1s linear infinite;
}
.loader01::after {
  content: "";
  width: 8px;
  height: 8px;
  background: var(--color);
  border-radius: 50%;
  position: absolute;
  top: -1px;
  left: 33px;
}

@keyframes loader-rotate {
  0% {
    transform: rotate(0);
  }
  100% {
    transform: rotate(360deg);
  }
}
</style>

<template>
  <div class="loader01" :style="pStyle"></div>
</template>

<script>
export default {
  props: {
    color: String,
    speed: Number,
  },
  data() {
    return {
      pStyle: {
        '--color': '#2595f1',
      },
    };
  },
  mounted() {
    this.setStyle();
  },
  methods: {
    setStyle() {
      if (this.color) this.$set(this.pStyle, "--color", this.color);
    },
  },
};
</script>

LoadingA/index.js如下:将您的组件注册

import LoadingA from './src/main.vue';
DsTree.install = function(Vue) {
  Vue.component(LoadingA.name,LoadingA);
}
export default LoadingA;

批量导出组件
packages/index.js如下:

import LoadingA from './LoadingA/index.js';
import LoadingB from './LoadingB/index.js';

let components = [
  LoadingA,
  LoadingB
]

const install = function (Vue) {
  if (install.installed) return;
  components.map(component=> Vue.component(component.name,component))
}
if (typeof window !== 'undefined' && window.Vue) {
  install(window.Vue);
}

export default {
  install,
  LoadingA,
  LoadingB
}

本地测试组件

接下来,就可以在本地进行组件的测试了,您可以在您的examples文件夹下,创建一个vue文件,然后引用此组件,进行组件的测试,我是在App.vue中引用了组件
App.vue内容如下:

<template>
  <div id="app">
    <div class="load-list">
      <div class="load-box load01">
        <div class="name">loading-A</div>
        <div class="demo">
          <loading-a/>
        </div>
      </div>
    </div>
  </div>
</template>

<script>

import LoadingA from '../packages/src/LoadingA.vue'
export default {
  name: 'App',
  components: {
    LoadingA
  }
}
</script>

<style>
*{
  margin: 0;
  padding: 0;
  box-sizing: content-box;
}
.load-box{
  width: 200px;
  height: 132px;
  border: solid 1px #cdcdcd;
}
.load-box .name{
  width: 100%;
  text-align: center;
  border-bottom: solid 1px #cdcdcd;
  height: 32px;
  line-height: 32px;
  flex-shrink: 0;
  flex-flow: 0;
}
.load-box .demo{
  height: 100px;
  display: flex;
  justify-content: space-around;
  align-items: center;
}

</style>

打包编译

本地测试完成后,就可以进行打包编译了,请您在package.json文件中的script中增加一行代码

"build-lib": "vue-cli-service build --target lib --name load --dest lib packages/index.js"

执行编译命令

接下来,我们执行下编译命令

npm run build-lib

等待命令执行完成后,项目根目录下会生成一个lib的文件夹,生成文件结构如下:


6c9ef490dddd4399bcaeefbb86f07401.png

如果您想生成内联样式,则需要在vue.config.js中增加

css: { extract: false }

还要在package.json中设置入口文件

"main": "lib/index.umd.min.js"

发布前的准备工作

首先,你需要确定你要发布的npm源
这块可以简单了解下nrm
什么事nrm呢,其实nrm 是一个 npm 源管理器,允许你快速地在 npm源间切换。
nrm安装

npm install -g nrm

查看源

nrm ls

使用源

nrm use XXX

之后你需要在项目根目录下创建一个.npmignore的文件
文件中写明你要忽略的文件,一般保留只需要保留README.md,lib和package.json即可,我的文件是这样写的

examples/
packages/
public/
node_nodules/
preview/
vue.config.js
postcss.config.js
babel.config.js
.gitignore
jsconfig.json
package-lock.json
*.map

另外,你需要将你项目的package.json文件中的 private设置为false

private:false

编写README.md文件,这里边的内容是其他人引用你的包的使用说明哦

登录到私服

npm login

根据命令提示,输入用户名,密码,以及绑定的邮箱

发布到私服

npm publish

稍等几秒钟后,您就可以到npm官网上搜索下您发布的包啦。


f634d724bcf74a209cec9fe5bbbbc7cc.png

到这里,整个发布流程就差不多了,但是您还是需要在做下引用测试哦

引用测试

在项目中按照使用说明进行操作,安装依赖包,使用组件测试是否正常发布,并可用即可


f0027127dcf34fcc8feb33b108aa733c.png

本地调试npm包

在你的npm源码项目中,也就是vue-loading-icon这个项目根目录下执行

npm link

创建一个软链,相当于全局安装了一个 vue-loading-icon依赖
在测试项目根目录下,也就是 vue-comp-demo中执行

npm link vue-loading

执行上述操作后,在npm包中修改了内容,重新编译后,即可实时的反应到引用的项目下,无需发布到npm中,待调试完成后,就可以发不到npm上了

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

推荐阅读更多精彩内容