Android实现扫一扫功能之绘制指定区域透明区域

一、概述

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

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

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

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

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_width="match_parent"
 android:id="@+id/frame_layout"
 android:layout_height="match_parent">

 <ImageView
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:background="@mipmap/bg_image" />

 <per.juan.scandome.CustomLayout
  android:layout_width="match_parent"
  android:id="@+id/layout"
  android:background="#8c565658"
  android:layout_height="match_parent">

  <!-- 根据这个子View所在的位置,计算出透明矩形的位置 -->
  <FrameLayout
   android:id="@+id/iv_scan"
   android:layout_width="200dp"
   android引用块内容center" />

 </per.juan.scandome.CustomLayout>
</FrameLayout>

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

附录:

自定义Drawable之:在Drawable中部指定透明区域

总结

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

(0)

相关推荐

  • Android开发仿扫一扫实现拍摄框内的照片功能

    就是仿照现在扫一扫的形式,周围是半透明的遮挡,然后中间是全透明的,拍摄后只截取框内的内容 查了很多博客,实现起来真的太复杂了,本人比较怕麻烦所以在很多地方偷懒了 先上效果图: 第一步:设置照相机预览以及拍照 这是所有步骤的前提,没有预览,用户怎么知道自己拍的什么呢.预览用的是SurfaceView 这篇博文写得已经十分详细了,打开照相机,然后拍照,而且十分简洁!不想别的博客一下就几百行代码不知所云.这篇代码可以复制下去当相机模版使用. 这里遇到一个问题,就是预览的效果是左转90度的,拍出来也是左

  • Android实现扫一扫识别数字功能

    1.准备工作 首先实现识别数字等字符,我们要知道需要采用OCR (Optical Character Recognition,光学字符识别)来实现.而tesseract是非常不错的开源OCR工具,但是要在Android中直接使用可能要费点功夫.不过不用担心,tess-two拯救了我们. 其次是扫一扫识别,那么很快联想到的就是常见的二维码扫描这类的项目.通过扫一扫实时拿到图像,来做识别. 接下来在Github上找到了QrCodeScanner项目,作者通过一定的优化,使得识别的效率有所提升.那么我

  • Android自定义Drawable之在Drawable中部指定透明区域方法示例

    前言 Drawable是什么? 一种可以在Canvas上进行绘制的抽象的概念 颜色.图片等都可以是一个Drawable Drawable可以通过XML定义,或者通过代码创建 Android中Drawable是一个抽象类,每个具体的Drawable都是其子类 Drawable的优点 使用简单,比自定义View成本低 非图片类的Drawable所占空间小,能减小apk大小 在实际的开发工程中,不免想有一个中间是空洞的Drawable,也就是中间是透明的,而其他区域正常显示的Drawable. 主要用

  • Android实现扫一扫功能之绘制指定区域透明区域

    一.概述 在实现扫一扫的功能的时候,我们需要绘制一个中间为透明的扫码框,其余部分为半透明.通常情况下,例如微信或者支付宝的扫码框都是矩形的,如果中间的扫码框是一个矩形,那么布局是很简单的,可是如果扫码框是一个圆角矩形,或者圆形等情况怎么办呢?这篇文章主要是记录绘制一个中间透明带圆角的矩形. 按照惯例,我们先来看看效果图 : 二.按照流程我们就开始来看看代码啦 1.CustomDrawable,支持中间出现透明区域的drawable package per.juan.scandome; impor

  • Android集成zxing扫码框架功能

    我们知道zxing是一个强大的处理二维码和条形码等的开源库,本篇文章记录一下自己在项目中集成zxing开源库的过程. 导入依赖 implementation 'com.google.zxing:core:3.3.3' 申请权限 在AndroidManifest中申请相应权限: <!--相机--> <uses-permission android:name="android.permission.CAMERA" /> <!--震动--> <use

  • 超简单Android集成华为HMS Scankit 扫码SDK实现扫一扫二维码

    前言 最近要做一个停车场扫码收费的app,在网上搜了一圈,首先接触到了ZXing,上手试了下,集成过程不复杂,但是感觉效果欠佳,比如距离稍微远点儿就扫不出来了,另外角度对的不好,反光或者光线比较暗的时候,成功率也比较低,集成好给我们老大看了下,感觉不是很满意.最近偶尔看到了华为一个发布会里面有介绍扫码功能,稍微研究了下,居然是一款免费扫码神器,官方称之为Scan Kit,号称还同时支持Android和iOS,半信半疑上手搞了一把发现效果还真不错!测了下发现对于一些有反光,污损,形变,超远距离的场

  • vue使用微信扫一扫功能的实现代码

    第一步: 安装weixin-js-sdk 和 jquery 包 npm install weixin-js-sdk jquery 第二部: 配置wx.config (配置都是后端返回来的,菜鸟前端只需要按需传值过去就可) 代码如下 import wx from "weixin-js-sdk"; import $ from "jquery"; goSao() { //这里[url参数一定是去参的本网址],请求后端接口换取signature //(兼容安卓和ios) l

  • iOS实现“摇一摇”与“扫一扫”功能示例代码

    "摇一摇"功能的实现: iPhone对 "摇一摇"有很好的支持,总体说来就两步: 在视图控制器中打开接受"摇一摇"的开关; - (void)viewDidLoad { // 设置允许摇一摇功能 [UIApplication sharedApplication].applicationSupportsShakeToEdit = YES; // 并让自己成为第一响应者 [self becomeFirstResponder]; } 在"摇一摇

  • jQuery实现“扫码阅读”功能

    今天看到一个用户发了个话题,"PC端的URL在移动端上打开要一个个敲好麻烦,有什么好的办法?". 确实现在已经是移动时代了,在移动设备上阅读慢慢会成为主流,网站如果没有便捷的方式让用户在移动设备阅读的话还真有点落伍,于是想想就做个"扫码阅读"的功能吧.其实很简单,就是将网址生成二维码就行了. 无论用PHP生成,还是用JavaScript生成都是可以的,从代码改动来说,用JavaScript会更省事些.所以这里就用jQuery吧.正好网上有个 jquery.qrcod

  • 微信JSSDK调用微信扫一扫功能的方法

    如何利用微信JSSDK调用微信扫一扫功能?具体内容如下 1. 确保有 调起微信扫一扫接口 权限,测试号可能不行: 2. 导入相关JS <script type="text/javascript" http://test.com/zepto_touch.js"></script> <script type="text/javascript" src="http://res.wx.qq.com/open/js/jweix

  • JS 调用微信扫一扫功能

    1.第一步: 设置调用微信js安全域名,就可以在该域名下调用微信的js接口 2.第二步: 将下面的js附在需要调用微信扫一扫的页面上,前提是需要引入微信的js[] 3.第三步: 参考官方文档 https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421141115&token=876521668&lang=zh_CN 通过后台获取js权限签名jsapi_ticket[参考文档:http://mp.weixin.qq.com/

随机推荐