当前位置:网站首页>Jetpack Compose使用自定义操作符实现绘制五角星效果
Jetpack Compose使用自定义操作符实现绘制五角星效果
2022-04-21 15:06:00 【安安安安卓】
公众号:安安安安卓
说明
compose中我们的所有ui操作,包括一些行为,例如:点击、手势等都需要使用Modifier来进行操作。因此对Modifier的理解可以帮助我们解决很多问题的
自定义星行Modifier
本文我们打算自定义一个Modifier,通过这个modifier我们可以实现用一个操作符就画出五角星的效果
原理
我们实现绘制五角星的原理如下图,首先我们会虚构两个圆,将内圆和外圆角度平分五份,然后依次连接内圆和外圆的切点的坐标,然后使用path绘制完成。

实现
代码中的实现涉及到自定义绘制,难度并不大。需要注意的点:
- composse中角度的锚点是弧度(Math.PI)、而原生的锚点是角度(360)
- 默认的原点在左上角,我们绘制的时候需要主动移动到组合的中心点
- path的绘制使用Fill可以填充闭合路径图形,使用Stroke可以绘制线性闭合路径图形
代码
fun Modifier.customDraw(
color: Color,
starCount: Int = 5,
checked: Boolean = false,
) =
this.then(CustomDrawModifier(color, starCount, checked = checked))
class CustomDrawModifier(
private val color: Color,
private val starCount: Int = 5,//星的数量
private var checked: Boolean = false,
) :
DrawModifier {
override fun ContentDrawScope.draw() {
log("$size")
val radiusOuter = if (size.width > size.height) size.height / 2 else size.width / 2 //五角星外圆径
val radiusInner = radiusOuter / 2 //五角星内圆半径
val startAngle = (-Math.PI / 2).toFloat() //开始绘制点的外径角度
val perAngle = (2 * Math.PI / starCount).toFloat() //两个五角星两个角直接的角度差
val outAngles = (0 until starCount).map {
val angle = it * perAngle + startAngle
Offset(radiusOuter * cos(angle), radiusOuter * sin(angle))
}//所有外圆角的顶点
val innerAngles = (0 until starCount).map {
val angle = it * perAngle + perAngle / 2 + startAngle
Offset(radiusInner * cos(angle), radiusInner * sin(angle))
}//所有内圆角的顶点
val path = Path()//绘制五角星的所有内圆外圆的点连接线
(0 until starCount).forEachIndexed {
index, _ ->
val outerX = outAngles[index].x
val outerY = outAngles[index].y
val innerX = innerAngles[index].x
val innerY = innerAngles[index].y
// drawCircle(Color.Red, radius = 3f, center = outAngles[index])
// drawCircle(Color.Yellow, radius = 3f, center = innerAngles[index])
if (index == 0) {
path.moveTo(outerX, outerY)
path.lineTo(innerX, innerY)
path.lineTo(outAngles[(index + 1) % starCount].x,
outAngles[(index + 1) % starCount].y)
} else {
path.lineTo(innerX, innerY)//移动到内圆角的端点
path.lineTo(outAngles[(index + 1) % starCount].x,
outAngles[(index + 1) % starCount].y)//连接到下一个外圆角的端点
}
if (index == starCount - 1) {
path.close()
}
}
translate(size.width / 2, size.height / 2) {
drawPath(path, color, style = if (checked) Fill else Stroke(width = 5f))
}
}
}
最终实现效果

版权声明
本文为[安安安安卓]所创,转载请带上原文链接,感谢
https://blog.csdn.net/ymeddmn/article/details/124211249
边栏推荐
猜你喜欢

MySQL下载和安装教程

解锁OpenHarmony技术日!年度盛会,即将揭幕!

DBeaver无法连接数据库,如何解决?

数字化时代,企业运维面临现状及挑战分析解读

阿里云云效研发协同服务相关协议条款 |云效

Dbaver cannot connect to the database. How to solve it?

SAP ui5 application development tutorial 70 - how to trigger page route jump using button control

想请问一下如何从数据库中找到软件注册码

SAP UI5 应用开发教程之七十 - 如何使用按钮控件触发页面路由跳转

vs2019中libmysql.lib乱码
随机推荐
pyspark中RDD和DataFrame之间的转换利用RDD处理DataFrame:数据分段等功能实现
虫子 Makefile
专题测试04·多元函数微分学【李艳芳全程班】
LNK2001 - unresolved external symbol in PCL test program
二叉树的堂兄弟节点-c语言
Mysql数据库(3)
mysql 多条件查询
Qt网络与通信(TCP聊天室)
虫子 12864
异步操作之后让await后续的代码能够继续执行
.Net C# Newtonsoft. JSON serializersettings configuration
Ali's monthly salary is 15K. The interview is so simple
如何选择合适的 Neo4j 版本(2022)
如何在代码层面提供CPU分支预测效率
使用wx.showActionSheet选择框修改数据库中的信息,为什么会报data未定义的错呢
leetcode答题笔记(一)
Golang zap log
MySQL数据库实际运用(一)
Insect 1602
Mysql8.0以上重置初始密码的方法