首頁 > 軟體

Vue extend學習範例講解

2022-09-20 22:01:53

vue中通過extend動態建立全域性元件;

1、什麼是動態建立元件

只有在觸發事件的時候,才產生某元件,平時它並不存在;

2、Vue.extend()

使用基礎 Vue 構造器,建立一個“子類”。引數是一個包含元件選項的物件;其實就是一個子類構造器是Vue元件的核心api,實現思路就是使用原型繼承的方法返回了Vue的子類,並且利用mergeOptions把傳入元件的options和父類別的options進行了合併。

extend建立的是一個元件構造器,而不是一個具體的範例;

接收一個物件(包含元件選項的物件)作為引數,需要使用new來建立範例,並且需要$mount手動掛載到一個元素上,才可以獲取到這個元素的相應的資訊。

  • 脫離填鴨式的寫法;程式碼自由
  • 程式碼複用,解耦
  • 原生JS語法結合vue(jsx)
  • 通過傳入引數,可以顯示不同狀態的模板

基礎用法:

<div id="mount-point"></div>
// 建立構造器
/* Vue.extend( options )
  引數:{Object} options
  用法:使用基礎 Vue 構造器,建立一個「子類」。引數是一個包含元件選項的物件;
       data 選項是特例,需要注意: 在 Vue.extend() 中它必須是函數;*/
var Profile = Vue.extend({
  template: '<p>{{firstName}} {{lastName}} aka {{alias}}</p>',
  data: function () {
    return {
      firstName: 'Walter',
      lastName: 'White',
      alias: 'Heisenberg'
    }
  }
})
// 建立 Profile 範例,並掛載到一個元素上。
new Profile().$mount('#mount-point')
// 結果如下:
<p>Walter White aka Heisenberg</p>
/*
可以看到,extend 建立的是 Vue 構造器,而不是我們平時常寫的元件範例,所以不可以通過 new Vue({ components: testExtend }) 來直接使用,需要通過 new Profile().$mount('#mount-point') 來掛載到指定的元素上。
*/

3、通過extend實現彈窗的動態建立

3.1、建立動態元件

<!--動態元件的模板-->
<template>
  <!--  可以用MessageBox做蒙塵-->
  <div :class="['MessageBox',type]">
    <div class="inner">
      <header class="header">
        <h1 class="title">{{ title }}</h1>
        <span @click="$messageBox.hide()">x</span>
      </header>
      <div class="content">{{ content }}</div>
    </div>
  </div>
</template>
<script>
export default {
  name: "MessageBox",
  props: {
    title: {
      type: String,
      default: "this is title",
    },
    content: {
      type: String,
      default: "this is content",
    },
    type: {
      type: String,
      default: "primary",
      //檢測傳進來的型別是否是這四種,通過ES6提供的includes方法模糊查詢
      validator(value) {
        return [
          "primary",
          "success",
          "warn",
          "danger"
        ].includes(value);
      }
    }
  }
}
</script>
<style scoped lang="less">
.MessageBox {
  position: fixed;
  left: 50%;
  top: 0;
  //蒙塵的大小設定
  width: 50%;
  height: 400px;
  background-color: rgba(0, 0, 0, .5);
  //不同彈窗的樣式
  &.primary {
    .header {
      background-color: blue;
      color: #fff;
    }
  }
  &.success {
    .header {
      background-color: green;
      color: #fff;
    }
  }
  &.warn {
    .header {
      background-color: rgba(255, 138, 71, 0.96);
      color: #fff;
    }
  }
  &.danger {
    .header {
      background-color: red;
      color: #fff;
    }
  }
  .inner {
    position: absolute;
    top: 100px;
    left: 50%;
    width: 500px;
    margin-left: -250px;
    background-color: #fff;
    box-shadow: 1px 3px 5px #ddd;
    border-radius: 5px;
    overflow: hidden;
    .header {
      height: 44px;
      padding: 0 10px;
      line-height: 44px;
      box-sizing: border-box;
      h1 {
        margin: 0;
        font-weight: normal;
      }
      .title {
        font-size: 16px;
        float: left;
      }
      span {
        //將滑鼠改為小手樣式
        cursor: pointer;
        float: right;
      }
    }
    .content {
      padding: 20px;
      box-sizing: border-box;
    }
  }
}
</style>

3.2、編輯動態元件的邏輯

//引入需要動態建立的模板
import _MessageBox from "@/components/messageBox/MessageBox";
export default {
    //install開發外掛的方法,install帶有Vue的構造器,可以使用Vue.extend,和Vue.component(註冊元件)
    //在Vue.use的時候就會呼叫這個install
    install(Vue) {
        let messageBox = null;
        //使用Vue.component全域性註冊元件
        Vue.component(_MessageBox.name, _MessageBox);
        //將方法新增到Vue的prototype屬性中,這樣範例就可以繼承裡面的方法
        Vue.prototype.$messageBox = {
            show, hide,
            info({title, content, type}, callback) {
                this.show({title, content, type: "primary"}, callback)
            },
            success({title, content, type}, callback) {
                this.show({title, content, type: "success"}, callback)
            },
            warn({title, content, type}, callback) {
                this.show({title, content, type: "warn"}, callback)
            },
            danger({title, content, type}, callback) {
                this.show({title, content, type: "danger"}, callback)
            }
        }
        //顯示彈窗
        function show(props, callback) {
            //判斷這個元件是否存在,如果不存在
            if (!messageBox) {
                //生成建構函式、構造器
                const MessageBox = Vue.extend({
                    /*
                    render該渲染函數接收一個 createElement 方法作為第一個引數用來建立 VNode(節點)。
                   如果元件是一個函陣列件,渲染函數還會接收一個額外的 context 引數,為沒有範例的函陣列件提供上下文資訊。
                  */
                    //此處傳入的是一個函陣列件,所以渲染的函數還可以額外接收一個引數
                    render(h) {
                        //h函數就是vue中的createElement函數,這個函數作用就是建立虛擬dom,追蹤dom變化的
                        return h("MessageBox", {
                            //用於接收傳遞的引數
                            props: {...props}
                        })
                    }
                });
                //將動態模板元件範例化
                messageBox = new MessageBox();
                //將這個範例手動掛載,掛載後可以通過$el獲取這個元素
                this.vm = messageBox.$mount();
                //將元件新增到body上,脫離了根節點,不在"id=app中"
                document.body.appendChild(this.vm.$el)
                callback && callback();
            }
        }
        //關閉彈窗
        function hide(callback) {
            //移出這個元件
            document.body.removeChild(this.vm.$el);
            //將這個範例銷燬
            messageBox.$destroy();
            messageBox = null;
            this.vm = null;
            //如果存在才會執行
            callback && callback();
        }
    }
}

3.3、在main.js中引入使用

import Vue from 'vue'
import App from './App.vue'
//1、引入
import MessageBox from "@/components/messageBox";
//2、全域性註冊使用
Vue.use(MessageBox);
new Vue({
    render: h => h(App)
}).$mount('#app')

3.4、在需要的地方通過觸發事件顯示彈窗

<template>
  <div>
    <button @click="showMessageBox">show</button>
    <button @click="showInfoMessageBox">info</button>
    <button @click="showSuccessMessageBox">success</button>
    <button @click="showWarnMessageBox">warn</button>
    <button @click="showDangerMessageBox">danger</button>
  </div>
</template>
<script>
export default {
  name: "Extend",
  methods: {
    //通過this.$messageBox可以存取到Vue範例的屬性和方法
    showMessageBox() {
      this.$messageBox.success({
        title: 'App',
        content: 'this is content of extend study',
        type: 'success'
      }, () => {
        console.log('show over')
      })
    },
    showInfoMessageBox() {
      this.$messageBox.info({
        title: 'App',
        content: 'this is content of extend study',
      }, () => {
        console.log('info over')
      })
    },
    showSuccessMessageBox() {
      this.$messageBox.success({
        title: 'App',
        content: 'this is content of extend study',
        type: 'success'
      }, () => {
        console.log('success over')
      })
    },
    showWarnMessageBox() {
      this.$messageBox.warn({
        title: 'App',
        content: 'this is content of extend study',
        type: 'warn'
      }, () => {
        console.log('warn over')
      })
    },
    showDangerMessageBox() {
      this.$messageBox.danger({
        title: 'App',
        content: 'this is content of extend study',
        type: 'danger'
      })
    }
  }
}
</script>

3.5、效果圖

到此這篇關於Vue extend學習範例講解的文章就介紹到這了,更多相關Vue extend內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com