[compose] 仿刮刮乐效果

需求

下班路上新开了一家彩票店,路过时总是心痒,本着小D怡情的心态,偶尔去刮几张,可是随着时间久了,发现也花了不少钱,看网上有人开发电子木鱼,突然奇想,为什么不做一张电子彩票。

分析

传统View,网上有很多解决方案,大多数是通过混合模式进行两个图层的合并。

大致思路:

1、使用onDraw()方法中的Canvas绘制底层中奖层

2、在上面绘制一个蒙版层Bitmap, 在蒙版层Bitmap里面,放置一个新的Canvas

3、绘制一个灰色的矩阵,绘制一个path,将paint的Xfermode设置为 PorterDuff.Mode.DST_IN 4、手指移动时,更新Path路径

Compose实现

1、通过实现DrawModifier,重写draw() 方法

2、绘制原始内容层,drawContent() 3、绘制蒙版和手势层,

kotlin 复制代码
//配置画笔 blendMode = Xfermode
private val pathPaint = Paint().apply {
    alpha = 0f
    style = PaintingStyle.Stroke
    strokeWidth = 70f
    blendMode = BlendMode.SrcIn
    strokeJoin = StrokeJoin.Round
    strokeCap = StrokeCap.Round
} 

drawIntoCanvas {
    //设置画布大小尺寸
    val rect = Rect(0f, 0f, size.width, size.height)
    //从原始画布层,转换一个新的画布层
    it.saveLayer(rect, layerPaint)
    //设置新画布大小尺寸
    it.drawRect(rect, layerPaint)
    startPath.lineTo(moveOffset.x, moveOffset.y)
    //绘制手指移动path
    it.drawPath(startPath, pathPaint)
    it.restore()
}

完整代码

kotlin 复制代码
fun ScrapeLayePage(){
    var linePath by remember { mutableStateOf(Offset.Zero) }
    val path by remember { mutableStateOf(Path()) }
    Column(modifier = Modifier
        .fillMaxWidth()
        .pointerInput("dragging") {
            awaitEachGesture {
                while (true) {
                    val event = awaitPointerEvent()
                    when (event.type) {
                        //按住时,更新起始点
                        Press -> {
                            path.moveTo(
                                event.changes.first().position.x,
                                event.changes.first().position.y
                            )
                        }
                        //移动时,更新起始点 移动时,记录路径path
                        Move -> {
                            linePath = event.changes.first().position
                        }
                    }
                }
            }
        }
        .scrapeLayer(path, linePath)
    ) {
        Image(
            modifier = Modifier.fillMaxWidth(),
            painter = painterResource(id = R.mipmap.cat),
            contentDescription = ""
        )
        Text(text = "这是一只可爱的猫咪~~")
    }
}

fun Modifier.scrapeLayer(startPath: Path, moveOffset: Offset) =
    this.then(ScrapeLayer(startPath, moveOffset))

class ScrapeLayer(private val startPath: Path, private val moveOffset: Offset) : DrawModifier {

    private val pathPaint = Paint().apply {
        alpha = 0f
        style = PaintingStyle.Stroke
        strokeWidth = 70f
        blendMode = BlendMode.SrcIn
        strokeJoin = StrokeJoin.Round
        strokeCap = StrokeCap.Round
    }

    private val layerPaint = Paint().apply {
        color = Color.Gray
    }

    override fun ContentDrawScope.draw() {
        drawContent()
        drawIntoCanvas {
            val rect = Rect(0f, 0f, size.width, size.height)
            //从当前画布,裁切一个新的图层
            it.saveLayer(rect, layerPaint)
            it.drawRect(rect, layerPaint)
            startPath.lineTo(moveOffset.x, moveOffset.y)
            it.drawPath(startPath, pathPaint)
            it.restore()
        }
    }
}

参考资料

相关推荐
夜郎king2 小时前
HTML5 SVG 实现日出日落动画与实时天气可视化
前端·html5·svg 日出日落
夏幻灵3 小时前
HTML5里最常用的十大标签
前端·html·html5
Mr Xu_3 小时前
Vue 3 中 watch 的使用详解:监听响应式数据变化的利器
前端·javascript·vue.js
未来龙皇小蓝3 小时前
RBAC前端架构-01:项目初始化
前端·架构
程序员agions3 小时前
2026年,微前端终于“死“了
前端·状态模式
万岳科技系统开发3 小时前
食堂采购系统源码库存扣减算法与并发控制实现详解
java·前端·数据库·算法
程序员猫哥_3 小时前
HTML 生成网页工具推荐:从手写代码到 AI 自动生成网页的进化路径
前端·人工智能·html
龙飞053 小时前
Systemd -systemctl - journalctl 速查表:服务管理 + 日志排障
linux·运维·前端·chrome·systemctl·journalctl
我爱加班、、3 小时前
Websocket能携带token过去后端吗
前端·后端·websocket
AAA阿giao3 小时前
从零拆解一个 React + TypeScript 的 TodoList:模块化、数据流与工程实践
前端·react.js·ui·typescript·前端框架