記錄--Vue自定義指令實現載入中效果v-load(不使用Vue.extend)

来源:https://www.cnblogs.com/smileZAZ/archive/2023/03/09/17199475.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 網站效果演示:ashuai.work:8888/#/myLoad GitHub倉庫地址代碼:github.com/shuirongshu… 載入中思路分析 實現載入中效果,一般有兩種方式: 第一種是:搞一個load組件,然後使用Vue.e ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

Vue自定義指令實現載入中效果v-load(不使用Vue.extend)

網站效果演示:ashuai.work:8888/#/myLoad

GitHub倉庫地址代碼:github.com/shuirongshu…

載入中思路分析

實現載入中效果,一般有兩種方式:

  • 第一種是:搞一個load組件,然後使用Vue.extend()方法去繼承一個載入組件去使用,比如筆者的這篇文章:juejin.cn/post/702172…
  • 第二種是:直接使用指令去在需要載入的dom上去創建一個載入中的dom元素,並指定相應的樣式即可。本篇文章說的是第二種。

我們先看一下效果圖

v-load效果圖

v-load.gif

實現步驟一:加上自定義指令

假設我有一個dom元素,我給其加上一個自定義的指令v-load="loading",綁定一個具體的布爾值loading,用於控制開啟載入中和關閉載入中

<div class="box" v-load="loading">111</div>

loading: true

.box {
  width: 160px;
  height: 80px;
  border: 2px solid #666;
}

接下來,我就要在自定義指令的相關鉤子函數中去操作這個dom元素。

關於自定義指令的入門基礎知識可以看官方文檔,或者參見筆者之前的關於自定義指令的文章:juejin.cn/post/702960…

實現步驟二:給目標元素創建一個子元素dom用於載入

在自定義指令的初始化bind鉤子函數中,我們可以拿到這個dom元素,首先給這個目標元素開始相對定位,讓用於載入中的子元素dom去絕對定位,以這個相對定位的父元素進行參考

bind(el, binding) {
    const target = el;
    // 父元素相對定位
    target.style.position = "relative";
    // 子元素遮罩層部分
    let loadChild = document.createElement("div");
    loadChild.className = "loadClass";
}

上述代碼中給載入中的子元素loadChild指定一個樣式類名loadClass

在這裡小伙伴可能有疑問了,這個自定義指令的樣式怎麼寫呢?自定義指令中也沒有style標簽啊?

是的,自定義指令中不能直接寫樣式,不過沒關係,我們可以先寫好一個css樣式,然後引入過來使用啊,如下:

// 引入拆分的樣式,便於自定義指令中使用
import './index.css'

bind(el, binding) {
    ......
    loadChild.className = "loadClass";
}

這樣的話,className的樣式,可以在引入的同級目錄下的./index.css文件中設置了,loadClass樣式如下:

.loadClass {
    /* 寬高百分百 */
    position: absolute;
    top: 0;
    bottom: 0;
    left: 0;
    right: 0;
    /* 預設背景色和顏色 */
    background-color: rgba(255, 255, 255, 0.99);
    color: #0b6ed0;
    /* 透明度過渡使用搭配display:none; */
    opacity: 0.8;
    transition: all 0.3s;
    /* 居中 */ 
    display: flex;
    align-items: center;
    justify-content: center;
}

註意,載入中效果開啟和消失,不用使用vue自帶的過渡組件transition,咱們可以使用透明度搭配搭配display:none;去設置

註意,載入中要以父元素為邊界去控制,可不能滿屏載入哦

然後初始化的時候,看看v-load綁定的值是true還是false,同時加上一個用於隱藏的類名:load-hide。再把這個載入中的dom元素追加到目標父元素身上。

loadChild.classList.add('load-hide') // 添加類名
target.appendChild(loadChild); // 追加載入中子元素

/* 通過透明度實現過渡動畫 */
.load-hide {
    opacity: 0;
}

這樣的話,初始化的載入中就做好了。v-load綁定的值是false的時,就隱藏之

實現步驟三:當組件更新時,去添加或移除這個load-hide類名

componentUpdated(el, binding, vnode, oldVnode) {
        let flag = binding.value
        let loadChild = el.querySelector('.loadClass')
        if (flag) { // v-load綁定的值為true,就移除這個類名,就能看到了
            loadChild.style.display = 'flex'
            setTimeout(() => {
                loadChild.classList.remove('load-hide')
            }, 0);
        } else { // 綁定的值為false時,再添加這個類名,同時隱藏dom
            loadChild.classList.add('load-hide')
            setTimeout(() => {
                loadChild.style.display = 'none'
            }, 360);

        }
    },

註意上述代碼中為啥不直接隱藏,而是使用定時器延長360毫秒再去隱藏,因為筆者設置的載入dom的過渡時間是0.3秒,即要等到透明度過渡完了以後,再隱藏載入中dom,這樣看著就自然一些了。

.loadClass { transition: all 0.3s; }

到這裡,咱們的v-load自定義指令的載入中效果,就初步完成了。不過功能有點少,自定義載入中我還想,去動態控制:

  • 自定義載入圖標名
  • 自定義載入文字
  • 自定義載入文字顏色
  • 自定義載入背景色

那這樣怎麼辦呢?

實現步驟四:優化自定義指令,支持傳入更多的參數

此時,我的v-load自定義指令,就不用綁定一個布爾值了。可以考慮綁定一個對象啊,通過控制這個對象的具體值,來動態控制載入中的效果。

  • 原來自定義指令綁定:v-load = loading // typeof loading == 'boolean'
  • 現在自定義指令綁定:v-load = loading2 // typeof loading2 == 'object'
<div class="box" v-load="loading2">222</div>

// 如果想要有更多的配置項,就傳一個對象,註意要指定欄位
loading2: {
    value: true,
    icon: "el-icon-eleme", // 自定義載入圖標名
    text: "客官稍等哦...", // 自定義載入文字
    color: "red", // 自定義載入文字顏色
    bgColor: "#baf", // 自定義載入背景色
}

傳參指定相應欄位,自定義指令中接參,就要在相關的鉤子中去接收並處理這些參數。

如何處理這些參數?

  • v-load綁定的值的類型的判斷,是布爾值,還是對象,執行不同的操作
  • 使用原生js的方式去,創造dom元素、給dom元素指定類名(或添加刪除類名)
  • 考慮到性能緣故,可以使用文檔碎片優化document.createDocumentFragment()
  • 最後丟入遮罩層dom內部即可

完整代碼

自定義指令樣式文件index.css

.loadClass {
    /* 寬高百分百 */
    position: absolute;
    top: 0;
    bottom: 0;
    left: 0;
    right: 0;
    /* 預設背景色和顏色 */
    background-color: rgba(255, 255, 255, 0.99);
    color: #0b6ed0;
    /* 透明度過渡使用搭配display:none; */
    opacity: 0.8;
    transition: all 0.3s;
    /* 禁止文字選擇 */
    user-select: none;
    display: flex;
    align-items: center;
    justify-content: center;
}

/* 通過透明度實現過渡動畫 */
.load-hide {
    opacity: 0;
}

.loadClass>i {
    margin-right: 4px;
}

自定義指令的js文件index.js

註意,自定義指令還需要註冊一下才能使用哦

// 引入拆分的樣式,便於自定義指令中使用
import './index.css'
export default {
    // 初始化綁定dom鉤子函數
    bind(el, binding) {
        const target = el;
        // 傳參類型判斷變數控制
        let flag;
        let isObj;
        if (typeof binding.value == 'boolean') {
            flag = binding.value
            isObj = false
        }
        if (typeof binding.value == 'object') {
            flag = binding.value.value
            isObj = true
        }
        // 有dom元素才去做操作
        if (target) {
            // 父元素相對定位
            target.style.position = "relative";
            // 子元素遮罩層部分
            let loadChild = document.createElement("div");
            loadChild.className = "loadClass";

            // 創建文檔碎片性能稍微優化一點點
            let fragment = document.createDocumentFragment()
            // 孫子元素1載入圖標部分
            let iSun = document.createElement("i");
            iSun.className = isObj ? binding.value.icon : "el-icon-loading";
            // 孫子元素2文字提示部分
            let spanSun = document.createElement("span");
            spanSun.innerHTML = isObj ? binding.value.text : '載入中...'
            // 使用文檔碎片將iSun和spanSun裝起來
            fragment.appendChild(iSun);
            fragment.appendChild(spanSun);
            // 將文檔碎片丟入子元素遮罩層內
            loadChild.appendChild(fragment);
            // 樣式判斷設置
            if (isObj) {
                loadChild.style.color = binding.value.color
                loadChild.style.backgroundColor = binding.value.bgColor
            }
            // 若為false,就隱藏
            if (!flag) {
                loadChild.classList.add('load-hide')
                loadChild.style.display = 'none'
            }

            // 父元素添加子元素遮罩層使用
            target.appendChild(loadChild);
        }
    },
    // dom組件更新操作控制顯示和隱藏
    componentUpdated(el, binding, vnode, oldVnode) {
        let flag = typeof binding.value == 'boolean' ? binding.value : binding.value.value
        let loadChild = el.querySelector('.loadClass')
        if (flag) {
            loadChild.style.display = 'flex'
            setTimeout(() => {
                loadChild.classList.remove('load-hide')
            }, 0);
        } else {
            loadChild.classList.add('load-hide')
            setTimeout(() => {
                loadChild.style.display = 'none'
            }, 360);

        }
    },
}

使用自定義load指令的地方

<template>
  <div>
    <h3>指令方式載入中...</h3>
    <br />
    <button @click="loadFn">點一下</button>
    <br />
    <br />
    <el-table v-load="loading" border :data="tableData" style="width: 80%">
      <el-table-column prop="name" label="姓名"></el-table-column>
      <el-table-column prop="age" label="年齡"></el-table-column>
      <el-table-column prop="home" label="家鄉"></el-table-column>
      <el-table-column prop="like" label="愛好"></el-table-column>
    </el-table>
    <br />
    <div class="box" v-load="loading">111</div>
    <br />
    <div class="box" v-load="loading2">222</div>
  </div>
</template>

<script>
export default {
  name: "myLoadName",
  data() {
    return {
      // 自定義指令方式,預設綁定的值是布爾值
      loading: true,
      // 如果想要有更多的配置項,就傳一個對象,註意要指定欄位
      loading2: {
        value: true,
        icon: "el-icon-eleme", // 自定義載入圖標名
        text: "客官稍等哦...", // 自定義載入文字
        color: "red", // 自定義載入文字顏色
        bgColor: "#baf", // 自定義載入背景色
      },
      tableData: [
        {
          name: "孫悟空",
          age: 500,
          home: "花果山水簾洞",
          like: "桃子",
        },
        {
          name: "豬八戒",
          age: 88,
          home: "高老莊",
          like: "肉包子",
        },
        {
          name: "沙和尚",
          age: 1000,
          home: "通天河",
          like: "游泳",
        },
      ],
    };
  },
  methods: {
    loadFn() {
      this.loading = !this.loading;
      this.loading2.value = !this.loading2.value;
    },
  },
};
</script>

<style lang='less' scoped>
.box {
  width: 160px;
  height: 80px;
  border: 2px solid #666;
  box-sizing: border-box;
}
</style>

本文轉載於:

https://juejin.cn/post/7182375025368891429

如果對您有所幫助,歡迎您點個關註,我會定時更新技術文檔,大家一起討論學習,一起進步。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1 mysql邏輯架構 mysql邏輯架構圖: Mysql伺服器、存儲引擎 是兩個獨立的組件,彼此通過api交互 第一層:連接處理、授權認證、安全管理 第二層:核心服務功能 查詢解析、分析、優化、緩存以及所有的內置函數(日期、時間、數學、加密函數等) 跨存儲引擎的功能:存儲過程、觸發器、視圖等。 第 ...
  • 數據治理是推動大型集團企業轉型升級、提升競爭優勢、實現高質量發展的重要引擎。通過全鏈數據結構化,實現業務對象、業務規則、業務流程數字化,推進全鏈業務深度數字化,夯實數據運營底座。 廈門國貿集團股份有限公司(簡稱“國貿股份”)是國有控股上市公司,同時也是首批全國供應鏈創新與應用示範企業,在“十四五”規 ...
  • 摘要:其實游戲客戶對資料庫的訴求是很明確的,資料庫應當“放心存放心用”。 本文分享自華為雲社區《華為雲GaussDB(for Redis)揭秘第27期:聊聊游戲業務怎麼用高斯Redis》,作者:高斯Redis官方博客。 華為雲資料庫團隊是比較重視技術洞察的,對客戶真實的業務場景也比較看重。年初出差了 ...
  • GreatSQL社區原創內容未經授權不得隨意使用,轉載請聯繫小編並註明來源。 GreatSQL是MySQL的國產分支版本,使用上與MySQL一致。 作者: 花家舍 文章來源:GreatSQL社區原創 前文回顧 實現一個簡單的Database系列 譯註:cstack在github維護了一個簡單的、類似 ...
  • 生活中存在同時使用兩個微信的情況,一個工作一個生活,這時希望同時在電腦上登錄兩個賬號。如何做到呢?步驟如下: 右鍵單擊“微信”圖標,選擇屬性,目標框內的路徑就是微信安裝路徑,複製目標框里的內容。 將如下命令複製到 TXT 文件保存,再將該文件重命名,主要是將尾碼名改成“.bat”文件。 @echo ...
  • 好家伙,本篇為做題思考 書接上文 題目如下: 1.請給出下列代碼的輸出結果,並配合"消息隊列"寫出相關解釋 async function foo() { console.log(2); console.log(await Promise.resolve(8)); console.log(9); } ...
  • 一般來說,項目由子模塊組成,拿到後端提供過來的介面,一般也是按照子模塊來分類提供的.請教一下各位,你們前端項目是如何管理api的? 希望各位貼點你們的優秀代碼段上來學習學習. 常見: 各個模塊的api存放到單獨的js文件里,返回一個請求實例promise對象 使用的時候根據需求引入相應的請求方法 / ...
  • 其他章節請看: webgl 系列 變換矩陣和動畫 動畫就是不停地將某個東西變換(transform)。例如將三角形不停地旋轉就是一個動畫 和 CSS transform 類似,變換有三種形式:平移、縮放和旋轉。 簡單的變換用普通表達式容易實現,如果事情複雜,比如旋轉後平移,這時就可以使用變換矩陣。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...