記錄--Vue3自定義一個Hooks,實現一鍵換膚

来源:https://www.cnblogs.com/smileZAZ/archive/2023/05/30/17444051.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 核心 使用CSS變數, 準備兩套CSS顏色, 一套是在 light模式下的顏色,一套是在dark模式下的顏色 dark模式下的 CSS 權重要比 light 模式下的權重高, 不然當我們給html添加自定義屬性[data-theme='d ...


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

核心

  • 使用CSS變數, 準備兩套CSS顏色, 一套是在 light模式下的顏色,一套是在dark模式下的顏色
  • dark模式下的 CSS 權重要比 light 模式下的權重高, 不然當我們給html添加自定義屬性[data-theme='dark']的時候, dark模式權重比light低,會一直不起效果
  • 當我們點擊 dark 模式的時候, 給 html 設置自定義屬性[data-theme='dark']
  • 當我們點擊 light 模式的時候, 給 html 設置自定義屬性[data-theme='light']
  • 在 dark 模式下, 會匹配到html[data-theme='dark']選擇器下的樣式
  • 在 light 模式下,由於我們沒有設置html[data-theme='light']的方案, 那麼他就匹配:root(即html)下的樣式

兩套樣式代碼大概如下(列了一部分):

:root {
  --color-body-bg: #ffffff;
  --color-text: #000;
  --color-secondary-bg-for-transparent: rgba(209, 209, 214, 0.28);
}

html[data-theme='dark'] {
  --color-body-bg: #222222;
  --color-text: #ffffff;
  --color-primary-bg-for-transparent: rgba(255, 255, 255, 0.12);
}
然後我們點擊的時候,通過
 let theme = 'light'            // light / dark  
 document.documentElement.setAttribute('data-theme', theme)

這樣就能實現簡單的更換膚色功能了

什麼? 你以為這就完了?好戲剛開始

跟隨系統顏色

首先利用Window 的 'matchMedia()' 方法返回一個新的 MediaQueryList 對象,表示指定的媒體查詢 (en-US)字元串解析後的結果。

如運行媒體查詢(max-width: 600px)併在<span>;中顯示MediaQueryListmatches屬性值。如果視口的寬度小於或等於 600 像素,則輸出將為 true,而如果視窗的寬度大於此寬度,則將輸出 false。

<span class="mq-value"></span>
let mql = window.matchMedia('(max-width: 600px)');
document.querySelector(".mq-value").innerText = mql.matches; //此時小於或等於600像素時span 裡面的結果為false

利用prefers-color-scheme [CSS媒體特性] 用於檢測用戶是否有將系統的主題色設置為亮色或者暗色。

.day   { background: #eee; color: black; }
.night { background: #333; color: white; }

@media (prefers-color-scheme: dark) {
  .day.dark-scheme   { background:  #333; color: white; }
  .night.dark-scheme { background: black; color:  #ddd; }
}

@media (prefers-color-scheme: light) {
  .day.light-scheme   { background: white; color:  #555; }
  .night.light-scheme { background:  #eee; color: black; }
}

兩者相結合

matchMedia()prefers-color-scheme 結合在一起, 我們就可以通過 js 去給系統顏色為dark或 light 的情況下更換對應的 html自定義屬性, 即[data-theme='dark'][data-theme='light']

首先,我們先去獲取主題顏色, 我們還沒設置的時候,就預設是系統顏色, 設置了就把他存儲起來,下次直接獲取這個顏色

// 獲取主題變數
let appearance = ref<string>(localStorage.getItem('appearance') || 'auto')

// 查詢當前系統主題顏色
const match = window.matchMedia("(prefers-color-scheme: dark)")

// 如果主題變數為 auto, 則跟隨系統主題
if (appearance.value === 'auto') {
    followSystem()
} else {
    document.documentElement.setAttribute('data-theme', appearance.value)
}

function followSystem() {
    // 當前系統顏色是亮色還是暗色 , 設置對應的html[data-theme= 'dark' 或者'light']
    const theme = match.matches ? 'dark' : 'light'
    document.documentElement.setAttribute('data-theme', theme)
}

// 監聽系統主題變化,電腦主題發生改變的時候就調用followSystem函數
match.addEventListener('change', followSystem)

封裝成一個hooks

暴露出一個 useThemeColor函數, 返回一個對象, 對象裡面返回我們的主題變數

/ 獲取主題變數
let appearance = ref<string>(localStorage.getItem('appearance') || 'auto')
// 查詢當前系統主題顏色
const match:MediaQueryList = window.matchMedia("(prefers-color-scheme: dark)")
// 監聽系統主題變化
match.addEventListener('change', followSystem)

function followSystem() {
    const theme = match.matches ? 'dark' : 'light'
    document.documentElement.setAttribute('data-theme', theme)
}

watchEffect(() => {
// 如果主題變數為 auto, 則跟隨系統主題
    if (appearance.value === 'auto') {
        followSystem()
    } else {
        document.documentElement.setAttribute('data-theme', appearance.value)
    }
})


export default function useThemeColor() {
    return {
        appearance,
    }
}

使用hooks

導入我們export出來的函數

import useThemeColor from '../hooks/useThemeColor'

使用函數,註意, 這裡返回的 apprance 已經是一個響應式數據了

const { appearance } = useThemeColor()

使用 v-model 綁定apprance,直接使用apprance , 當我們切換顏色的時候, 就會調用watchEffect裡面的函數, 達到一鍵換膚效果

 <div class="item">
     <div class="left">
         <div class="title">外觀</div>
     </div>
     <div class="right">
         <select v-model="appearance">
             <option value="auto">{{ "自動" }}</option>
             <option value="light">

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

-Advertisement-
Play Games
更多相關文章
  • 今天這個《1萬7千道法律職業考試題ACCESS資料庫》集收了海量的法考題庫試題,是從法律職業考試軟體取提出來的,讓你備考通關更加高效。今天這個《1萬7千道法律職業考試題ACCESS資料庫》集收了海量的法考題庫試題,是從法律職業考試軟體取提出來的,讓你備考通關更加高效。 包含分類:1.法理學(607) ...
  • 雖然之前弄到過《1萬多公務員考試基礎知識題庫ACCESS資料庫》,但完全沒有今天這份資料庫那麼美。今天這份數據是從一款考試學習類的軟體中破解提取出來的,據數非常不錯,不但有大小分類,而且題型包含:單項選擇題(25575條)、簡答題(942條)。 題庫中有些包含圖片問答,或者選項中有含圖片,如: 本數 ...
  • 摘要:究竟是不是cpu占比高的問題導致redis超時的呢? 本文分享自華為雲社區《我又和redis超時杠上了》,作者:藍胖子的編程夢 。 背景 經過上次redis超時排查,並聯繫雲服務商解決之後,redis超時的現象好了一陣子,但是最近又有超時現象報出,但與上次不同的是,這次超時的現象發生在業務高峰 ...
  • # Rollup ROLLUP 在多維分析中是“上捲”的意思,即將數據按某種指定的粒度進行進一步聚合。 通過建表語句創建出來的表稱為 Base 表(Base Table,基表) 在 Base 表之上,我們可以創建任意多個 ROLLUP 表。這些 ROLLUP 的數據是基於 Base 表產生的,並且在 ...
  • 推理題類的數據有一些,比如《1000道邏輯推理考題ACCESS資料庫》、《近5千偵探腦筋急轉彎選擇題ACCESS資料庫》等,但是今天遇到了一份有些圖片的推理題庫,感覺非常不錯,就是記錄數少了一些,請看以下截圖,截圖包含所有欄位,所有圖片放在一個文件夾中。 分類情況如下:邏輯推理(60)、腦筋急轉彎( ...
  • 截圖下方有顯示“共有記錄數”,截圖包含了表的所有欄位列。該數據提供ACCESS資料庫文件(擴展名是MDB)以及EXCEL文件(擴展名是XLS)。 共有23710條記錄,根據AUTHOR_ID關聯AUTHORS作者表中的ID欄位 包含6567個作者,根據ID關聯QUOTES表中的AUTHOR_ID欄位 ...
  • Health Kit文檔全新升級,開發場景更清晰,聚焦你關心的問題,快來一起嘗鮮! 文檔入口請戳:[文檔入口~](https://developer.huawei.com/consumer/cn/doc/development/HMSCore-Guides/description-000000155 ...
  • Flutter開發中三棵樹的重要性不言而喻,瞭解其原理有助於我們開發出性能更優的App,此文主要從源碼角度介紹Element樹的管理類BuildOwner。 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:本文代碼示例演示瞭如何在WPF中使用LiveCharts庫創建動態條形圖。通過創建數據模型、ViewModel和在XAML中使用`CartesianChart`控制項,你可以輕鬆實現圖表的數據綁定和動態更新。我將通過清晰的步驟指南包括詳細的中文註釋,幫助你快速理解並應用這一功能。 先上效果: 在 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • 概述:本示例演示了在WPF應用程式中實現多語言支持的詳細步驟。通過資源字典和數據綁定,以及使用語言管理器類,應用程式能夠在運行時動態切換語言。這種方法使得多語言支持更加靈活,便於維護,同時提供清晰的代碼結構。 在WPF中實現多語言的一種常見方法是使用資源字典和數據綁定。以下是一個詳細的步驟和示例源代 ...
  • 描述(做一個簡單的記錄): 事件(event)的本質是一個委托;(聲明一個事件: public event TestDelegate eventTest;) 委托(delegate)可以理解為一個符合某種簽名的方法類型;比如:TestDelegate委托的返回數據類型為string,參數為 int和 ...
  • 1、AOT適合場景 Aot適合工具類型的項目使用,優點禁止反編 ,第一次啟動快,業務型項目或者反射多的項目不適合用AOT AOT更新記錄: 實實在在經過實踐的AOT ORM 5.1.4.117 +支持AOT 5.1.4.123 +支持CodeFirst和非同步方法 5.1.4.129-preview1 ...
  • 總說周知,UWP 是運行在沙盒裡面的,所有許可權都有嚴格限制,和沙盒外交互也需要特殊的通道,所以從根本杜絕了 UWP 毒瘤的存在。但是實際上 UWP 只是一個應用模型,本身是沒有什麼許可權管理的,許可權管理全靠 App Container 沙盒控制,如果我們脫離了這個沙盒,UWP 就會放飛自我了。那麼有沒... ...
  • 目錄條款17:讓介面容易被正確使用,不易被誤用(Make interfaces easy to use correctly and hard to use incorrectly)限制類型和值規定能做和不能做的事提供行為一致的介面條款19:設計class猶如設計type(Treat class de ...
  • title: 從零開始:Django項目的創建與配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 後端開發 tags: Django WebDev Python ORM Security Deployment Op ...
  • 1、BOM對象 BOM:Broswer object model,即瀏覽器提供我們開發者在javascript用於操作瀏覽器的對象。 1.1、window對象 視窗方法 // BOM Browser object model 瀏覽器對象模型 // js中最大的一個對象.整個瀏覽器視窗出現的所有東西都 ...