前端主题切换的多种方式

动态link标签加载不同主题css

**原理:**提前准备好几套CSS主题样式文件,在点击切换主题时,创建link标签动态加载到head标签中,或者是动态改变link标签的href属性。
缺点:

动态加载样式文件,如果文件过大网络情况不佳的情况下可能会有加载延迟,导致样式切换不流畅

维护不便!

**优点:**实现了按需加载,提高了首屏加载时的性能且兼容性好

  1. 先创建主题css文件dark.csslight.css
  2. html中link引入
html 复制代码
<head>
	<link rel="stylesheet" href="./css/dark.css" >
	<link rel="stylesheet" href="./css/light.css" disable>
</head>
  1. 切换主题
typescript 复制代码
const themesCssSheet = [
	document.querySelector("link[data-theme='light']"),
	document.querySelector("link[data-theme='dark']"),
]
btn.addEventlistner("click",()=>{
	const dataTheme = btn.getAttribute("data-theme")
	themesCssSheet.forEach(theme=>{
		t.disable = theme.dataset.theme === dataTheme 
	})
})

提前引入不同主题的css样式,通过切换类名切换主题

  1. 定义好css样式
css 复制代码
/* light样式主题 */
body{
  color: #f90;
  background: #fff;
  --text-color:black
  --header-bg:orange
}
/* dark样式主题 */
.dark {
  color: #eee;
  background: #333;
  --text-color:#fff
  --header-bg:blue
}
  1. 点击切换主题时 js切换
typescript 复制代码
document.body.classList.toggle("dark")

css预编译器

scss 复制代码
$themes:(
	light:(
		textColor:black
		headerBg:orange
	),
	dark:(
		textColor:#fff
		headerBg:blue
	)
)
$currentTheme:light;
@mixin changeTheme(){
	@each $key,$value in $themes{
		$currentTheme:$key !global;
		html[data-theme='#{$key}'] & {
			@content;
		}
	}
}

@function getTextColor(){
	$currentThemeObj: map-get($themes,$curTheme);
	@return map-get($currentThemeObj,"textColor");
}
@function getHeaderBg(){
	$currentThemeObj: map-get($themes,$curTheme);
	@return map-get($currentThemeObj,"headerBg");
}

@function getThemeValue($key){
	$currentThemeObj: map-get($themes,$curTheme);
	@return map-get($currentThemeObj,$key);
}
.content{
	width:100%;
	height:100%;
	@changeTheme{
		textColor:getTextColor();
		headerBg:getHeaderBg();
	}
}

基于CSS变量(最优方式)

原理 :根据html元素标签的属性切换所使用的主题变量。
缺点 :IE兼容性不行(基本可以忽略)。
优点:便于维护,基本无需css参与

  1. 定义两套(多主体多套)变量
css 复制代码
:root{
	--text-color:black
	--header-bg:orange
	...
}
// 暗色主题
html[data-theme='dark']{
	--text-color:#fff
	--header-bg:blue
	...
}
  1. 在main.js项目入口文件导入该变量,便于组件使用
  2. 组件中主题相关的直接使用变量
  3. 主题切换时
typescript 复制代码
type Theme = "dark" | "light"
export const changeTheme = (theme:Theme )=>{
	document.documentElement.dataset.theme = theme
}

CSS变量+动态setProperty

  1. 定义变量
css 复制代码
:root{
	--text-color:black
	--header-bg:orange
	...
}
  1. 定义js更改属性值方法
typescript 复制代码
export const setCssVar = (prop: string, val: any, dom = document.documentElement) => {
  dom.style.setProperty(prop, val)
}
// 当样式改变
setCssVar('--text-color', color)

总结

还有其他方式(例如vue3 v-bind css,变量和类名结合),但是总体都大同小异。

相关推荐
Chef_Chen4 分钟前
从0开始学习机器学习--Day33--机器学习阶段总结
人工智能·学习·机器学习
搏博5 分钟前
神经网络问题之:梯度不稳定
人工智能·深度学习·神经网络
GL_Rain22 分钟前
【OpenCV】Could NOT find TIFF (missing: TIFF_LIBRARY TIFF_INCLUDE_DIR)
人工智能·opencv·计算机视觉
shansjqun27 分钟前
教学内容全覆盖:航拍杂草检测与分类
人工智能·分类·数据挖掘
狸克先生29 分钟前
如何用AI写小说(二):Gradio 超简单的网页前端交互
前端·人工智能·chatgpt·交互
baiduopenmap44 分钟前
百度世界2024精选公开课:基于地图智能体的导航出行AI应用创新实践
前端·人工智能·百度地图
hopetomorrow1 小时前
学习路之PHP--使用GROUP BY 发生错误 SELECT list is not in GROUP BY clause .......... 解决
开发语言·学习·php
小任同学Alex1 小时前
浦语提示词工程实践(LangGPT版,服务器上部署internlm2-chat-1_8b,踩坑很多才完成的详细教程,)
人工智能·自然语言处理·大模型
网络安全-杰克1 小时前
网络安全概论
网络·web安全·php
新加坡内哥谈技术1 小时前
微软 Ignite 2024 大会
人工智能