Android自定义有限制区域图例角度自识别涂鸦工具类
作者:似曾相识2022 发布时间:2021-12-21 14:35:03
自定义View分为继承自View和ViewGroup,继承ViewGroup相比继承View
在事件分发上ViewGroup多dispatchTouchEvent(事件分发)和onInterceptTouchEvent(事件拦截)两个方法
在继承上ViewGroup需要着重处理onmeasure(测量)与onLayout(控制位置等),而继承View重点在于onDrow(绘制)上
这篇我们就简单实现一个自定义签名类,通过手指触摸屏幕移动,显示手指滑过的路径。首先创建SignatureView并继承自View:
class SignatureView constructor(context: Context?, attrs: AttributeSet? = null) :
View(context, attrs, 0) {
private lateinit var paint: Paint
private var mWidth = 0 //父类宽度
private var mHeight = 0//父类高度
override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec)
mWidth = MeasureSpec.getSize(widthMeasureSpec)
mHeight = MeasureSpec.getSize(heightMeasureSpec)
paint = Paint()
//设置抗锯齿
paint.isAntiAlias = true
//设置签名笔画样式
paint.style = Paint.Style.STROKE
setMeasuredDimension(mWidth, mHeight)
}
override fun onDraw(canvas: Canvas) {
super.onDraw(canvas)
......
}
}
接下来需要做的其实就只有触摸事件处理和绘制路径两个任务了。绘制需要在onDraw中通过咱们初始化的画笔paint和canvas完成,那么触摸事件则需要重写onTouchEvent方法监听:
override fun onTouchEvent(event: MotionEvent): Boolean {
super.onTouchEvent(event)
}
这里需要注意,由于签名是实时记录的,所以我们需要记录下滑动的点,将其连接成一条线,再通过pain画线。画的过程是按下移动的操作,所以在ACTION_MOVE的过程我们需要实时画线。画笔完成是抬起的动作,所以在ACTION_UP时我们需要将之前的所有点绘制在画布上。这里通过一个点的集合收集手指移动的点,用户画笔绘制。
//收集用户绘制的点
private var allPoints: MutableList<Point> = ArrayList()
val p = Point(event.x.toInt(), event.y.toInt())
when (event.action) {
MotionEvent.ACTION_DOWN -> {
//用户按下,表示重新开始保存点
allPoints.clear()
allPoints.add(p)
}
MotionEvent.ACTION_UP -> {
postInvalidate() //重绘图像
}
MotionEvent.ACTION_MOVE -> {
allPoints.add(p)
postInvalidate() //子线程可刷新
}
}
接下来就是在onDrow()中绘制,通过canvas.drawLine:
val first = allPoints[0] //第一个点
val last = allPoints[allPoints.size-1] //最后一个点
canvas.drawLine(first.x.toFloat(),first.y.toFloat(),last.x.toFloat(),last.y.toFloat(),paint)
到这里基本就结束了,但我们还需要撤回和清除的功能。这时候就需要想一想了,如果按照按下—>移动—>抬起为一笔,那么撤回的话就需要将这个过程的所有点都清除。此时我们就可以将这个过程想象成一条线,在按下时记录线的开始,抬起记录线的结束,再将这条线存放在一个集合中,如果需要撤回就将这条数据删除。
//用于画完图后的显示,在抬起时,添加来自于allPoints的所有点的数据
private var allList: MutableList<List<Point>> = ArrayList()
//用于划线过程中的显示,当集合allList存储后,清空本集合中所有数据
private var allPoints: MutableList<Point> = ArrayList()
override fun onTouchEvent(event: MotionEvent): Boolean {
super.onTouchEvent(event)
val p = Point(event.x.toInt(), event.y.toInt())
when (event.action) {
MotionEvent.ACTION_DOWN -> {
//用户按下,表示重新开始保存点
allPoints = ArrayList()
allPoints.add(p)
}
MotionEvent.ACTION_UP -> {
//用户松开
allList.add(allPoints)
allPoints = ArrayList() //添加集合后,清除子集合
postInvalidate() //重绘图像
}
MotionEvent.ACTION_MOVE -> {
allPoints.add(p)
postInvalidate() //子线程可刷新
}
}
return true
}
//上一步(清除本次绘画)
fun clearLatestData() {
if (allList.size > 0) {
allList.removeAt(allList.size - 1)
paints.removeAt(paints.size - 1)
}
postInvalidate()
}
清除就更简单了,直接清空集合中的所有线条即可,但一定不要忘记刷新视图:
//重置(清除所有绘画)
fun clearAllData() {
allList.clear()
postInvalidate()
}
来源:https://juejin.cn/post/7202260878930591802


猜你喜欢
- java8的stream取max public static void main(String[] args) { &
- 获取非公平锁(基于JDK1.7.0_40)非公平锁和公平锁在获取锁的方法上,流程是一样的;它们的区别主要表现在“尝试获取锁的机制不同”。简单
- 本文实例讲述了Android开发实现浏览器全屏显示功能。分享给大家供大家参考,具体如下:业务需求:浏览器设置中支持全屏显示的功能。分析:只需
- 一、下载步骤首先明确自己的操作系统下载地址:点击跳转进入界面后我们可以看到有ultimate版本(收费)和community版本(免费),学
- Mybatis的Dao层实现传统开发方式1、编写UserDao接口public interface UserMapper {public L
- 本文以实例详述了C#实现Socket通信的解决方法,具体实现步骤如下:1、首先打开VS新建两个控制台应用程序:ConsoleApplicat
- 一、 添加 maven 依赖<dependency> <groupId>com.google.guava
- 将索引设置为只读可能听起来很奇怪,但在 Elasticsearch 中执行此类操作是可能的。想象一下这样一种情况,你特别需要限制对索引的写入
- 以前的左右滑动效果采用自定义scrollview或者linearlayout来实现,recyclerview可以很好的做这个功能,一般的需求
- 本文实例讲述了Android实现的仿淘宝购物车。分享给大家供大家参考,具体如下:夏的热情渐渐退去,秋如期而至,丰收的季节,小编继续着实习之路
- using System; using System.IO; using System.Data; using System.Text; u
- java模拟TCP通信实现客户端上传文件到服务器端,供大家参考,具体内容如下客户端package com.zr;import java.io
- 过滤器、 * 、 * 概念概念1、servlet:servlet是一种运行服务器端的java应用程序,具有独立于平台和协议的特性,可以动态生
- 使用自定义注解实现接口限流在高并发系统中,保护系统的三种方式分别为:缓存,降级和限流。限流的目的是通过对并发访问请求进行限速或者一个时间窗口
- Cloneable这个接口设计得十分奇葩,不符合正常人的使用习惯,然而用这个接口的人很多也很有必要,所以还是有必要了解一下这套扭曲的机制。以
- // 获取国家省市区信息$(document).ready(function(){//从程序
- 最近有很多同学,竟然不知道如何使用Intellij IDEA打开Java项目并启动现在来讲一下,希望不要忘记了 1、打开IDEA开机页面 M
- 第一步:在d盘新建android文件夹,在此文件夹中再建三个子文件夹,Android Studio 、 Android_SDK、Androi
- 本章讲述:FileStream类的基本功能,以及简单示例;1、引用命名空间:using System.IO;2、注意:使用IO操作文件时,要
- 为什么使用Swagger 在实际开发中我们作为后端总是给前端或者其他系统提供接口,每次写完代码之后不可避