一、概述

在实现扫一扫的功能的时候,我们需要绘制一个中间为透明的扫码框,其余部分为半透明。通常情况下,例如微信或者支付宝的扫码框都是矩形的,如果中间的扫码框是一个矩形,那么布局是很简单的,可是如果扫码框是一个圆角矩形,或者圆形等情况怎么办呢?这篇文章主要是记录绘制一个中间透明带圆角的矩形。

按照惯例,我们先来看看效果图 :

二、按照流程我们就开始来看看代码啦

1、CustomDrawable,支持中间出现透明区域的drawable

package per.juan.scandome;

import android.graphics.Canvas;

import android.graphics.ColorFilter;

import android.graphics.Paint;

import android.graphics.Path;

import android.graphics.PorterDuff;

import android.graphics.PorterDuffXfermode;

import android.graphics.drawable.Drawable;

import android.support.annotation.NonNull;

import android.support.annotation.Nullable;

/**

* 支持中间出现透明区域的drawable

* 通过{@link #setSrcPath(Path)}设定透明区域的形状

* Created by juan on 2018/07/20.

*/

public class CustomDrawable extends Drawable {

private Paint srcPaint;

private Path srcPath = new Path();

private Drawable innerDrawable;

public CustomDrawable(Drawable innerDrawable) {

this.innerDrawable = innerDrawable;

srcPath.addRect(100, 100, 200, 200, Path.Direction.CW);

srcPaint = new Paint(Paint.ANTI_ALIAS_FLAG);

srcPaint.setColor(0xffffffff);

}

/**

* 设置内部透明的部分

*

* @param srcPath

*/

public void setSrcPath(Path srcPath) {

this.srcPath = srcPath;

}

@Override

public void draw(@NonNull Canvas canvas) {

innerDrawable.setBounds(getBounds());

if (srcPath == null || srcPath.isEmpty()) {

innerDrawable.draw(canvas);

} else {

//将绘制操作保存到新的图层,因为图像合成是很昂贵的操作,将用到硬件加速,这里将图像合成的处理放到离屏缓存中进行

int saveCount = canvas.saveLayer(0, 0, canvas.getWidth(), canvas.getHeight(), srcPaint, Canvas.ALL_SAVE_FLAG);

//dst 绘制目标图

innerDrawable.draw(canvas);

//设置混合模式

srcPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));

//src 绘制源图

canvas.drawPath(srcPath, srcPaint);

//清除混合模式

srcPaint.setXfermode(null);

//还原画布

canvas.restoreToCount(saveCount);

}

}

@Override

public void setAlpha(int alpha) {

innerDrawable.setAlpha(alpha);

}

@Override

public void setColorFilter(@Nullable ColorFilter colorFilter) {

innerDrawable.setColorFilter(colorFilter);

}

@Override

public int getOpacity() {

return innerDrawable.getOpacity();

}

}

(1)主要用到的技术是PorterDuffXfermode的PorterDuff.Mode.XOR模式

(2)核心思想是先正常绘制出整个drawable,然后将指定的区域混合成透明色

2、CustomLayout

package per.juan.scandome;

import android.annotation.SuppressLint;

import android.content.Context;

import android.graphics.Path;

import android.support.annotation.NonNull;

import android.support.annotation.Nullable;

import android.util.AttributeSet;

import android.view.View;

import android.widget.FrameLayout;

/**

* 根据layout中子View的位置,确定局部透明区域

* Created by juan on 2018/07/20.

*/

public class CustomLayout extends FrameLayout {

private Context mContext;

private CustomDrawable background;

public CustomLayout(@NonNull Context context) {

super(context);

initView(context, null, 0);

}

public CustomLayout(@NonNull Context context, @Nullable AttributeSet attrs) {

super(context, attrs);

this.mContext=context;

initView(context, attrs, 0);

}

public CustomLayout(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

initView(context, attrs, defStyleAttr);

}

@SuppressLint("NewApi")

private void initView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {

background = new CustomDrawable(getBackground());

setBackground(background);

}

@Override

protected void onLayout(boolean changed, int left, int top, int right, int bottom) {

super.onLayout(changed, left, top, right, bottom);

resetBackgroundHoleArea();

}

@SuppressLint("NewApi")

private void resetBackgroundHoleArea() {

Path path = null;

// 以子View为范围构造需要透明显示的区域

View view = findViewById(R.id.iv_scan);

if (view != null) {

path = new Path();

// 矩形透明区域

path.addRoundRect(view.getLeft(), view.getTop(), view.getRight(), view.getBottom(), dp2Px(mContext,10), dp2Px(mContext,10),Path.Direction.CW);

}

if (path != null) {

background.setSrcPath(path);

}

}

public int dp2Px(Context context, float dp) {

final float scale = context.getResources().getDisplayMetrics().density;

return (int) (dp * scale + 0.5f);

}

}

3、然后在XML布局中声明我们的自定义View

android:layout_width="match_parent"

android:id="@+id/frame_layout"

android:layout_height="match_parent">

android:layout_width="match_parent"

android:layout_height="match_parent"

android:background="@mipmap/bg_image" />

android:layout_width="match_parent"

android:id="@+id/layout"

android:background="#8c565658"

android:layout_height="match_parent">

android:id="@+id/iv_scan"

android:layout_width="200dp"

android引用块内容center" />

好了,本篇文章就这样了,存在不足的地方还望指导,感谢^_^

附录:

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对找一找教程网的支持。

android 指定区域内绘制,Android实现扫一扫功能之绘制指定区域透明区域相关推荐

  1. android locale app内设置,Android app应用多语言切换功能实现

    本文实例为大家分享了Android app应用实现多语言切换功能,供大家参考,具体内容如下 1.添加多语言文件在不同的 value 文件夹下(例如 value .value-en.values-zh- ...

  2. android 经纬度的范围内,在Android里如何判断一个指定的经纬度点是否落在一个多边形区域内...

    在lbs开发中,可能要碰到这样的问题,如何判断一个指定的经纬度点是否落在一个多边形区域内,比如我在地图上画了一个多边形区域,然后给出一个经纬度点,怎样判断这个点是否在这个多边形范围之内,由于我用的是a ...

  3. 在Android里如何判断一个指定的经纬度点是否落在一个多边形区域内

    在lbs开发中,可能要碰到这样的问题,如何判断一个指定的经纬度点是否落在一个多边形区域内,比如我在地图上画了一个多边形区域,然后给出一个经纬度点,怎样判断这个点是否在这个多边形范围之内,由于我用的是a ...

  4. 百度地图高级实例2-如何利用自己的数据制作社交地图?只显示可视区域内的标注

    摘要:如果你自己的数据已经超过1万个,如何进行合理的显示?除了聚合marker外,还有一个办法.那就是,只显示可视区域内的标注. ----------------------------------- ...

  5. 如何利用自己的数据制作社交地图?只显示可视区域内的标注

    地址:http://www.cnblogs.com/milkmap/archive/2012/02/02/2335989.html [百度地图API]如何利用自己的数据制作社交地图?只显示可视区域内的 ...

  6. 【百度地图API】如何利用自己的数据制作社交地图?只显示可视区域内的标注...

    原文:[百度地图API]如何利用自己的数据制作社交地图?只显示可视区域内的标注 摘要:如果你自己的数据已经超过1万个,如何进行合理的显示?除了聚合marker外,还有一个办法.那就是,只显示可视区域内 ...

  7. 【百度地图API】如何利用自己的数据制作社交地图?只显示可视区域内的标注

    [百度地图API]如何利用自己的数据制作社交地图?只显示可视区域内的标注 摘要:如果你自己的数据已经超过1万个,如何进行合理的显示?除了聚合marker外,还有一个办法.那就是,只显示可视区域内的标注 ...

  8. 【Android 内存优化】自定义组件长图组件 ( 获取图像宽高 | 计算解码区域 | 设置图像解码属性 复用 像素格式 | 图像绘制 )

    文章目录 一.获取图像真实宽高 二.计算解码区域 三.设置解码参数 内存复用 像素格式 四.图像绘制 五.执行效果 六.源码及资源下载 官方文档 API : BitmapRegionDecoder 在 ...

  9. Android系统将内置滤镜功能

    from: http://cblog.cc/2015/09/03/Android-Camera-%E5%AE%9E%E6%97%B6%E6%BB%A4%E9%95%9C/ 一.Android系统将内置 ...

最新文章

  1. 学python的游戏app_Python教学软件
  2. [代码]ural 1655 Somali Pirates
  3. 更新Oracle的Date字段
  4. boost::math::non_central_chi_squared用法的测试程序
  5. revit如何根据坐标进行画线_在工程设计中如何根据工艺阀门的结构与特点来进行设计呢?...
  6. 一个适合于Python初学者的入门练手项目
  7. html两行文字右侧对齐,HTML在同一行左右对齐文本
  8. vaspkit使用_VASP 的光学性质计算及 vaspkit 的安装与使用
  9. 三 spring源码解析--- Bean解析接口结构分析
  10. sql中exists与in的区别
  11. 如何根据相机的参数知道摄像机的内参数矩阵
  12. 矩阵公式(转置公式+求导公式)
  13. 中国知网如何下载外文文献
  14. 什么样的人适合学编程?
  15. (莫队算法)CodeForces - 617E XOR and Favorite Number
  16. c#推箱子小游戏代码_C# 简单推箱子游戏源码
  17. excel制作特殊图表记录
  18. 上半年要写的博客文章26
  19. 安卓开发 投屏安卓设备到电脑端 scrcpy
  20. 最简单的P2P加密聊天软件开发准备

热门文章

  1. WIN Vista系统汇总!!![讯雷下载]
  2. 用Airtest和poco实现APP自动登录和退出
  3. 网络信息手机:web scrapper数据获取 、数据采集器、公众号信息收集
  4. 科学计算机 百分号,普通计算器上百分号(%)有什么用?
  5. 华为起诉三星是中国企业改头换面的开始
  6. CUTEFTP 权限的修改 之CuteFTP问题 修复CUTEFTP unable to transfer
  7. 网络支付结算{网银、第三方支付、超级网银}
  8. Verilog专题(二十二)Lemmings1
  9. iOS APP 启动页面的使用
  10. uni 获取本地文件_APP读取本地文件夹内视频播放