欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于Vue3+ts封裝一個(gè)簡(jiǎn)單版的Message組件

 更新時(shí)間:2023年09月01日 10:03:58   作者:實(shí)在想不出什么名字  
近日項(xiàng)目中需要使用信息提示框的功能,ui組件庫(kù)使用的是字節(jié)的arco-design-vue,看了一下,現(xiàn)有的Message不滿足要是需求,直接使用message組件的話,改樣式太麻煩,所以本文就本就介紹了基于Vue3+ts封裝一個(gè)簡(jiǎn)單版的Message組件,需要的朋友可以參考下

Vue3+ts封裝一個(gè)Message組件

項(xiàng)目中需要使用信息提示框的功能,ui組件庫(kù)使用的是字節(jié)的arco-design-vue??戳艘幌?,現(xiàn)有的Message不滿足要是需求,直接使用message組件的話,改樣式太麻煩。Notification組件樣式倒是符合了,但是彈出的位置不符合,查看了一下相關(guān)api,這位置不支持"top"。既然如此,那就去查看它倆的源碼,找到我想要的,修修改改,自己也寫一個(gè),嘻嘻。

源碼分析

源碼(以Message組件為例)主要分為三個(gè)模塊:message-item.vue、message-list.vue、message.ts。下面簡(jiǎn)單介紹一下這三個(gè)模塊

  • message-item.vue: 主要是編寫Message信息彈框的樣式和彈框按時(shí)(定時(shí))自動(dòng)消失(關(guān)閉)的邏輯
  • message-list.vue:主要就是添加動(dòng)畫
  • message.ts:這個(gè)最主要的。這里面要實(shí)現(xiàn)創(chuàng)建message、刷新message、關(guān)閉message、銷毀message以及message分類等一系類邏輯,同時(shí)還要把這些邏輯封裝(暫時(shí)找不到更好的詞來表達(dá))成一個(gè)對(duì)象拋出去。沒錯(cuò),這是一個(gè)單例設(shè)計(jì)模式(全局永遠(yuǎn)只有一個(gè)Message對(duì)象)

代碼實(shí)現(xiàn)

在完全實(shí)現(xiàn)之前,我看 message-list.vue 組件中的邏輯那么簡(jiǎn)單,想也不想就直接把它跟 message-item.vue 組成了一個(gè)組件 message-ul.vue,等到要實(shí)現(xiàn)邏輯的時(shí)候,才發(fā)現(xiàn)自己這個(gè)寫不行,因?yàn)槊恳粋€(gè)Message都可以設(shè)置自己的持續(xù)時(shí)長(zhǎng),時(shí)間一到就自己關(guān)閉了,不會(huì)影響到其他的。但是在 message-ul.vue 組件中,處理的是一個(gè)列表,要保證每個(gè)列表項(xiàng)的定時(shí)關(guān)閉不影響其他的,我沒想到解決辦法(嗚嗚嗚,我太笨了 ),所以最后還是老老實(shí)實(shí)跟 arco-design-vue 一樣(其實(shí)就是直接copy它)把 message-ul 組件拆分。
ps:后來我想了想,那可是字節(jié)哎,里面的大佬這么寫肯定是有原因的,我老老實(shí)實(shí)照著抄就行了,居然妄想組合操作。不過實(shí)踐檢驗(yàn)真理,我也算是.....好吧,我就是瞎搞了。
不嗶嗶了,看代碼吧!

message-item.vue

<template>
  <li class="message-item message">
    <p class="title">
      <img v-if="type === 'success'" src="@/assets/icons/task-sucess.png" alt="" />
      <img v-if="type === 'error'" src="@/assets/icons/task-error.png" alt="" />
      <img v-if="type === 'warn'" src="@/assets/icons/task-cancel.png" alt="" />
      <span>{{ title }}</span>
    </p>
    <p v-if="prompt" class="name">
      {{ prompt.length >= 20 ? `${prompt.slice(0, 15)}...` : prompt }}
    </p>
    <span class="close" @click="handleClose">
      <IconClose />
    </span>
  </li>
</template>
<script lang="ts" setup>
  import { onMounted, onUnmounted } from 'vue';
  import { IconClose } from '@arco-design/web-vue/es/icon';
  const emits = defineEmits(['close']);
  let timer = 0; // 每個(gè)實(shí)例?(item)有自己的定時(shí)器,這樣就不會(huì)影響其他的item了
  const props = defineProps({
    title: String, // 標(biāo)題
    prompt: {
      type: String,
      default: undefined,
    },
    id: {
      type: [String, Number],
      required: true,
    },
    type: {
      type: String,
      required: true,
    },
    duration: {
      type: Number,
      default: 3000,
    },
  });
  const clearTimer = () => {
    if (timer) {
      window.clearTimeout(timer);
      timer = 0;
    }
  };
  function handleClose() {
    emits('close', props.id);
  }
  const startTimer = () => {
    if (props.duration > 0) {
      timer = window.setTimeout(handleClose, props.duration);
    }
  };
  onMounted(() => {
    startTimer();
  });
  onUnmounted(() => {
    clearTimer();
  });
</script>

message-list.vue

<template>
  <TransitionGroup tag="ul" class="message-box" name="list" :theme="appStore.theme">
    <Message v-for="item in messages" :key="item.id" v-bind="item" @close="handleClose" />
  </TransitionGroup>
</template>
<script lang="ts" setup>
  import type { PropType } from 'vue';
  import { useAppStore } from '@/store';
  import { MessageItem } from './types';
  import Message from './message-item.vue';
  const appStore = useAppStore();
  const emits = defineEmits(['close']);
  defineProps({
    messages: {
      type: Array as PropType<MessageItem[]>,
      default: () => [],
    },
  });
  function handleClose(id: string | number) {
    emits('close', id);
  }
</script>
<style lang="less">
  // 樣式這里就放個(gè)動(dòng)畫的,其他的就不放了
  .list-enter-active,
  .list-leave-active {
    transition: all 0.5s ease;
  }
  .list-enter-from {
    opacity: 0;
    transform: translateY(30px);
  }
  .list-leave-to {
    opacity: 0;
    transform: translateY(-30px);
  }
</style>

message.ts

import type { AppContext, Ref } from 'vue';
import { createVNode, render, ref, reactive } from 'vue';
import { MessageItem, MessageConfig, MessageMethod } from './types';
import messageVue from './message-list.vue';
type _MessageConfig = MessageConfig & {
  type: 'success' | 'error' | 'warn';
};
class MessageManger {
  // 定義一個(gè)集合,用于存儲(chǔ)所有message的id,這里使用Set是為了自動(dòng)去重
  private readonly messageIds: Set<number | string>;
  // 定義一個(gè)響應(yīng)式集合用于存儲(chǔ)所有message對(duì)象,用響應(yīng)式的好處,數(shù)據(jù)變?cè)噲D變,試圖變數(shù)據(jù)變
  private readonly messages: Ref<MessageItem[]>;
  private container: HTMLElement | null;
  private messageCount = 0;
  constructor(config: _MessageConfig, appContext?: AppContext) {
    this.messageIds = new Set();
    this.messages = ref([]);
    this.container = document.createElement('div');
    this.container.setAttribute('class', 'my-message');
    // 創(chuàng)建一個(gè)虛擬DOM:同時(shí)messages傳遞給 messageVue;處理(實(shí)現(xiàn)) messageVue 向外拋出的close、afterClose等方法,
    const vm = createVNode(messageVue, {
      messages: this.messages.value,
      onClose: this.remove,
      onAfterClose: this.destroy,
    });
    // eslint-disable-next-line no-use-before-define
    if (appContext ?? Message.myContext) {
      // eslint-disable-next-line no-use-before-define
      vm.appContext = appContext ?? Message.myContext;
    }
    // 將虛擬DOM渲染到指定容器中,并將容器添加到body標(biāo)簽里
    render(vm, this.container);
    document.body.appendChild(this.container);
  }
  add = (config: _MessageConfig) => {
    // 添加message時(shí),如果沒有傳遞id就使用 messageCount 創(chuàng)建一個(gè)
    this.messageCount += 1;
    const id = config.id ?? `message_${this.messageCount}`;
    if (this.messageIds.has(id)) {
      return this.update(id, config);
    }
    const message: MessageItem = reactive({ id, ...config });
    this.messages.value.push(message);
    this.messageIds.add(id);
    return {
      close: () => this.remove(id),
    };
  };
  update = (id: number | string, config: _MessageConfig) => {
    for (let i = 0; i < this.messages.value.length; i += 1) {
      if (this.messages.value[i].id === id) {
        const resetOnUpdate = config.duration !== undefined;
        Object.assign(this.messages.value[i], { ...config, id, resetOnUpdate });
        break;
      }
    }
    return {
      close: () => this.remove(id),
    };
  };
  remove = (id: number | string) => {
    for (let i = 0; i < this.messages.value.length; i += 1) {
      const item = this.messages.value[i];
      if (item.id === id) {
        if (typeof item.onClose === 'function') {
          item.onClose(id);
        }
        this.messages.value.splice(i, 1);
        this.messageIds.delete(id);
        break;
      }
    }
    this.destroy();
  };
  clear = () => {
    this.messages.value.splice(0);
  };
  destroy = () => {
    // 如果所有message都關(guān)閉了,那就銷毀整個(gè)實(shí)例對(duì)象,并從body中移除容器
    if (this.messages.value.length === 0 && this.container) {
      render(null, this.container);
      document.body.removeChild(this.container);
      this.container = null;
      // eslint-disable-next-line no-use-before-define
      messageInstance = null;
    }
  };
}
let messageInstance: MessageManger | null = null;
const types = ['success', 'error', 'warn'] as const;
const message = types.reduce((pre, value) => {
  pre[value] = (config: MessageConfig, appContext?: AppContext) => {
    const newConfig: _MessageConfig = { ...config, type: value };
    if (!messageInstance) {
      messageInstance = new MessageManger(newConfig, appContext);
    }
    return messageInstance.add(newConfig);
  };
  return pre;
}, {} as MessageMethod);
message.clear = () => {
  if (messageInstance) {
    messageInstance?.clear();
  }
};
const Message = {
  ...message,
  myContext: null as AppContext | null,
};
// 上面這段代碼等同于
/* const message = {
  success: (config: MessageConfig, appContext?: AppContext) => {
    const newConfig: _MessageConfig = { ...config, type: 'success' };
    if (!messageInstance) {
      messageInstance = new MessageManger(newConfig, appContext);
    }
    return messageInstance.add(newConfig);
  },
  error: (config: MessageConfig, appContext?: AppContext) => {
    const newConfig: _MessageConfig = { ...config, type: 'error' };
    if (!messageInstance) {
      messageInstance = new MessageManger(newConfig, appContext);
    }
    return messageInstance.add(newConfig);
  },
  warn: (config: MessageConfig, appContext?: AppContext) => {
    const newConfig: _MessageConfig = { ...config, type: 'warn' };
    if (!messageInstance) {
      messageInstance = new MessageManger(newConfig, appContext);
    }
    return messageInstance.add(newConfig);
  },
  clear: () => {
    if (messageInstance) {
      messageInstance?.clear();
    }
  },
}; */
// 拋出去的是一個(gè)對(duì)象,每次調(diào)用 message.xxx() 都是調(diào)用的同一個(gè)對(duì)象,所以這是一個(gè)單例模式
export default message;

types.ts

import type { AppContext } from 'vue';
export interface MessageItem {
  id: number | string;
  title: string;
  prompt?: string;
  type: 'success' | 'error' | 'warn';
  duration?: number;
  closable?: boolean;
  onClose?: (id: number | string) => void;
}
export interface MessageConfig {
  prompt?: string;
  title: string;
  id?: string;
  closable?: boolean;
  duration?: number;
  onClose?: (id: number | string) => void;
  type?: 'success' | 'error' | 'warn';
}
export interface MessageReturn {
  close: () => void;
}
export interface MessageMethod {
  success: (config: MessageConfig, appContext?: AppContext) => MessageReturn;
  error: (config: MessageConfig, appContext?: AppContext) => MessageReturn;
  warn: (config: MessageConfig, appContext?: AppContext) => MessageReturn;
  remove: (id: string) => void;
  clear: () => void;
}

到此,一個(gè)使用Vue3+TS實(shí)現(xiàn)的簡(jiǎn)單版的Message組件就完成了。如果有更好的實(shí)現(xiàn)方法,歡迎在評(píng)論區(qū)討論。同時(shí)也歡迎各位大佬指出我的不足

以上就是基于Vue3+ts封裝一個(gè)簡(jiǎn)單版的Message組件的詳細(xì)內(nèi)容,更多關(guān)于Vue3+ts封裝Message組件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Vue入門之a(chǎn)nimate過渡動(dòng)畫效果

    Vue入門之a(chǎn)nimate過渡動(dòng)畫效果

    這篇文章主要介紹了Vue入門之a(chǎn)nimate過渡動(dòng)畫效果的相關(guān)資料,需要的朋友可以參考下
    2018-04-04
  • vue2單元測(cè)試環(huán)境搭建

    vue2單元測(cè)試環(huán)境搭建

    本篇文章給大家分享了vue2單元測(cè)試環(huán)境搭建的詳細(xì)步驟,對(duì)此有需要的朋友參考學(xué)習(xí)下。
    2018-05-05
  • Vue.js 3.x 中的響應(yīng)式數(shù)據(jù)ref 與 reactive詳解

    Vue.js 3.x 中的響應(yīng)式數(shù)據(jù)ref 與 reactive詳解

    ref 和 reactive 是 Vue.js 3 中用于創(chuàng)建響應(yīng)式數(shù)據(jù)的兩個(gè)關(guān)鍵函數(shù),它們分別適用于不同類型的數(shù)據(jù),幫助我們更好地組織和管理組件的狀態(tài),這篇文章主要介紹了Vue.js 3.x 中的響應(yīng)式數(shù)據(jù):ref 與 reactive,需要的朋友可以參考下
    2024-01-01
  • vue項(xiàng)目實(shí)現(xiàn)路由跳轉(zhuǎn)到新頁(yè)面,返回舊頁(yè)面,保留之前的數(shù)據(jù)記錄(操作代碼)

    vue項(xiàng)目實(shí)現(xiàn)路由跳轉(zhuǎn)到新頁(yè)面,返回舊頁(yè)面,保留之前的數(shù)據(jù)記錄(操作代碼)

    這篇文章主要介紹了vue項(xiàng)目實(shí)現(xiàn)路由跳轉(zhuǎn)到新頁(yè)面,返回舊頁(yè)面,保留之前的數(shù)據(jù)記錄,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-09-09
  • Vue插槽具體用法及實(shí)例分析

    Vue插槽具體用法及實(shí)例分析

    本文主要介紹了Vue框架中插槽的使用方法和應(yīng)用場(chǎng)景。通過具體實(shí)例分析,詳細(xì)講解了插槽的具體用法,幫助讀者深入理解Vue中插槽的使用和實(shí)現(xiàn)方式
    2023-05-05
  • 關(guān)于elementUI select控件綁定多個(gè)值(對(duì)象)

    關(guān)于elementUI select控件綁定多個(gè)值(對(duì)象)

    這篇文章主要介紹了關(guān)于elementUI select控件綁定多個(gè)值(對(duì)象),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-04-04
  • 詳解如何運(yùn)行vue項(xiàng)目

    詳解如何運(yùn)行vue項(xiàng)目

    這篇文章主要介紹了如何運(yùn)行vue項(xiàng)目,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • Vue組件中的自定義事件你了解多少

    Vue組件中的自定義事件你了解多少

    這篇文章主要為大家詳細(xì)介紹了Vue組件中的自定義事件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-02-02
  • 關(guān)于vue的路由模式及修改方法

    關(guān)于vue的路由模式及修改方法

    這篇文章主要介紹了關(guān)于vue的路由模式及修改方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • Vue3 響應(yīng)式高階用法之triggerRef()的使用

    Vue3 響應(yīng)式高階用法之triggerRef()的使用

    在Vue3響應(yīng)式系統(tǒng)中,shallowRef僅追蹤頂層屬性的變化,當(dāng)需要對(duì)內(nèi)層屬性作出反應(yīng)時(shí),可使用triggerRef()方法手動(dòng)觸發(fā)更新,本文介紹了triggerRef()的應(yīng)用場(chǎng)景、基本用法、功能和最佳實(shí)踐,感興趣的可以了解一下
    2024-09-09

最新評(píng)論