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
  • 移動開發(一):使用.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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...