真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網站制作重慶分公司

Android實現(xiàn)掃一掃功能之繪制指定區(qū)域透明區(qū)域

一、概述

創(chuàng)新互聯(lián)公司是一家專業(yè)提供都安企業(yè)網站建設,專注與做網站、成都網站設計、H5技術、小程序制作等業(yè)務。10年已為都安眾多企業(yè)、政府機構等服務。創(chuàng)新互聯(lián)專業(yè)網站設計公司優(yōu)惠進行中。

在實現(xiàn)掃一掃的功能的時候,我們需要繪制一個中間為透明的掃碼框,其余部分為半透明。通常情況下,例如微信或者支付寶的掃碼框都是矩形的,如果中間的掃碼框是一個矩形,那么布局是很簡單的,可是如果掃碼框是一個圓角矩形,或者圓形等情況怎么辦呢?這篇文章主要是記錄繪制一個中間透明帶圓角的矩形。

按照慣例,我們先來看看效果圖 :

Android實現(xiàn)掃一掃功能之繪制指定區(qū)域透明區(qū)域

二、按照流程我們就開始來看看代碼啦

1、CustomDrawable,支持中間出現(xiàn)透明區(qū)域的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;

/**
 * 支持中間出現(xiàn)透明區(qū)域的drawable
 * 通過{@link #setSrcPath(Path)}設定透明區(qū)域的形狀
 * 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,然后將指定的區(qū)域混合成透明色

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;

/**
 * 根據(jù)layout中子View的位置,確定局部透明區(qū)域
 * 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為范圍構造需要透明顯示的區(qū)域
  View view = findViewById(R.id.iv_scan);
  if (view != null) {
   path = new Path();
   // 矩形透明區(qū)域
   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"?>


 

 

  
  

 

好了,本篇文章就這樣了,存在不足的地方還望指導,感謝^_^

附錄:

自定義Drawable之:在Drawable中部指定透明區(qū)域

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對創(chuàng)新互聯(lián)的支持。


本文標題:Android實現(xiàn)掃一掃功能之繪制指定區(qū)域透明區(qū)域
轉載來源:http://weahome.cn/article/jeieih.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部