注册
web

我是如何实现网页颜色自适应的

前言


不知大家有没有留意过,当前大部分 App 或网页中,很少存在允许用户完全自定义要展示信息的颜色的功能。


例如在钉钉的自定义表情中,只允许用户从一组预设的配色中随机切换:


image.png转存失败,建议直接上传图片文件


再比如笔记应用 Notion 虽然允许用户改变文本颜色,但也只允许在一组预设色值中选取:


image.png转存失败,建议直接上传图片文件


原因无它,配色,不是一件容易事。


对于大众用户而言,没什么颜色理论知识,很可能挑出来的颜色在应用中很难看、看不清,这会极大的影响用户的使用体验(即使是用户自己造成的)。


因此大部分产品选择的做法是提供一组预先检验过的、不会对用户阅读造成困扰的颜色,放在应用中供用户挑选。


今天我来斗胆挑战一下这个业界难题。


在这篇文章中将会探讨两个具体问题:



  1. 如何让文本颜色自适应背景色
  2. 如何允许用户完全自定义主题色,同时保证可阅读性

文本颜色自适应背景色


在下面这张图中,文本的颜色默认都是黑色的,背景色设置了多个明暗不同的颜色。可以看到对于暗色的背景色,此时文本可阅读性特别差(不太明显,想看清楚会很累)。


image.png转存失败,建议直接上传图片文件


如果能够自动根据背景色的明暗,决定使用白色还是黑色的文本,那便是实现了文本颜色的自适应了。


首先介绍下借助第三方库实现的方案。


第三方库实现


color 是 JavaScript 生态中在颜色处理方面最流行的库,它有诸多功能:颜色空间转换、颜色通道分解、获取对比度、颜色混合……


在文本颜色自适应这个场景中,最为方便的两个 API 是 isDark()isLight() ,它们分别用来表示一个颜色是否为深色、是否为浅色。


实际应用:


import Color from 'color'

const BgColors = ['#f87171', '#fef08a', '#042f2e', ...]
export default function Page() {
return (
<main>
{BgColors.map((bg) => (
<div
style={{
background: bg,
// 根据背景是否为深色决定文本用白色还是黑色
color: Color(bg).isDark() ? 'white' : 'black'
}}
>

恍恍惚惚
</div>
))}
</main>

)
}

实际效果:


image.png转存失败,建议直接上传图片文件


很 Nice ~


下面再来看下使用 CSS 的解决方案。


mix-blend-mode: difference


mix-blend-mode: difference 用于指定一个元素的颜色与背景色进行「差值」混合,可以使用如下公式表达:


# || 表示取绝对值
# 最终元素显示的颜色 = |元素原有的颜色 - 背景色|
result_color = | element_color - background_color |

例如:



  • 文本颜色为白色 rgb(255 255 255) 背景色为蓝色 rgb(0 0 255) ,最终文本颜色为黄色 rgb(255 255 0)
  • 文本颜色为黑色 rgb(0 0 0) ,此时无论背景色是什么颜色,最终文本的颜色一定和背景色完全相同,因为 | 0 - x | = x

下面来看个实际的 demo,这里我们让文本颜色为 rbg(255, 255, 255),背景色动态调整:


屏幕录制2024-08-31 14.22.02-3.gif


可以看到这个方案会有如下问题:



  • 当背景色为灰色时,文本的颜色和背景色很相似,对比度低,可阅读性差
  • 当背景色为彩色时,混合出来的颜色也是彩色,而且颜色比较脏,不太美观

CSS 相对颜色


以 CSS 颜色函数 rgb() 举例,相对颜色的语法是通过 from 关键词扩展了该函数的能力:


color: rgb(from red r g b);

由于 red 的 RGB 是 (255, 0, 0) ,因此后面的 r g b 值分别为 255 0 0 。对于 r g b 还可以调用 calc() 或其他 CSS 函数进一步处理。


除了 rgb()rgba() hsl() hwb() lch() 等等 CSS 颜色函数都是支持相对颜色语法的。


CSS 相对颜色语法带来的能力有调节亮度、调节饱和度、获取反转色、获取补充色……其中反转色就可以用于文本颜色自适应的场景中。


在上面 Demo 上,使用如下规则使文本的颜色为背景色的反转色:


color: rgb(from var(--bg) calc(255 - r) calc(255 - g) calc(255 - b));

实际效果:


image.png转存失败,建议直接上传图片文件


可以看到虽然能一定程度上提升可阅读性,但是有些反转色奇奇怪怪的,和背景色搭配起来实在不美观,并不是特别推荐使用。


color-constract


color-constract() 可以说是最适合文本颜色自适应场景的 CSS 函数了,用法简单,效果好!


color: color-constract(var(--bg) vs color1, color2, ...);

它的作用即从 vs 右边的一堆色值中,挑选一个和 vs 左边的色值对比度最大的返回。


color-constract(#ccc vs #000, #fff) ,由于 #ccc 是个浅灰色,色值和 #000 对比度更大,因此这个函数会返回 #000


很美好。


但是!这个函数现在几乎不能用!


image.png转存失败,建议直接上传图片文件


目前只有 Safari 中能使用,而且必须开启相关的实验性功能 Flag 。


完全允许用户自定义主题色


让文本的颜色根据背景色自适应,本质是在背景色(background)未知、前景色(foreground)有限的情况下,选择一个合适的前景色,使页面的可读性得到保障。


在上面的 Demo 中,背景色有淡紫色、淡黄色、深绿色、深棕色……前景色即文本的颜色只会有白色、黑色两种情况,依据背景色的明暗决定使用白色还是黑色。


相反的,我们讨论下前景色未知、背景色有限的情况。


看下这个实际应用场景:


image.png转存失败,建议直接上传图片文件


在这款应用中,支持明、暗两种主题,明亮主题为左边的白色(#FFFFFF),暗夜主题为右边的深蓝色(#020617),这是两个背景色;标签主题色(即前景色)支持用户自己设定,图中以红色(#FF0000)为例。


通过截图可以看到,红色在这两种背景色上展示效果都还不错,主要是因为他们的对比度足够。



  • 红色 vs 白色,对比度:3.998
  • 红色 vs 深蓝色,对比度:5.045

要使页面的可读性得到保障,对比度至少要 > 3。


如果允许用户完全自定义前景色,就不可避免的出现用户选择的颜色和背景色的对比度 < 3,这时页面阅读起来会很费力,影响用户体验。


下面给出两种解决方案。


及时给出提示


允许用户完全自定义,意味着用户可以从色盘上选取任意颜色。当用户选取的颜色和背景色对比度 < 3 时,界面上可以给出适当提示,让用户自己决定用一个「难看的颜色」还是遵从应用的建议,选择一个对比度合理,在当前应用中可以和背景色合理搭配的颜色。


实际效果:



相关代码并没有太多难点,主要还是借助 color 库,通过 color1.contrast(color2) 获取两个颜色之间的对比度实现,这里就不放代码了。


自动计算对比度安全的颜色


另一种解决方案,就有点「强制」的意思了:在用户从色盘选色时,实时计算色值和背景色的对比度,如果 < 3 了,就使用 color.lightness() API 逐步的调整颜色的明暗,确保最后界面上使用的是安全对比度的色值。


核心代码:


function calcLightColor(originColor: string) {
const white = Color('#fff')
let c = Color(originColor)
// 对比度 < 3,循环迭代使颜色越来越暗
while (c.contrast(white) < 3) {
// lightness() 可以读取/赋值颜色的 HSL 中的亮度值
// 如果是计算暗夜模式下的安全前景色,这里应该是 + 1,即让颜色越来越亮
c = c.lightness(c.lightness() - 1)
}
return c.hex()
}

function calcDarkColor(originColor: string() {
// ...
}

实际效果:


录屏2024-08-28 15.54.31.mov转存失败,建议直接上传图片文件


可以看到当用户选择了偏白的颜色时,明亮主题中实际使用的是灰色作为前景色,当用户选了择偏黑的颜色时,也有同样的自适应处理。


总结


在 2024 年的今天,CSS 看似已经足够强大,但是在颜色自适应类似的需求中还是略显不足,还好有 color 这个方便的 JavaScript 库帮助我们实现类似的需求。


无论背景色未知,还是前景色未知,只要设计界面时通过各种手段能保证前景色和背景色的对比度 > 3,那就可以保证界面的可阅读性。当然了,这里的安全对比度阈值是可以调整的,设为 3.5、3.75 都是可以的,但也非常不建议低于 3。


作者:人间观察员
来源:juejin.cn/post/7407983735661936674

0 个评论

要回复文章请先登录注册