vue3 vite2 封裝 SVG 圖標組件 - 基於 vite 創建 vue3 全家桶項目續篇

来源:https://www.cnblogs.com/youyacoder/archive/2022/10/04/16754511.html
-Advertisement-
Play Games

在 《基於 vite 創建 vue3 全家桶》一文整合了 *Element Plus*,並將 *Element Plus* 中提供的圖標進行全局註冊,這樣可以很方便的延續 *Element UI* 的風格 —— 通過 *el-icon-xxx* 的方式使用圖標(如果有問題的朋友可以先閱讀前面的文章:... ...


《基於 vite 創建 vue3 全家桶》一文整合了 Element Plus,並將 Element Plus 中提供的圖標進行全局註冊,這樣可以很方便的延續 Element UI 的風格 —— 通過 el-icon-xxx 的方式使用圖標(如果有問題的朋友可以先閱讀前面的文章:基於 Vite 創建 vue3 全家桶項目)。

在真實的企業級開發中,Element Plus 內置的圖標通常很難滿足業務需求,項目中需要引入大量的 SVG 圖標資源,本文描述如何在 Vue3 + Vite2 環境中使用 SVG 圖標,封裝一個支持本地 SVG 圖標和線上 SVG 圖標的組件 svg-icon

文中實現的 svg-icon 組件會內置到 yyg-cli 腳手架中(通過 yyg 命令行創建的項目內置該組件),如果在此之前已經通過 yyg-cli 創建了項目,可以按照本文的步驟封裝 SVG 圖標組件 svg-icon

1 創建組件

src/components/ 目錄下創建目錄 svg-icon,該在目錄中創建 svg-icon 組件 index.vue

1.1 輸入屬性

該組件需要兩個輸入屬性(props):

  • icon:SVG 圖標的名稱或線上 URL
  • className:動態傳遞給該組件的樣式類名

代碼如下:

const props = defineProps({
  // SVG 圖標名稱或線上URL
  icon: {
    type: String,
    required: true
  },
  // 圖標類名
  className: {
    type: String,
    default: ''
  }
})

1.2 SVG 圖標樣式

style 中定義 svg-icon 的樣式類:

.svg-icon {
  width: 1em;
  height: 1em;
  fill: currentColor;
  overflow: hidden;
}

2 線上 SVG 圖標

svg-icon 組件需要支持線上 SVG 圖標本地 SVG 圖標。首先實現線上 SVG 圖標的顯示。如下 URL 為一個線上 SVG 圖標,可在瀏覽器中直接訪問:

http://www.yygnb.com/demo/car.svg

2.1 判斷線上圖標

script 中通過計算屬性判斷 props 中的 icon 是否是線上圖標:

const isOnlineSvg = computed(() => /^(https?:)/.test(props.icon))

該判斷比較簡單,如果 icon 屬性以 http:https: 開頭,則該圖標為線上圖標,其他情況均為本地的 SVG 圖標。各位朋友可以根據自己項目情況添加或完善該判斷邏輯。

2.2 模板和樣式

線上 SVG 圖標通過 HTML 元素 div 來顯示,css3 有個 mask 屬性,該屬性表示遮罩,可以部分或者完全隱藏一個元素的可見區域,使用方式與 background 很類似。

template 如下:

<div v-if="isOnlineSvg"
     :style="{ '--svg-icon-url': `url(${icon})` }"
     class="svg-icon svg-icon-online"
     :class="className"/>

style 追加 svg-icon-online 樣式類:

.svg-icon-online {
  background-color: currentColor;
  mask-image: var(--svg-icon-url);
  -webkit-mask-image: var(--svg-icon-url);
  mask-size: cover;
  -webkit-mask-size: cover;
  display: inline-block;
}

上面的 templatestyle 使用到 vue3 的新特性,演示瞭如何將一個 script 中的 props 屬性傳遞給 scss“:

  1. 首先在模板中通過 style 屬性定義了一個變數 --svg-icon-url,該變數的值為 props 中的 icon 屬性。
  2. 在 scss 中設置 mask-image 時,使用 var 函數獲取變數 --svg-icon-url 的值。

3.3 測試線上圖標

about.vue 中引入 svg-icon

import SvgIcon from '@/components/svg-icon/index.vue'

測試使用該組件:

<div>
  <svg-icon class-name="icon" icon="http://www.yygnb.com/demo/car.svg"></svg-icon>
</div>

添加自定義樣式:

.icon {
  color: cornflowerblue;
  font-size: 30px;
}

在瀏覽器中訪問 about 頁面,可以看到線上 SVG 圖標可以成功顯示:

image-20221003225232929

3 本地 SVG 圖標

在 webpack 中載入 svg 資源可以使用 svg-sprite-loader,而 vite 中可以使用插件 vite-plugin-svg-icons

3.1 安裝開發依賴

首先安裝 vite-plugin-svg-icons 為開發依賴:

yarn add vite-plugin-svg-icons -D

3.2 配置 vite

vite.config.ts 中配置該插件:

...
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons'
...

export default defineConfig({
	...
  plugins: [
    ...
    createSvgIconsPlugin({
      // 要緩存的圖標文件夾
      iconDirs: [path.resolve(__dirname, 'src/svg')],
      // 執行 icon name 的格式
      symbolId: 'icon-[name]'
    })
  ],
  ...
}

通過 createSvgIconsPlugin() 入參指定了svg 文件所在的目錄和 symbolId

3.3 修改 main.ts

main.ts 中添加如下語句:

import 'virtual:svg-icons-register'

3.4 完成 svg-icon 組件

通過上述步驟,便完成了 vite-plugin-svg-icons 的配置,接下來實現 svg-icon 組件即可。前面已經完成了線上 svg、樣式等,現在只需要在 template 中補充本地 svg 的代碼即可:

<svg v-else
     class="svg-icon"
     :class="className"
     aria-hidden="true">
  <use :xlink:href="`#icon-${icon}`"/>
</svg>

組件 components/svg-icon/index.vue 完整代碼如下:

<template>
  <div v-if="isOnlineSvg"
       :style="{ '--svg-icon-url': `url(${icon})` }"
       class="svg-icon svg-icon-online"
       :class="className"/>
  <svg v-else
       class="svg-icon"
       :class="className"
       aria-hidden="true">
    <use :xlink:href="`#icon-${icon}`"/>
  </svg>
</template>

<script lang="ts" setup>
import { computed } from 'vue'

const props = defineProps({
  // SVG 圖標名稱或線上URL
  icon: {
    type: String,
    required: true
  },
  // 圖標類名
  className: {
    type: String,
    default: ''
  }
})

const isOnlineSvg = computed(() => /^(https?:)/.test(props.icon))
</script>

<style scoped lang="scss">
.svg-icon {
  width: 1em;
  height: 1em;
  fill: currentColor;
  overflow: hidden;
}

.svg-icon-online {
  background-color: currentColor;
  mask-image: var(--svg-icon-url);
  -webkit-mask-image: var(--svg-icon-url);
  mask-size: cover;
  -webkit-mask-size: cover;
  display: inline-block;
}
</style>

3.5 測試本地圖標

由於 vite.config.ts 中配置的 svg 目錄為 src/svg,首先將 car.svg 拷貝到該目錄下。繼續在 about.vue 中添加如下代碼:

<div>
  <svg-icon icon="http://www.yygnb.com/demo/car.svg"></svg-icon>
  <svg-icon icon="car"></svg-icon>

  <svg-icon class-name="icon" icon="http://www.yygnb.com/demo/car.svg"></svg-icon>
  <svg-icon class-name="icon" icon="car"></svg-icon>
</div>

上面的代碼分別顯示線上圖標和本地圖標,頁面顯示結果如下:

image-20221003231820378

可以看出線上圖標、本地圖標、自定義樣式類都可以正常顯示,這樣便完成了 svg-icon 的封裝。

感謝你閱讀本文,如果本文給了你一點點幫助或者啟發,還請三連支持一下,點贊、關註、收藏,作者會持續與大家分享更多乾貨


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

-Advertisement-
Play Games
更多相關文章
  • JPA 即Java Persistence API。是一款持久層框架,中文名Java持久層API,是JDK 5.0註解或XML描述對象-關係表的映射關係,並將運行期的實體對象持久化到資料庫中。 ...
  • 漢諾塔問題 在經典漢諾塔問題中,有 3 根柱子及 n 個不同大小的穿孔圓盤,盤子可以滑入任意一根柱子。一開始,所有盤子自上而下按升序依次套在第一根柱子上(即每一個盤子只能放在更大的盤子上面)。移動圓盤時受到以下限制: (1) 每次只能移動一個盤子; (2) 盤子只能從柱子頂端滑出移到下一根柱子; ( ...
  • 一、進行線程池創建 import cn.hutool.core.thread.ThreadFactoryBuilder; import lombok.extern.slf4j.Slf4j; import org.springframework.aop.interceptor.AsyncUncaugh ...
  • 一. 怎麼開啟斷點調試? 隨著開發的深入,越來越覺得高效的調試方法是多麼的重要了,但我們一般上來就是敲一些代碼,誰會去靜下心來學一些看似沒什麼用的調試技巧呢?但這恰恰就是新手和老手之間的區別。 斷點調試是很簡單的,只需要點擊idea上方的小蟲子,啟動調試即可,如下所示。 這當然不是本文的重點,只是開 ...
  • 一:背景 1. 背景 前段時間有位朋友咨詢說他的程式出現了非托管記憶體泄漏,說裡面有很多的 HEAP_BLOCK 都被標記成了 Internal 狀態,而且 size 都很大, 讓我幫忙看下怎麼回事? 比如下麵這樣。 1cbea000: 42000 . 42000 [101] - busy (41fe ...
  • U盤自動讀寫的小玩意 共有四種方法(我知道的方法,全是轉載。轉載也很不易,可望給個硬幣) 方法一(vbs方法 全自動,轉載自bilibili 點我跳轉)文件下載鏈接(點我下載) 方法二(cmd方法 需手動,轉載自bilibili 點我跳轉)文件下載鏈接(點我下載) 方法三(python方法 全自動, ...
  • 5.MySQL常用函數 5.1合計/統計函數 5.1.1合計函數-count count 返回行的總數 Select count(*)|count (列名) from table_name [WHERE where_definition] 練習 -- 統計一個班級共有幾個學生 SELECT COUN ...
  • 就像在學習之前先要識字,我想在介紹優化 JavaScript 代碼之前,先介紹一下自己對編程語言的理解。故事要從一隻叫做 Theseus 的機械鼠和其發明人克勞德-香農(Claude Shannon)說起。在傳記《A Mind at Play:How Claude Shannon Invented ... ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...