Android 自定义View——蒙版擦除效果实现

2023-10-29 05:21

本文主要是介绍Android 自定义View——蒙版擦除效果实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本博客转自http://blog.csdn.net/to_be_designer/article/details/48553657

控件功能介绍

  • 首先介绍一下控件的功能:

    1. 在View中有背景图片和蒙版,通过手指触碰屏幕和滑动,可以将背景上层的蒙版擦除进而显示出背景图片。
    2. 可以在xml布局文件中设置背景图片,且背景只能是mipmap中的图片。
    3. 可以设置蒙版的颜色。
    4. 可以设置擦除画笔的宽度大小。

        功能就这么多,接下来我们看代码的实现……

擦除功能实现

  1. 创建一个MyBitmapViewAnother继承View。(这里命名不太规则,不要介意……)
  2. 实现自定义View中的构造器。
    public MyBitmapViewAnother(Context context) {super(context);}/*该构造器必须实现,因为AttributeSet代表了xml布局文件在引用布局文件时传入的参数集合。当我们在布局文件中设置控件的相关参数:宽、高、颜色等时,我们要通过AttributeSet对象获得。*/public MyBitmapViewAnother(Context context, AttributeSet attrs) {super(context, attrs);}
  1. 重写View的onMeasure(int widthMeasureSpec, int heightMeasureSpec)和onDraw(Canvas canvas)方法。onMeasure()方法实现是在自定义View构造器调用结束之后调用的,只有onMeasure()方法调用结束之后,我们才可以获得xml布局中设置的我们这个控件的宽和高。onDraw()方法是绘制我们的View界面,通过方法中传入的Canvas对象绘制我们的View。当我们使用自定义的控件时,UI主线程会调用onDraw()方法绘制控件的界面。

  2. 定义一个Bitmap代表背景图片,将Bitmap添加到中View的Canvas上。

  3. 然后在定义一个Bitmap,为这个Bitmap定义一个Canvas,我们将在这个Bitmap的Canvas上绘制蒙版,和我们手指的路径。然后通过PorterDuff.Mode将我们手指划过的路径与蒙版相交的部分去除掉。

代码:

public class MyBitmapViewAnother extends View {private int width;//设置高private int height;//设置高private Paint mPaint;//设置一个Bitmapprivate Bitmap bitmap;//创建该Bitmap的画布private Canvas bitmapCanvas;private Paint mPaintCover;private Paint mPaintRect;//定义一样个背景的Bitmapprivate Bitmap mBitmapBackground;private Matrix matrix;private Path mPath;public MyBitmapViewAnother(Context context) {super(context);}public MyBitmapViewAnother(Context context, AttributeSet attrs) {super(context, attrs);mPaint = new Paint();//Bitmap的画笔//设置背景mBitmapBackground = BitmapFactory.decodeResource(getResources(), R.mipmap.bb);mPaintCover = new Paint();mPaintCover.setAntiAlias(true);mPaintCover.setColor(Color.GRAY);mPaintCover.setStrokeWidth(50);//设置图形混合方式,这里使用PorterDuff.Mode.XOR模式,与底层重叠部分设为透明PorterDuffXfermode mode = new PorterDuffXfermode(PorterDuff.Mode.XOR);mPaintCover.setXfermode(mode);//这三句代码很重要,别忘记写了mPaintCover.setStyle(Paint.Style.STROKE);//设置笔刷的样式,默认为BUTT,如果设置为ROUND(圆形),SQUARE(方形),需要将填充类型Style设置为STROKE或者FILL_AND_STROKEmPaintCover.setStrokeCap(Paint.Cap.ROUND);//设置画笔的结合方式mPaintCover.setStrokeJoin(Paint.Join.ROUND);//绘制蒙版的画笔mPaintRect = new Paint();mPaintRect.setAntiAlias(true);mPaintRect.setColor(Color.LTGRAY);//路径记录滑动屏幕的路径。mPath = new Path();}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);width = getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec);height = getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec);setMeasuredDimension(width, height);//设置宽和高//创建一个Bitmap,用于绘图。bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);bitmapCanvas = new Canvas(bitmap);//该画布为bitmap。//绘制背景BitmapBackground大小的矩阵matrix = new Matrix();//如果在构造器中初始化,需要使用reset()方法matrix.postScale((float)width/mBitmapBackground.getWidth(), (float)height/mBitmapBackground.getHeight());}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);//将bitmapBackground设置该View画布的背景canvas.drawBitmap(mBitmapBackground,matrix,null);//然后画布添加背景的基础上添加bitmap。canvas.drawBitmap(bitmap, 0, 0, mPaint);bitmapCanvas.drawRect(0, 0, width, height, mPaintRect);//bitmap上绘制一个蒙版bitmapCanvas.drawPath(mPath, mPaintCover);//bitmap上绘制手 划过的路径}//这里设置初始值是为了不点击屏幕时 ,不显示路径private float down_x=-100;private float down_y=-100;private float move_x=-100;private float move_y=-100;@Overridepublic boolean onTouchEvent(MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN://获得点击屏幕时的坐标down_x= event.getX();down_y=event.getY();//将Path移动到点击点mPath.moveTo(down_x, down_y);invalidate();//更新画面return true;case MotionEvent.ACTION_MOVE://获得在屏幕上移动的坐标move_x= event.getX();move_y=event.getY();//将移动的轨迹画成直线mPath.lineTo(move_x, move_y);//然后移动到下一个点。mPath.moveTo(move_x, move_y);invalidate();//更新画面return true;}return super.onTouchEvent(event);}
}

这里写图片描述

  接下来就是我们要通过在xml布局文件中为我们的控件自定义设置背景图片,蒙版颜色,以及擦除画笔宽度的属性,这里就用到了自定义View中,自定义View属性的实现了。
  

自定义属性实现

这里就直接上实现的步骤了:

  1. 在res资源文件中的values文件夹中创建一个mybitmapviewanother_attrs的xml文件。在这个xml文件中定义一个的标签,并定义标签的”name”。在标签内添加添加属性,“name”为属性名,“format”为属性值的类型。
      注意:这里的属性名“name”不要定义“background”之类的,因为Android中已经使用的这写名字,我们在去使用的时候就会出错。
<?xml version="1.0" encoding="utf-8"?>
<resources><declare-styleable name="MyBitmapViewAnother"><!--背景图片属性,其属性值可以是res中的资源文件--><attr name="mybitmapviewanother_background" format="reference"></attr><!--擦除画笔宽度属性,其属性值可以是具体的值"10dp",也可以是res中的资源文件--><attr name="mybitmapviewanother_paintwidth" format="dimension|reference"></attr></declare-styleable>
</resources>
  1. 在使用自定义View的布局文件中添加标签:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"<!--首先声明这里添加注释是错误的,使用时请删除。下面这一句就是对控件属性的声明。格式:xmlns:自定义的名字(就像上面的android)="http://schemas.android.com/apk/res-auto"-->xmlns:mybitmapviewanother="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"tools:context="com.example.administrator.mywidgetdemo.activity.PathActivity"><com.example.administrator.mywidgetdemo.bitmap.MyBitmapViewAnother
        android:id="@+id/mypicture"android:layout_width="match_parent"android:layout_height="match_parent"mybitmapviewanother:mybitmapviewanother_background="@mipmap/aa"/><!--使用时,通过如上方式使用。注意:冒号签名的名字和声明时的名字是相同的。-->
</LinearLayout>

3.在然后就是在代码的构造器中获取我们布局中的属性了

        //通过自定义的属性文件,来获取相关的属性值if (attrs!=null){TypedArray array=getContext().obtainStyledAttributes(attrs, R.styleable.MyView);int width= (int) array.getDimension(R.styleable.MyView_myview_paintwidth,30);coverPaint.setStrokeWidth(width);int pic=array.getResourceId(R.styleable.MyView_myview_pic,R.mipmap.ic_launcher);backGroundBitmap= BitmapFactory.decodeResource(getResources(), pic);}

  这样 我们通过布局文件更改画笔宽度,背景图片:
  
这里写图片描述

这篇关于Android 自定义View——蒙版擦除效果实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/298507

相关文章

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义