uniapp微信小程序 canvas绘制圆形半透明阴影 createCircularGradient函数不支持透明度部分解决方案

背景

我需要在微信小程序中,用canvas绘制一个圆形钟表,在ui设计图中,有一部分阴影,这里我节选一下:

即深色发黑的部分

canvas通用阴影绘制

由于canvas中并不支持css那样简单的方式为圆形添加阴影或高光,于是我有了如下想法:

在下面绘制另一个带有黑色渐变效果的圆,在上面绘制原来的圆,这样圆看起来就有了影子,也就是单独给他画个影子出来

参见这篇BootWiki内容:微信小程序API gradient(如何绘制渐变效果)

以下是他提供的demo,由于背景色为白色,看不出来问题

这里借用他的图片,表达一下我们的想法:

这是不看起来就有影子了!

有了思路,这里我们付诸实践,代码大致如下

javascript 复制代码
//阴影绘制 version1
const grdD = ctx.createCircularGradient(x坐标+5, y坐标+5,半径)  // 与原图形适当错开
		grdD.addColorStop(0, 'rgba(0,0,0,255)')  // 渐变起始值为纯黑不透明
		grdD.addColorStop(1, 'rgba(0)') // 渐变终点为纯黑纯透明
		ctx.setFillStyle(grdD) //固定用法
		ctx.fillRect(0, 0, canvasLength, canvasLength)//绘制范围,这里取了全图,后面会改

实际使用上效果不佳,根据实际绘制,诸君按需调整参数。温馨提示:addColorStop()函数可以添加多个。

我就因为阴影效果不理想,调整了addColorStop() 函数,如下

javascript 复制代码
//阴影绘制 version2
const grdD = ctx.createCircularGradient(x坐标+5, y坐标+5,半径)  
		grdD.addColorStop(0, 'rgba(0,0,0,255)')  
		grdD.addColorStop(0.8, 'rgba(0,0,0,255)') // 在此处添加一个渐变节点
		grdD.addColorStop(1, 'rgba(0)') 
		ctx.setFillStyle(grdD) 
		ctx.fillRect(0, 0, canvasLength, canvasLength)

调整后的效果图

可以看到,我们实际上修改了渐变起始的位置,通过这种方式使边缘的色彩更加浓郁。避免直接渐变导致的外圈颜色过于浅淡。

但不能在微信小程序这样画

在使用官方 devtool 开发的时候,电脑端模拟器是完全支持这种形式的,但真机运行没有任何效果。(2024年8月6日)

经过我控制变量的测试,我发现这是因为真机小程序是不支持 rgba()形式定义颜色的。甚至也不支持 #000000FF 这种形式

而如果你使用 #000000FF 形式定义透明度,devtool 会直接报错,可是 rgba() 形式定义的颜色则是一个能运行,一个没效果。

尝试在微信端巧妙绘制

我们迎难而上,找了半天,没找到方案,于是决定不使用透明度。如果我直接使用当前底色作为渐变的终点色呢?

如图:白色底色实现参照组

我们不再使用 fillRect() 函数全图绘制,而是指定一个范围,并使用底色作为渐变终点,黑色作为渐变起点绘制阴影。

实现效果大致如此,此图为夸张表现

可以见得如果绘制的内容足够简单(没有背景色,或者背景色为纯色),那么只要梳理好绘制顺序,看到这里就够了。

但显然这种大开大合,外面套一个大正方形的绘制形式,实际应用场景有限。

最终方案

方案1(简单场景可用,即起始色为黑色画阴影的情况,或其他较深颜色):

这个方案其实可以放在前面说,因为真的简单,只需要将之前代码,渐变终点的颜色设置为 transparent 就可以了

javascript 复制代码
grd.addColorStop(0, 'green')
grd.addColorStop(0.8, 'green')
grd.addColorStop(1, 'transparent') // 透明作为渐变终点

方案2(不完美,聊胜于无):

但说实话彩色的并不常见,常见的还是绘制黑色和白色,作为阴影或者高光使用。

但方案1不支持白色,为什么?请看图👇

javascript 复制代码
//上图代码如下
grd.addColorStop(0, 'white')
grd.addColorStop(0.5, 'white')
grd.addColorStop(1, 'transparent')

怎么说呢,我就觉得这效果挺抽象的。。。

你说,能不能是这玩意对white的支持有问题,那么我写 #FFFFFF 行不行,或者近似颜色 #FFFFF0 呢?

能想到这一层我只能说真是个小机灵鬼,但没用,我试过了

javascript 复制代码
grd.addColorStop(0, '#fefefe')
grd.addColorStop(0.8, '#dedede')//这两种颜色我分别试过了,这里放在一张图展示
grd.addColorStop(1, 'transparent')

后经测试,当白色程度超过约 #888888 的情况下,就会出现黑边问题 ,如图使用 #FFFF88较为明显

经查询,我发觉问题是: transparent 指代的是 rgba(0,0,0,0) 的值,也就是完全透明的黑色。但我们需要的是完全透明的白色。所以浅色下才会有一个同时向黑色和透明色过度的过程。暂时来看似乎没办法解决。

能切图还是切图!!!不要在这个功能上再浪费时间了!!!

这里给出一个凑合用的替代方案,即绘制半透明白色圆。半透明代码如下:

javascript 复制代码
// 直接画个半透明白色圆替代
ctx.beginPath();
ctx.globalAlpha = 0.5 //调整全局画笔透明度
ctx.setStrokeStyle("white")
ctx.arc(outsideX, outsideY, outsideR, 0 * deg, 360 * deg)
ctx.setFillStyle("white")
ctx.fill()
ctx.stroke()
ctx.globalAlpha = 1//别忘了改回去

你也可以叠加多个半透明圆,但我还是建议你切图。

相关推荐
井眼12 分钟前
微信小程序-prettier 格式化
微信小程序·小程序
我开心就好o1 小时前
uniapp点左上角返回键, 重复来回跳转的问题 解决方案
前端·javascript·uni-app
Random_index1 小时前
#Uniapp篇:支持纯血鸿蒙&发布&适配&UIUI
uni-app·harmonyos
qq_17448285753 小时前
springboot基于微信小程序的旧衣回收系统的设计与实现
spring boot·后端·微信小程序
wqq_9922502773 小时前
springboot基于微信小程序的食堂预约点餐系统
数据库·微信小程序·小程序
初遇你时动了情9 小时前
uniapp 城市选择插件
开发语言·javascript·uni-app
licy__9 小时前
微信小程序登录注册页面设计(小程序项目)
微信小程序·小程序
小小黑00711 小时前
uniapp+vue3+ts H5端使用Quill富文本插件以及解决上传图片反显的问题
uni-app·vue
草字12 小时前
uniapp input限制输入负数,以及保留小数点两位.
java·前端·uni-app
前端小胡兔12 小时前
uniapp rpx兼容平板
uni-app