亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

手寫(xiě)實(shí)現(xiàn)vue2下拉菜單dropdown組件實(shí)例

 更新時(shí)間:2022年08月24日 15:07:09   作者:陌年微涼_  
這篇文章主要為大家介紹了手寫(xiě)vue2下拉菜單dropdown組件實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

概述

一般后臺(tái)項(xiàng)目結(jié)合分頁(yè)組件用到這個(gè)dropdown組件,用來(lái)做顯示分頁(yè)條數(shù),其他用到這個(gè)組件的地方倒不是很多,其實(shí)現(xiàn)思路和select組件有那么些相似,現(xiàn)記錄下這個(gè)組件的實(shí)現(xiàn)。

最終效果(動(dòng)圖沒(méi)顯示出來(lái),請(qǐng)稍定會(huì)兒,可以先看后面)

實(shí)現(xiàn)原理

這個(gè)組件和select組件記起相似,可以參考我之前的文章【手寫(xiě)vue2select下拉組件】,要做這個(gè)組件,需要注意以下幾點(diǎn):

組件分為兩部分:

  • 供我們點(diǎn)擊的文字,按鈕,超鏈接等等(當(dāng)成插槽供用戶(hù)提供)
  • 下拉菜單項(xiàng)(支持邊框,禁用等)

使用該組件應(yīng)當(dāng)提供的事件應(yīng)該是點(diǎn)擊item項(xiàng),然后將對(duì)應(yīng)的item的對(duì)應(yīng)value暴露出來(lái),供用戶(hù)使用。

組件菜單項(xiàng)的顯示隱藏需要過(guò)渡動(dòng)畫(huà)。

默認(rèn)菜單項(xiàng)方向向下,當(dāng)下方可視區(qū)的高度不足以容納下拉菜單的高度的時(shí)候,自動(dòng)讓菜單方向向上。

具體實(shí)現(xiàn)

目錄結(jié)構(gòu)

emitter.js

這個(gè)在之前的組件實(shí)現(xiàn)過(guò)程中介紹過(guò)這個(gè)文件,主要是為了解決跨多層級(jí)父子組件之前數(shù)據(jù)通信的,本質(zhì)上實(shí)現(xiàn)原理為發(fā)布訂閱模式。

/**
 * @Description 由于涉及到跨組件之間通信,因此我們只有自己實(shí)現(xiàn)發(fā)布訂閱的模式,來(lái)實(shí)現(xiàn)組件之間通信,靈感主要來(lái)源于element-ui組件庫(kù)源碼中跨層級(jí)父子組件通信方案,本質(zhì)上也是發(fā)布訂閱和$emit和$on
 * @param { String } componentName 組件名
 * @param { String } eventName 事件名
 * @param { argument } params 參數(shù)
 **/
// 廣播通知事件
function _broadcast(componentName, eventName, params) {
  // 遍歷當(dāng)前組件的子組件
  this.$children.forEach(function (child) {
    // 取出componentName,組件options上面可以自己配置
    var name = child.$options.componentName;
    // 如果找到了需要通知的組件名,觸發(fā)組件上面的$eimit方法,觸發(fā)自定義事件
    if (name === componentName) {
      child.$emit.apply(child, [eventName].concat(params));
    } else {
      // 沒(méi)找到,遞歸往下找
      _broadcast.apply(child, [componentName, eventName].concat([params]));
    }
  });
}
const emiiter = {
  methods: {
    // 派發(fā)事件(通知父組件)
    dispatch(componentName, eventName, params) {
      var parent = this.$parent || this.$root;
      var name = parent.$options.componentName;
      // 循環(huán)往上層父組件,知道知道組件名和需要觸發(fā)的組件名相同即可,然后觸發(fā)對(duì)應(yīng)組件的事件
      while (parent && (!name || name !== componentName)) {
        parent = parent.$parent;
        if (parent) {
          name = parent.$options.componentName;
        }
      }
      if (parent) {
        parent.$emit.apply(parent, [eventName].concat(params));
      }
    },
    // 廣播事件(通知子組件)
    broadcast(componentName, eventName, params) {
      _broadcast.call(this, componentName, eventName, params);
    },
  },
};
export default emiiter;

MyDropdown.vue

主要暴露給用戶(hù)使用的組件

<template>
  <div
    class="my-dropdown"
    @click.stop="trigger == 'click' ? (showMenu = !showMenu) : ''"
    @mouseenter="trigger == 'hover' ? (showMenu = true) : ''"
    @mouseleave="trigger == 'hover' ? (showMenu = false) : ''"
    ref="myDropDdown"
  >
    <div class="tip-text" ref="tipText">
      <slot></slot>
    </div>
    <slot name="list"></slot>
  </div>
</template>
<script>
import emitter from "./emitter";
export default {
  name: "MyDropdown",
  componentName: "MyDropdown",
  mixins: [emitter],
  props: {
    // 觸發(fā)顯示方式
    trigger: {
      type: String,
      default: "click",
    },
    // 下來(lái)菜單的出現(xiàn)位置(上方,下方)
    placement: {
      type: String,
      default: "bottom",
      validator: function (value) {
        // 這個(gè)值必須匹配下列字符串中的一個(gè)
        return ["bottom", "top"].includes(value);
      },
    },
  },
  data() {
    return {
    //控制菜單是否顯示
      showMenu: false,
    };
  },
  mounted() {
      //初始化自定義事件
    this.initEvent();
  },
  methods: {
    // 初始化
    initEvent() {
    //訂閱當(dāng)item點(diǎn)擊的時(shí)候,發(fā)布o(jì)n-click事件,告知外部
      this.$on("item-click", (params) => {
        this.$emit("on-click", params);
        this.showMenu = false;
      });
      //空白點(diǎn)擊要隱藏菜單,需要執(zhí)行的函數(shù)需要綁定this指向
      this.handleEmptyDomElementClickBindThis =
        this.handleEmptyDomElementClick.bind(this);
      window.addEventListener("click", this.handleEmptyDomElementClickBindThis);
    },
    // 處理空白區(qū)域點(diǎn)擊,隱藏菜單列表
    handleEmptyDomElementClick(e) {
      if (!Array.from(this.$refs.myDropDdown.childNodes).includes(e.target)) {
        this.showMenu = false;
      }
    },
  },
  beforeDestroy() {
    // 移除window上面的事件
    window.removeEventListener(this.handleEmptyDomElementClickBindThis);
  },
  watch: {
  //變化的時(shí)候,通知子組件隱藏菜單列表
    showMenu() {
      this.broadcast("MyDropdownMenu", "set-menu-status", this.showMenu);
    },
  },
};
</script>
<style lang="less">
.my-dropdown {
  position: relative;
}
</style>

MyDropdownMenu.vue

主要暴露給用戶(hù)使用的組件,菜單列表組件

<template>
<!-- 涉及到高度,位移,過(guò)渡使用js鉤子函數(shù)的方式比較好處理些 -->
  <transition
    @before-enter="beforeEnter"
    @enter="enter"
    @leave="leave"
    v-bind:css="false"
  >
    <div class="my-dropdown-menu" v-if="showMeune" ref="myDroupdownMenu">
      <slot></slot>
    </div>
  </transition>
</template>
<script>
import emitter from "./emitter";
export default {
  name: "MyDropdownMenu",
  componentName: "MyDropdownMenu",
  mixins: [emitter],
  data() {
    return {
      showMeune: false,
      timer: null,
    };
  },
  mounted() {
    this.$on("set-menu-status", (status) => {
      this.showMeune = status;
    });
  },
  methods: {
  //進(jìn)入前,初始化需要過(guò)渡的屬性
    beforeEnter: function (el) {
      // 初始化
      el.style.opacity = 0;
      el.style.transform = "scaleY(0)";
      el.style.transformOrigin = "top center";
    },
  //dom進(jìn)入
    enter: function (el, done) {
    //獲取文檔可視區(qū)高度
      const htmlClientHeight = document.documentElement.clientHeight;
      //菜單列表相對(duì)于父元素的top偏移量
      const offsetTop = el.offsetTop;
      const scrollHeight = el.scrollHeight;
      //獲取當(dāng)前元素和可視區(qū)的一些長(zhǎng)度(top,left,bottom等)
      const { bottom } = el.getBoundingClientRect();
      // 說(shuō)明底部高度不夠顯示菜單了,這時(shí)候我們需要調(diào)整菜單朝上面顯示
      if (htmlClientHeight - bottom < scrollHeight) {
        el.style.transformOrigin = "bottom center";
        el.style.top = -(scrollHeight + 20) + "px";
      } else {
      //查看是否placement屬性,是的話(huà)我們主動(dòng)處理
        if (this.$parent.placement == "top") {
          el.style.transformOrigin = "bottom center";
          el.style.top = -(scrollHeight + 20) + "px";
        } else {
          el.style.top = offsetTop + "px";
        }
      }
      el.style.transform = "scaleY(1)";
      el.style.opacity = 1;
    //根據(jù)官網(wǎng)事例,必須在enter和leave里面調(diào)用done函數(shù),不然過(guò)渡動(dòng)畫(huà)不生效(切記)
      done();
    },
    //dom元素離開(kāi) 
    leave: function (el, done) {
      el.style.transform = "scaleY(0)";
      el.style.opacity = 0;
      clearTimeout(this.timer);
      this.timer = setTimeout(() => {
      //根據(jù)官網(wǎng)事例,必須在enter和leave里面調(diào)用done函數(shù),不然過(guò)渡動(dòng)畫(huà)不生效(切記)
        done();
      }, 250);
    },
  },
};
</script>
<style lang="less">
.my-dropdown-menu {
  min-width: 100px;
  max-height: 200px;
  overflow: auto;
  margin: 5px 0;
  padding: 5px 0;
  background-color: #fff;
  box-sizing: border-box;
  border-radius: 4px;
  box-shadow: 0 1px 6px rgb(0 0 0 / 20%);
  z-index: 900;
  transform-origin: top center;
  position: absolute;
  transition: transform 0.25s ease, opacity 0.25s ease;
}
</style>

MyDropdownItem.vue

主要暴露給用戶(hù)使用的組件,菜單列表項(xiàng)組件,組件內(nèi)容很簡(jiǎn)單,主要就是展示item數(shù)據(jù)和綁定點(diǎn)擊事件。

<template>
  <div
    :class="[
      'my-dropdownItem',
      divided ? 'my-dropdownItem-divided' : '',
      disabled ? 'my-dropdownItem-disabled' : '',
    ]"
    @click.stop="handleItemClick"
  >
    <slot></slot>
  </div>
</template>
<script>
import emitter from "./emitter";
export default {
  name: "MyDropdownItem",
  componentName: "MyDropdownItem",
  mixins: [emitter],
  props: {
    divided: {
      type: Boolean,
      default: false,
    },
    disabled: {
      type: Boolean,
      default: false,
    },
    name: {
      type: String,
      default: "",
    },
  },
  data() {
    return {};
  },
  methods: {
    handleItemClick() {
      if (this.disabled) return;
      // item項(xiàng)點(diǎn)擊通知dropdown組件派發(fā)到外部的自定義事件
      this.dispatch("MyDropdown", "item-click", this.name);
    },
  },
};
</script>
<style lang="less">
.my-dropdownItem {
  margin: 0;
  line-height: normal;
  padding: 7px 16px;
  clear: both;
  color: #515a6e;
  font-size: 14px !important;
  white-space: nowrap;
  list-style: none;
  cursor: pointer;
  transition: background 0.2s ease-in-out;
  &:hover {
    background: #f3f3f3;
  }
}
.my-dropdownItem-divided {
  border-bottom: 1px solid #eee;
}
.my-dropdownItem-disabled {
  color: #cacdd2;
  cursor: not-allowed;
  &:hover {
    background: #fff;
  }
}
</style>

總結(jié)

類(lèi)似組件庫(kù)中的這種經(jīng)常出現(xiàn)跨多層級(jí)組件通信,需要特殊處理,一般emitter.js文件里面的封裝我們?cè)陂_(kāi)發(fā)中一般是用不到的,我們寫(xiě)組件經(jīng)常也就父子組件之間,很少會(huì)跨祖孫級(jí)別,但是在組件庫(kù)中,這種關(guān)系就很多,因此需要單獨(dú)利用發(fā)布訂閱來(lái)處理,這種模式用到實(shí)際項(xiàng)目里面也是很管用的。

以上就是手寫(xiě)實(shí)現(xiàn)vue2下拉菜單dropdown組件實(shí)例的詳細(xì)內(nèi)容,更多關(guān)于vue 下拉菜單dropdown的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Vue修改Dom不生效的解決

    Vue修改Dom不生效的解決

    這篇文章主要介紹了Vue修改Dom不生效的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-06-06
  • vue-calendar-component 封裝多日期選擇組件的實(shí)例代碼

    vue-calendar-component 封裝多日期選擇組件的實(shí)例代碼

    這篇文章主要介紹了vue-calendar-component 封裝多日期選擇組件,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-12-12
  • Vue生命周期詳解

    Vue生命周期詳解

    這篇文章詳細(xì)介紹了Vue的生命周期,文中通過(guò)代碼示例介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-04-04
  • Vue如何獲取兩個(gè)時(shí)間點(diǎn)之間的所有間隔時(shí)間

    Vue如何獲取兩個(gè)時(shí)間點(diǎn)之間的所有間隔時(shí)間

    這篇文章主要介紹了Vue如何獲取兩個(gè)時(shí)間點(diǎn)之間的所有間隔時(shí)間,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-07-07
  • vue 實(shí)現(xiàn)v-for循環(huán)回來(lái)的數(shù)據(jù)動(dòng)態(tài)綁定id

    vue 實(shí)現(xiàn)v-for循環(huán)回來(lái)的數(shù)據(jù)動(dòng)態(tài)綁定id

    今天小編就為大家分享一篇vue 實(shí)現(xiàn)v-for循環(huán)回來(lái)的數(shù)據(jù)動(dòng)態(tài)綁定id,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-11-11
  • vue實(shí)現(xiàn)圖片拖拽功能

    vue實(shí)現(xiàn)圖片拖拽功能

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)圖片拖拽功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • Vue computed計(jì)算屬性總結(jié)記錄

    Vue computed計(jì)算屬性總結(jié)記錄

    在vue中,有時(shí)候你需要對(duì)data中的數(shù)據(jù)進(jìn)行處理,或者對(duì)抓取的數(shù)據(jù)進(jìn)行處理之后再掛載呈現(xiàn)到標(biāo)簽中,這時(shí)候你就需要計(jì)算屬性了,當(dāng)然看到這里你可能還是不了解那下面我舉幾個(gè)實(shí)例并附代碼解釋
    2023-02-02
  • vue實(shí)現(xiàn)指定日期之間的倒計(jì)時(shí)

    vue實(shí)現(xiàn)指定日期之間的倒計(jì)時(shí)

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)指定日期之間的倒計(jì)時(shí),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-05-05
  • vue前端路由以及vue-router兩種模式實(shí)例詳解

    vue前端路由以及vue-router兩種模式實(shí)例詳解

    路由這個(gè)概念最先是后端出現(xiàn)的,下面這篇文章主要給大家介紹了關(guān)于vue前端路由以及vue-router兩種模式的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-03-03
  • vue.js異步上傳文件前后端實(shí)現(xiàn)代碼

    vue.js異步上傳文件前后端實(shí)現(xiàn)代碼

    這篇文章主要為大家詳細(xì)介紹了vue.js異步上傳文件前后端的實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-08-08

最新評(píng)論