问题解决|使用Less变量和媒体查询实现深浅色模式适配

grtsinry43
6/10/2024
137 views
预计阅读时长 13 分钟

AI Summary

Powered By DeepSeek-R1
|

使用Less变量简化css写法,并不影响css变量和媒体查询在深浅模式切换时的效果

<!--more-->

分析问题

首先要清楚问题的来源是什么,使用媒体查询api可以根据深浅色为css变量赋予不同的值,而Less(或是scss)不是原生被浏览器所支持,运行时需要先转为为原生css,这一步也就是编译,有点类似ts和js的关系。这里采用的是vite创建的Vue3项目
而Less/Scss在编译的过程中,进行的是完全的字段替换,这也就意味着在运行之前css已经被固定,无法在运行的过程中动态修改,当然也不支持媒体查询的所有属性,但一些还是支持的哈,比如设备类型,最大宽度什么的,这里深浅色默认是不支持的。

问题出现

这里首先我是想办法完全使用Less的,也就是媒体查询+正常定义变量

less
1@primary: #1890ff; // 全局主色
2@bg: #ffffff; // 全局背景
3@font: #333333; // 全局字体颜色
4@warning: #faad14; // 警告
5@success: #52c41a; // 成功
6@error: #f5222d; // 错误
7@info: #1890ff; // 信息
8@disabled: #bfbfbf; // 失效
9@link: #1890ff; // 链接
10@hover: #001764; // hover

但是实际测试中发现所编译出的css根本没有@media的字样,所以又是inline编译了,导致完全写死,不可能在运行中改变

问题解决

因为测试Less写法一直有问题,而原生css是可以完美解决的,为了既保留Less的简便的变量写法,又解决变量无法修改的问题,因此利用字段替换编译的特性,我们可以将Less变量赋值为css变量,这样当编译之后所生成的css,依然是变量形式储存内容,具体代码可以参考:

colors.less

less
1:root {
2  --primary: #1890ff; // 全局主色
3  --bg: #ffffff; // 全局背景
4  --font: #333333; // 全局字体颜色
5  --warning: #faad14; // 警告
6  --success: #52c41a; // 成功
7  --error: #f5222d; // 错误
8  --info: #1890ff; // 信息
9  --disabled: #bfbfbf; // 失效
10  --link: #1890ff; // 链接

global.less

less
1@import "@/styles/colors";
2
3*,
4*::before,
5*::after {
6  box-sizing: border-box;
7  margin: 0;
8  padding: 0;
9}
10

随后正常在main.js引入global.less即可:

js
1import '@/styles/global.less'

问题延伸

问题到这里解决了吗?当然,但是还有个需求场景,某些网站会提供用户手动切换深色浅色模式的功能,而css变量可在运行中动态修改的优势就显现了出来

这里就需要js在运行中更改变量啦,具体的实现逻辑如下:

这里采用触发事件的方式进行,在Vue3中,你可以使用mitt进行事件的监听,

shell
1npm install mitt

可以直接注册在全局app上,也可以单独延续Vue2的习惯创建Vue实例挂载到上面:

这里的举例是创建Vue实例挂载到上面,新建一个eventBus.js

js
1import { createApp } from 'vue';
2import mitt from 'mitt';
3
4const emitter = mitt();
5
6const bus = createApp({});
7
8bus.config.globalProperties.$bus = emitter;
9
10export default bus;

这样就可以全局触发和监听事件

接下来创建一个工具函数,在挂载时候添加监听,新建util/useColorScheme.js

js
1import { onMounted, onUnmounted } from 'vue';
2import bus from "@/eventBus.js";
3/**
4 * 当触发themeChange事件,切换页面的颜色主题,在onMounted,onUnmounted生命周期钩子函数中设置
5 * 修改原理是css变量
6 */
7export function useColorScheme() {
8    const themeChange = (theme) => {
9        console.log('themeChange', theme)
10        if (theme) {

这个就是在当触发themeChange事件,切换页面的颜色主题,在onMounted,onUnmounted生命周期钩子函数中设置,工具都准备好啦,就可以在任意地方使用啦

js
1import {ref, watchEffect} from 'vue';
2import {useColorScheme} from "@/util/useColorScheme.js";
3import bus from "@/eventBus.js";
4
5// 定义深浅模式的变量
6const isDark = ref(false);
7
8// 切换深浅模式
9const handleThemeButton = () => {
10  isDark.value = !isDark.value;

总结一下

其实有点复杂不太优雅,不过在大的Vue项目中,一定会用到pinia,mitt等工具来进行状态管理,全局事件总线监听的,折腾一下利于积累经验()

回到最开始的问题,原生css变量可用于动态修改,给了实时切换的可能,而less/scss的便捷写法又能大大简化开发,好啦,就到这里呀

效果演示

COPYRIGHT
作者grtsinry43
版权年份© 2024
许可协议

问题解决|使用Less变量和媒体查询实现深浅色模式适配》采用知识共享署名 4.0 国际许可协议

转载请注明出处并遵循 CC BY 许可协议条款

相关推荐

折腾记录|使用 Nuxt.js 重写个人主页,使用 SSR 优化 SEO ,实现一些期待已久的效果

在 22 年刚创建个人主页的时候,由于我的技术水平不够,只能用一些 wordpress type...

grtsinry43
9/19/2024
227
0
1

使用 Spring Boot + MyBatisPlus 提高效率,简化开发

最近一段时间,由于一些项目的需求,于是被迫用很快的速度学完了 Spring Framework,...

grtsinry43
8/11/2024
109
0
0
使用 pf4j-spring 实现插件注入和 api 接口动态注册 | 插件系统构建(上)

使用 pf4j-spring 实现插件注入和 api 接口动态注册 | 插件系统构建(上)

哪个男孩不想拥有一个自己的插件系统?(x)话说回来,这个我已经计划好久了,不过一直在学其他的东西,刚...

grtsinry43
1/26/2025
511
0
5

学习分享|原生三件套的网页效果

这是学习前端基础,用原生三件套尤其是js实现效果一些学习过程的记录,首先是静态的小米商城,之后实...

grtsinry43
6/10/2024
85
0
0

学习分享|跨域解决、安卓开发探索、油猴脚本探索

最近学习的一些内容,包括跨域问题及其解决方案,安卓开发的简单探索,OpenAI的api做了个小插...

grtsinry43
6/10/2024
105
2
2
COMMENT 7273217729056870400

发表评论

来这里畅所欲言吧!
支持 Markdown 语法 0 / 3000

网站运行时间

0
0
0
0

在风雨飘摇之中

感谢陪伴与支持

愿我们不负热爱,继续前行