第 65 篇原创好文~

本文首发于政采云前端团队博客:深色模式适配指南

深色模式适配指南

背景

随着 iOS 13 的发布,深色模式(Dark Mode)越来越多地出现在大众的视野中,支持深色模式已经成为现代移动应用和网站的一个潮流,前段时间更是因为微信的适配再度引起热议。深色模式不仅可以大幅减少电量的消耗,减弱强光对比 ,还能 提供更好的可视性和沉浸感。

那针对 一款 App 应用(原生 + H5)怎么进行深色模式的适配呢?今天就让我们一起来探究吧!

系统兼容

想要实现深色模式的效果,前提条件是要系统支持, 目前 常见系统支持情况如下:

H5 深色适配

随着深色模式的流行,越来越多的操作系统、浏览器开始支持深色模式,现在可以利用 CSS 的媒体查询方法(prefers-color-scheme)以及 CSS 变量(CSS variables、CSS custom properties)就可以实现页面主题跟随系统自动切换深浅模式 。CSS 变量除了 IE ,其余各大浏览器都支持的比较好, 但 prefers-color-scheme 方法还处于 W3C 草案规范,需要对不兼容浏览器做向下兼容,具体浏览器兼容性可以查询 Can I Use, 综合来说,高版本的主流浏览器都已经支持,IE 不支持。

可以通过以下两种方式来实现 Web 端的深色适配:

一、CSS 的媒体查询

prefers-color-scheme 是一种 用于检测用户是否有将系统的主题色设置为亮色或者暗色 的 CSS 媒体特性 。 利用其设置不同主题模式下的 CSS 样式, 浏览器会自动根据当前系统主题加载对应的 CSS 样式。light 适配浅色主题,dark 适配深色主题,no-preference 表示获取不到主题时的适配方案。CSS@media (prefers-color-scheme: light) {

.article {

background:#fff;

color: #000;

}

}

@media (prefers-color-scheme: dark) {

.article {

background:#000;

color: white;

}

}

@media (prefers-color-scheme: no-preference) {

.article {

background:#fff;

color: #000;

}

}Link 标签

来看一下效果,将系统设置为浅色外观:

然后将系统设置为深色外观:

页面已经加载了对应深色主题的样式:

二、CSS 变量 + 媒体查询

window.matchMedia 方法可以用来查询 指定的媒体查询字符串解析后的结果。 结合 CSS 变量和 matchMedia 的查询结果,设置对应的 CSS 主题颜色。该方法更灵活,可以单独抽离主题色进行适配。

CSS 变量的作用域与 CSS 的"层叠"规则一致,优先级最高的声明生效。所以当 body 上存在 "dark" 类名时,:root .dark 会生效,否则 :root 生效。.article {

color: var(--text-color, #eee);

background: var(--text-background, #fff);

}

:root {

--text-color: #000;

--text-background: #fff;

}

:root .dark {

--text-color: #fff;

--text-background: #000;

}

使用 matchMedia 匹配主题媒体,深色模式匹配 (prefers-color-scheme: dark) ,浅色模式匹配 (prefers-color-scheme: light) 。

监听主题模式,深色模式时为 body 添加类名 dark,根据 CSS 变量的响应式布局特点,自动生效 dark 类名下的 CSS。const darkMode = window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)');

// 判断是否匹配深色模式

if (darkMode && darkMode.matches) {

document.body.classList.add('dark');

}

// 监听主题切换事件

darkMode && darkMode.addEventListener('change', e => {

if (e.matches) {

document.body.classList.add('dark');

} else {

document.body.classList.remove('dark');

}

});

那么,针对不支持 CSS 变量的 IE 浏览器怎么办呢?不做兼容性处理的话那页面可能就是一团糟了。所以我们需要针对不兼容的浏览器做一些兜底处理,这里我们可以在 webpack 等构建工具中借助 post-css 的 postcss-css-variables 插件来自动解析 CSS 变量对应的色值,并在原始 CSS 定义之上添加一条新的 CSS 样式 , 做到对不支持 CSS 变量 浏览器 的兼容 。

用法如下:// 根目录 postcss.config.js

module.exports = {

plugins: {

"postcss-css-variables": {

preserve: true, // 保留 var() 定义

preserveInjectedVariables: false, // 去除其他模块的重复变量

variables: require("./page.json"), // CSS 变量,可以支持多个

}

}

};

项目实践

现在的 Web、App 项目大都引用第三方开源组件库,组件库一般使用会 Sass、Less 等 CSS 预处理器 定义颜色变量作为组件的基础色值,并单独抽离为配置文件。所以,项目使用组件库时可以根据修改基础色值来自定义主题。那么针对项目的 深色模式适配方案 也一样,主要分为三步:一、组件库深浅色主题 适配 ; 二是、项目中深浅色的颜色适配 ; 三、 完成 CSS 变量到页面的 注入。

组件库样式、自定义样式适配

如果第三方组件本身支持多主题或者深色模式,可以直接按说明给组件设置对应主题模式;如果第三方组件库不支持的话,只能用覆盖的方式。这里以 Less 为例进行简单实例说明:

修改前:// index.less

@white: #fff; // 颜色预定义

@background-color: @white; // 组件样式 panel.less

.panel-background-color {

background-color: @background-color; // 组件中使用 less 变量定义颜色样式

}

新增两个 js 或者 JSON 文件,分别定义深浅模式下的 CSS 变量, 并命名为 light-theme1.js、dark-theme2.js 他们并不会影响组件的样式,只是便于后期注入到全局 style 中。

修改后:// 浅色主题文件 light-theme1.js

const bgColor = '#fff';// 颜色预定义

module.exports = {

"--background-color": bgColor;

}

// 深色主题文件 dark-theme1.js

const bgColor = '#000';// 颜色预定义

module.exports = {

"--background-color": bgColor;

}// 组件样式 panel.less

.panel-background-color {

background-color: var(--background-color); //组件中颜色样式

}

CSS 变量支持第二参数,当变量不存在或者未注册成功时,可以为其设置默认值,优化如下:// 组件样式 panel.less

.panel-background-color {

background-color: var(--background-color, @background-color); // 组件中颜色样式,其中 @background-color 代表修改前组件的背景颜色变量,这里设其为默认值,在适配不成功情况下,可以保持适配前的样式。

}

项目才是真正使用组件的地方,并且项目本身也有很多 自定义 CSS 的 颜色样式,需要做与组件库类似的处理,结果也会得到两个 js / json 文件,分别命名为 light-theme2.js 、dark-theme2.js。

CSS 注入

在页面渲染前,需要把定义深浅 样式的 CSS 变量注入到页面。

以上两步得到了四个文件,合并浅色样式文件 light-theme1.js 和 light-theme2.js 得到 light-theme.js,合并深色样式文件dark-theme1.js 和 dark-theme2.js 得到 dark-theme.js, 最后 把 light-theme.js、 dark-theme.js 两个文件注入到页面中,注入脚本如下:import lightTheme from './light-theme';

import darkTheme from './dark-theme';

// 创建一个 style 元素,用于插入 css 定义

const createStyle = (content) => {

const style = document.createElement('style');

style.type = 'text/css';

style.innerHTML = content;

document.getElementsByTagName("script")[0].parentNode.appendChild(style);

// 在 body 标签中定义 css 变量

const createCssStyle = () => {

const lightThemeStr = Object.keys(lightTheme).map(key => key + ':' + lightTheme[key]).join(';');

const darkThemeStr = Object.keys(darkTheme).map(key => key + ':' + darkTheme[key]).join(';');

const lightContent = `body{${lightThemeStr}}`; // 浅色模式 CSS 变量定义

const darkContent = `body.dark{${darkThemeStr}}`; // 深色模式 CSS 变量定义

createStyle(lightContent);

createStyle(darkContent);

isDarkSchemePreference();

};

注入完成后,项目页面中就有了 css 变量定义,包括浅色模式 CSS 变量定义和深色模式 CSS 变量定义,具体哪一个生效, 就可以根据上面提到的两种适配方案 给 body 添加 class 来控制 。 默认时浅色模式生效,添加 dark 类名时,深色模式会生效。至此就实现了一套完整的深色模式适配方案。

native 深色适配

iOS

在 iOS 系统中, 开发者从颜色和图片两个方面来进行适配,我们不需要关心切换模式后该怎么操作,因为这些都由系统帮我们实现。颜色的适配,需要使用系统提供的API,在回调用中不同的模式下分别设置颜色,而图片的适配,需要在 XCode 的 工具栏中 Appearances 下选择 Any,Dark,在同一名称资源的配置下分别添加图片资源。当切换深色模式时,系统会根据适配的颜色和图片资源进行查找和自动切换对应模式下的颜色和资源文件。

Android

安卓在 Android 10(API 级别 29)及更高版本中提供深色主题背景,可以通过以下三种方法启用深色主题背景:使用系统设置(Settings -> Display -> Theme)启用深色主题背景

使用"快捷设置"图块,从通知托盘中切换主题背景(启用后)

在 Pixel 设备上,选择"省电模式"将同时启用深色主题背景,其他原始设备制造商 (OEM) 不一定支持这种行为

在应用中支持深色主题背景

如要支持深色主题背景,必须将应用的主题背景(通常可在 res/values/styles.xml 中找到)设置为继承 DayNight 主题背景:

这会将应用的主要主题背景与系统控制的夜间模式标记相关联,并将应用的默认主题背景设置为深色主题背景(如果已启用)。

主题背景和样式

主题背景和样式应避免使用旨在于浅色主题背景下使用的硬编码颜色或图标,您应改用主题背景属性(首选)或适合在夜间使用的资源,以下是需要了解的两个最重要的主题背景属性:?android:attr/textColorPrimary 这是一种通用型文本颜色,它在浅色主题背景下接近于黑色,在深色主题背景下接近于白色,该颜色包含一个停用状态。

?attr/colorControlNormal 一种通用图标颜色,该颜色包含一个停用状态。

Flutter

这里以Flutter 为例,简单介绍下跨平台开发框架如何适配深色模式。Flutter定义主题有两种方式:全局主题或使用Theme来定义应用程序局部的颜色和字体样式。

全局主题

全局主题就是有应用程序根 MaterialAPP 创建 的 Theme。为了在整个应用程序中共享包含颜色和字体样式的主题,我们可以提供 ThemeData 给 Material 的构造函数。theme 指定的是浅色模式,darkTheme 指定的是深色模式,程序会根据系统设定的暗黑模式自动匹配模式。new MaterialApp(

title: title,

theme: new ThemeData(

brightness: Brightness.light,

primaryColor: Colors.lightBlue[800],

accentColor: Colors.cyan[600] ,

),

darkTheme: new ThemeData(

brightness: Brightness.dark,

primaryColor: Colors.lightGreen[800] ,

accentColor: Colors.cyan[200],

),

);

局部主题

如果我们想在应用程序的一部分中覆盖应用程序的全局的主题,我们可以将要覆盖得部分封装在一个 Theme 的 Widget 中,有2种方法可解决:创建特有的 ThemeData 或扩展父主题。

创建特有的ThemeData

如果我们不想继承任何应用程序的颜色或字体样式,我们可以通过 new ThemeData() 创建一个实例并将其传递给 Theme Widget。// Create a unique theme with "new ThemeData"

new Theme(

data: new ThemeData(

accentColor: Colors.yellow,

),

child: new FloatingActionButton(

onPressed: () {},

child: new Icon(Icons.add),

),

);

扩展父主题

扩展父主题时无需覆盖所有的主题属性,我们可以通过使用 copyWith 方法来实现。// Find and Extend the parent theme using "copyWith". Please see the next section for more info on `Theme.of`.

new Theme(

data: Theme.of(context).copyWith(accentColor: Colors.yellow),

child: new FloatingActionButton(

onPressed: null,

child: new Icon(Icons.add),

),

);

使用主题

我们可以在Widget的 build 方法中通过 Theme.of(context) 函数使用自定义的主题。new Container(

color: Theme.of(context).accentColor,

child: new Text(

'Text with a background color',

style: Theme.of(context).textTheme.title,

),

);

渲染效果 如下 :

总结

以上分别介绍了在 App 应用中对 H5 页面和客户端的深色模式适配方案,当然其中 H5 的方案页同样适应于 PC 端。使用前一定要确保你的系统和浏览器是兼容深色模式的,不然就没有效果了呢。本篇只简单介绍了几种方案,欢迎有更好想法的小伙伴一起讨论~

参考资料

招贤纳士

政采云前端团队(ZooTeam),一个年轻富有激情和创造力的前端团队,隶属于政采云产品研发部,Base 在风景如画的杭州。团队现有 50 余个前端小伙伴,平均年龄 27 岁,近 3 成是全栈工程师,妥妥的青年风暴团。成员构成既有来自于阿里、网易的“老”兵,也有浙大、中科大、杭电等校的应届新人。团队在日常的业务对接之外,还在物料体系、工程平台、搭建平台、性能体验、云端应用、数据分析及可视化等方向进行技术探索和实战,推动并落地了一系列的内部技术产品,持续探索前端技术体系的新边界。

如果你想改变一直被事折腾,希望开始能折腾事;如果你想改变一直被告诫需要多些想法,却无从破局;如果你想改变你有能力去做成那个结果,却不需要你;如果你想改变你想做成的事需要一个团队去支撑,但没你带人的位置;如果你想改变既定的节奏,将会是“5 年工作时间 3 年工作经验”;如果你想改变本来悟性不错,但总是有那一层窗户纸的模糊… 如果你相信相信的力量,相信平凡人能成就非凡事,相信能遇到更好的自己。如果你希望参与到随着业务腾飞的过程,亲手推动一个有着深入的业务理解、完善的技术体系、技术创造价值、影响力外溢的前端团队的成长历程,我觉得我们该聊聊。任何时间,等着你写点什么,发给 ZooTeam@cai-inc.com

Android百度浏览器深色模式,深色模式适配指南相关推荐

  1. 自由浏览器 android,百度浏览器发布安卓6.1版 趣味视频弹幕吐槽不停

    9月15日,深受安卓用户喜爱的百度浏览器发布了全新Android6.1版本,继续主打趣星球专区功能,号召广大小伙伴们上趣星球哔哔,做个有趣的人.本次推出Android6.1版,不仅在功能方面富有创新, ...

  2. 百度浏览器android,百度浏览器app下载

    手机浏览器,并且还拥有强劲的极速内核,使用户的个性所求能够得到全面贴心的满足,结果更清晰,收录海量精品小说,优化你的阅读体验,视频栏目华丽变身,有钱赚,我的皮肤我做主,释放你的手和眼,省时省心,焦点内 ...

  3. android 百度浏览器内核,百度浏览器安卓6.2版上线:新一代内核更快更稳定

    10月20日,以趣星球闻名深受年轻群体喜爱的百度浏览器发布最新安卓6.2版本,跟随该版本,T7内核也正式亮相,T7内核是基于Blink内核的重构和产品化,针对中国人的使用习惯进行了大幅优化,大幅提升浏 ...

  4. Android targetSdkVersion从23升级到26适配指南

    根据华为开发者平台AppGallery目标 API等级(targetSdkVersion)重要变更要求的通知,自2018年7月18日,华为应用市场联合国内主流应用预置与分发服务提供者,作为发起单位,共 ...

  5. Android App Dark Theme(暗黑模式)适配指南,android实战mysql

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-v2qd7l5a-1636430548017)(https://user-gold-cdn.xitu.io/2020/3/ ...

  6. Android App Dark Theme(暗黑模式)适配指南

    在 2019 年的 Google I/O 和 Apple WWDC 上,新露面的 Android 10 和 iOS 13 都宣布将支持 Dark Theme 也就是我们常说的暗黑模式,并提供相关 AP ...

  7. android百度地图标方向,Android百度地图之方向感应和模式更改

    本文实例为大家分享了Android百度地图之方向感应和模式更改,供大家参考,具体内容如下 目标效果: 菜单中设置几种模式,点击可查看不同的地图形式,这里随便截几张图. 1.首先要去百度地图网站 htt ...

  8. [原创] Android Automotive 车载应用对驾驶模式(Safe Drive Mode)适配的几种方法

    目录 前言 开发环境 1. Android Automotive 和 Android Auto的区别 Android Auto: 2. Android Automotive 的驾驶模式介绍 3. An ...

  9. android 无障碍的菜单,讯飞输入法Android V9.1.9652 菜单及表情适配无障碍模式

    原标题:讯飞输入法Android V9.1.9652 菜单及表情适配无障碍模式 据悉,视障.听障.读写障碍人群约占到全国人口的五分之一,输入法作为手机必备可以为这些人群的不便做些什么吗?一直致力于信息 ...

  10. 【Web技术】935- 深色模式适配指南

    ????  这是第 65 篇不掺水的原创,想要了解更多,请戳上方蓝色字体:政采云前端团队 关注我们吧- 本文首发于政采云前端团队博客:深色模式适配指南 https://www.zoo.team/art ...

最新文章

  1. java queue size_Java中的PriorityQueue size() 方法 - Break易站
  2. 【转】POJ 1177 Picture(1)
  3. 快看!你连接世界的新入口,在那朵云上
  4. node html5,html5前端入门教程分享:Node.Js 框架
  5. 爬虫初窥day1:urllib
  6. hbaseRowkey设计
  7. thinkphp5做好的PHP项目上传服务器上,访问任何页面都只显示默认的欢迎页面
  8. Tensorflow教程2:使用卷积神经网络的图像分类器
  9. 工业软件下载大全202108
  10. 开启虚拟机电脑自动重启的解决
  11. 【小月电子】ALTERA FPGA开发板系统学习教程-LESSON2 LED灯闪烁
  12. 常用计算机检索算符,在计算机检索中,常用的布尔逻辑算符有哪几种
  13. js图片加载 完成之前显示loading中的图片
  14. 如何在scrapy中捕获并处理各种异常
  15. word怎样把多个标题设置成一级标题,二级标题的简便方法
  16. MES和ERP在生产过程优化上,谁更胜一筹?
  17. No qualifying bean of type 'com.xxx.xx.service.xxService' available: expected at leas
  18. Maven与Gradle项目构建工具
  19. 京东平台研发朱志国:领域驱动设计(DDD)理论启示
  20. pdf文件有密码怎么解除

热门文章

  1. Java微信小微商户进件,已经解决接口暂无权限,稍后再试的问题
  2. 我国街景地图向何处去
  3. 确定有限自动机DFA
  4. 计算机音乐对应的数字,音乐和数字之间的关系
  5. Iptables-外网地址及端口映射到内网地址及端口
  6. js调用Python函数
  7. java怎么做小人跑动的动画_纯CSS3实现人物跑步动画
  8. 图片(img)alt属性标签怎么写
  9. 渥太华大学计算机科学博士,加拿大渥太华大学计算机科学CO-OP录取案例
  10. 定性分析与定量分析的区别