Vue3已经正式发布有一段时间,最近也打算学习一下,这个项目是在学Vue3的时候看到别人分享的demo,然后自己在这基础上修改一下,加深映像和理解;

开发环境

查看vue-cli版本

vue -V

升级vue-cli3到vue-cli4

npm install -g @vue/cli

node版本 >10.x

初始化项目

vue create dalou-ui

选择 Manually select features (自定义):

vue typescript定义类_前端

这是我的项目配置

vue typescript定义类_vue.js_02

预览地址

项目架构

在项目根目录新建packages目录来存放组件,src文件夹下新建demoCode文件夹存放组件使用样例,新建markdown文件夹存放md文件。如下图

vue typescript定义类_vue.js_03

根目录下新建vue.config.js文件:

module.exports = {
  pages: {
    index: {
      entry: 'src/main.ts', // 入口
      template: 'public/index.html', // 模板
      filename: 'index.html' // 输出文件
    }
  },
  devServer: {
    port: 8080, //固定端口
    hot: true, //开启热更新
    open: 'Google Chrome' //固定打开浏览器
  }

组件开发

在packages文件夹下新建button文件夹,然后在button文件根目录下新建index.js文件:

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

button文件根目录下新建src文件夹,然后src文件夹里新建index.vue

<template>
  <button class="z-button" :class="classes" @click="onClick" :disabled="disabled">
    <span v-if="loading" class="z-button-spin"><svg class="menu" aria-hidden="true" @click="toggleMenu">
        <use xlink:href="#icon-jiazaizhong"></use>
      </svg></span>
    <span>
      <slot/>
    </span>
  </button>
</template>

<script>
  export default {
    name: 'ZButton',
    props: {
      //按钮主题
      theme: {
        type: String,
        default: 'default'
      },
      size: {
        type: String,
        default: 'normal'
      },
      loading: {
        type: Boolean,
        default: false
      },
      disabled: {
        type: Boolean,
        default: false,
      }
    },
    methods: {
      onClick() {
        if (this.disabled) {
          return
        }
        //this.$emit('click')
      },
    },
    computed: {
      classes() {
        const loadingClass = this.loading ? ' z-button-loading' : ''
        return `z-button-${this.theme} z-button-${this.size}${loadingClass}`
      }
    }
  }
</script>

<style lang="scss">
  //主题色
  $theme: #1E90FF;

  //浅色背景
  $simpleBg: #DEDDFF;

  @keyframes spin {
    0% {
      transform: rotate(0);
    }

    100% {
      transform: rotate(360deg);
    }
  }

  .z-button {
    display: inline-block;
    line-height: 1;
    white-space: nowrap;
    cursor: pointer;
    color: #606266;
    text-align: center;
    box-sizing: border-box;
    outline: none;

    &.z-button-loading {
      pointer-events: none;

      .z-button-spin {
        width: 20px;
        height: 20px;
        margin-right: 5px;
        animation: spin infinite 1s linear;

        >svg {
          width: 100%;
          height: 100%;
        }
      }
    }

    >span {
      display: inline-block;
      vertical-align: middle;

    }

    margin: 0;
    transition: .1s;
    font-weight: 500;
    font-size: 14px;
    border-radius: 4px;

    &.z-button-default {
      background: #fff;
      border: 1px solid #dcdfe6;

      &:hover,
      &:focus {
        color: #0049FF;
        border-color: #0049FF;
        background-color: #fff;
      }
    }

    &.z-button-primary {
      background: #0049FF;
      border: 1px solid #dcdfe6;
      color: #eeeeee;

      &:hover,
      &:focus {
        color: #eeeeee;
        background-color: #2664FF;
      }
    }

    &.z-button-success {
      background: #3CB371;
      border: 1px solid #dcdfe6;
      color: #eeeeee;

      &:hover,
      &:focus {
        color: #eeeeee;
        background-color: #6be3bc;
      }
    }

    &.z-button-danger {
      background: #d72323;
      border: 1px solid #d72323;
      color: #eeeeee;

      &:hover,
      &:focus {
        color: #eeeeee;
        background-color: #dd5656;
      }
    }

    &.z-button-info {
      background: #52616b;
      border: 1px solid #dcdfe6;
      color: #eeeeee;

      &:hover,
      &:focus {
        color: #eeeeee;
        background-color: #6c777d;
      }
    }

    &.z-button-warning {
      background: #fce38a;
      border: 1px solid #dcdfe6;
      color: #4a4444;

      &:hover,
      &:focus {
        color: #8e8282;
        background-color: #f7e4a4;
      }
    }

    &.z-button-text {
      background: #fff;
      border: 0;
      color: #4a4444;

      &:hover,
      &:focus {
        >span {
          color: #8e8282;
          border-color: #222831;
          background-color: #f7e4a4;
        }
      }
    }

    &.z-button-normal {
      padding: 12px 20px;
    }

    &.z-button-small {
      padding: 10px 20px;
      font-size: 14px;
      border-radius: 4px;
    }

    &.z-button-mini {
      padding: 9px 15px;
      font-size: 12px;
      border-radius: 3px;
    }
  }
</style>

 

在packages文件夹下新建index.js文件

import ZButton from './button'

// 组件集合,用于遍历
const components = [ZButton];

// 定义 install 方法
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,
  //所有组件,必须具有install方法才能使用Vue.use()
  ...components
};

//组件按需引入时需要里添加
export {
  install,
  ZButton
};

完成后项目结构如图:

vue typescript定义类_前端_04

修改main.ts文件,import xhwlComponent from "../packages"

import { createApp } from "vue";
import App from "./App.vue";
import {router} from "./router";
import store from "./store";
import dalouUi from "../packages";

createApp(App)
  .use(store)
  .use(dalouUi)
  .use(router)
  .mount("#app");

这样项目里就可以直接全局使用刚才开发的button组件了

使用文档

demoCode文件下新建button文件夹,在里面新建buttonModal.vue文件,这里就是使用组件的例子

<template>
  <div>
    <z-button class="b-m b-b">默认按钮</z-button>
    <z-button theme="primary" class="b-m b-b">主要按钮</z-button>
    <z-button theme="success" class="b-m b-b">成功按钮</z-button>
    <z-button theme="warning" class="b-m b-b">警告按钮</z-button>
    <z-button theme="info" class="b-m b-b">信息按钮</z-button>
    <z-button theme="danger" class="b-m">危险按钮</z-button>
    <z-button theme="text" class="b-m">文字按钮</z-button>
  </div>
</template>

<script >
export default {
  name: 'button-demo'
}
</script>

<style lang="scss" scoped>
.b-m {
  margin-right: 15px
}

.b-b {
  margin-bottom: 15px;
}
</style>

views文件夹下新增buttonDoc.vue文件,这个是使用文档页面

<template>
  <div class="container">
    <demo-doc title="基础用法" description="button的基础用法" :imgUrl="url" :modal="buttonModal" :component="modaljs">
    </demo-doc>
    <attr :data="data"></attr>
  </div>
</template>

<script>
  import buttonModal from "@/demoCode/button/buttonModal.vue";   // 组件展示
  import modaljs from "@/demoCode/button/buttonModal.ts";               // 代码展示
  import DemoDoc from "@/components/demoTemplate.vue";                
  import Attr from "@/components/Attr.vue";                                            //  展示需要传递的参数、方法等
  import {ref, onMounted, nextTick} from 'vue'
  export default {
    components: {
      Attr,
      DemoDoc
    },
    setup(props, context) {
      let url = ref('')
      const data = { 
        columns:[{
          params: 'size',
          desc: '尺寸',
          type: 'string',
          select: 'normal / small / mini',
          default: 'normal',
        },
        {
          params: 'theme',
          desc: '按钮类型',
          type: 'string',
          select: 'primary / success / warning / info / danger / text',
          default: 'primary',
        }]
      }
      return {
        buttonModal,
        modaljs,
        data,
        url
      }
    },
  }
</script>

<style lang="scss" scoped>
  .container {
    width: 100%;
  }
</style>

注意:显示md文件需要安装一些依赖

npm install markdown-loader 

npm install html-loader 

npm install marked

npm install github-markdown-css

修改vue.config.js 文件

const path = require('path')
module.exports = {
  pages: {
    index: {
      entry: 'src/main.ts', // 入口
      template: 'public/index.html', // 模板
      filename: 'index.html' // 输出文件
    }
  },
  devServer: {
    port: 8080, //固定端口
    hot: true, //开启热更新
    open: 'Google Chrome' //固定打开浏览器
  },
  chainWebpack: config => {
    // @ 默认指向 src 目录
    // 新增一个 ~ 指向 packages
    config.resolve.alias
      .set('~', path.resolve('packages'))
    config.module
      .rule('md')
      .test(/\.md$/)
      .use('html-loader')
      .loader('html-loader')
      .end()
      .use('markdown-loader')
      .loader('markdown-loader')
      .end()
      .rule('js')
      .include.add(/packages/)
      .end()
      .use('babel')
      .loader('babel-loader')
      .tap(options => {
        // 修改它的选项...
        return options
      })
  },
}

codePer.vue文件里展示代码时高亮需要安装prismjs

npm install prismjs

预览地址