前言

在日常开发中,圆形的图片效果还是很常见的。可以通过给Paint设置Xfermode来实现,这里简单记录如下。

实现

实现圆形效果的核心是PorterDuffXfermode,对于PorterDuffXfermode,这里不展开,可以查询相关资料。

核心代码

//绘制背景

canvas.drawCircle(mSize / 2, mSize / 2, mSize / 2, mPaint);

//设置模式为:显示背景层和上层的交集,且显示上层图像

mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));

//绘制要显示的图像

canvas.drawBitmap(mSrcBitmap, 0, 0, mPaint);

//重置Xfermode

mPaint.setXfermode(null);

自定义属性

自定义控件

public class CircleView extends View {

private static final int DEFAULT_SIZE = 200;

private static final int DEFAULT_RADIUS = 20;

private static final int TYPE_ROUND = 1;

private static final int TYPE_RECT = 2;

private int mSize;

private int mResourceId;

private int mType;

private Paint mPaint;

private Bitmap mSrcBitmap;

public CircleView(Context context) {

this(context, null);

}

public CircleView(Context context, AttributeSet attrs) {

this(context, attrs, 0);

}

public CircleView(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.CircleView);

mResourceId = ta.getResourceId(R.styleable.CircleView_src, R.mipmap.ic_launcher);

mType = ta.getInt(R.styleable.CircleView_type, TYPE_ROUND);

ta.recycle();

init();

}

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

int width = getMeasureSize(widthMeasureSpec);

int height = getMeasureSize(heightMeasureSpec);

mSize = Math.min(width, height);

setMeasuredDimension(mSize, mSize);

}

@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)

@Override

protected void onDraw(Canvas canvas) {

super.onDraw(canvas);

//绘制背景

if (mSrcBitmap == null) {

mSrcBitmap = getScaleBitmap();

}

if (mType == TYPE_ROUND) {

canvas.drawCircle(mSize / 2, mSize / 2, mSize / 2, mPaint);

} else if (mType == TYPE_RECT) {

canvas.drawRoundRect(0, 0, mSize, mSize, DEFAULT_RADIUS, DEFAULT_RADIUS, mPaint);

}

//设置模式为:显示背景层和上层的交集,且显示上层图像

mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));

//绘制要显示的图像

canvas.drawBitmap(mSrcBitmap, 0, 0, mPaint);

//重置Xfermode

mPaint.setXfermode(null);

}

private void init() {

//禁用硬件加速,否则可能无法绘制圆形

setLayerType(LAYER_TYPE_HARDWARE, null);

mPaint = new Paint();

mPaint.setAntiAlias(true);

mPaint.setStyle(Paint.Style.FILL);

}

private int getMeasureSize(int measureSpec) {

int mode = MeasureSpec.getMode(measureSpec);

int size = MeasureSpec.getSize(measureSpec);

return mode == MeasureSpec.EXACTLY ? size : DEFAULT_SIZE;

}

/**

* 获取缩放后的Bitmap

*

* @return

*/

private Bitmap getScaleBitmap() {

BitmapFactory.Options options = new BitmapFactory.Options();

options.inJustDecodeBounds = true;

BitmapFactory.decodeResource(getResources(), mResourceId, options);

options.inSampleSize = calcSampleSize(options, mSize, mSize);

options.inJustDecodeBounds = false;

return BitmapFactory.decodeResource(getResources(), mResourceId, options);

}

/**

* 计算缩放比例

*

* @param option

* @param width

* @param height

* @return

*/

private int calcSampleSize(BitmapFactory.Options option, int width, int height) {

int originWidth = option.outWidth;

int originHeight = option.outHeight;

int sampleSize = 1;

while ((originWidth = originWidth >> 1) > width && (originHeight = originHeight >> 1) > height) {

sampleSize = sampleSize << 1;

}

return sampleSize;

}

}

注意:如果没有圆形的效果,那么可能需要禁用硬件加速:setLayerType(LAYER_TYPE_HARDWARE, null)

布局

xmlns:app="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:gravity="center_horizontal"

android:orientation="vertical"

tools:context=".MainActivity">

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_margin="10dp"

app:src="@drawable/image" />

android:layout_width="100dp"

android:layout_height="100dp"

android:layout_margin="10dp"

app:src="@drawable/image" />

android:layout_width="100dp"

android:layout_height="100dp"

android:layout_margin="10dp"

app:src="@drawable/image"

app:type="rect" />

效果

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

android 圆形图片按钮,Android自定义View圆形图片控件代码详解相关推荐

  1. Android开源中国客户端学习 (自定义View)左右滑动控件ScrollLayout

    左右滑动的控件我们使用的也是非常多了,但是基本上都是使用的viewpager 等 android基础的控件,那么我们有么有考虑过查看他的源码进行定制呢?当然,如果你自我感觉非常好的话可以自己定制一个, ...

  2. android生命周期_Android开发 View的生命周期结合代码详解

    咱们以TextView控件为例: /** * Created by SunshineBoy on 2020/9/23. */ public class TestTextView extends and ...

  3. android控件使用大全,Android常见控件使用详解

    本文实例为大家分享了六种Android常见控件的使用方法,供大家参考,具体内容如下 1.TextView 主要用于界面上显示一段文本信息 2.Button 用于和用户交互的一个按钮控件 //为Butt ...

  4. android自定义控件中文乱码,Android笔记--自定义View之组合控件

    Android-自定义View 分享是最好的记忆-- 如需转发请注明出处 [强调]:共同学习 共同进步 不喜勿喷 内容简介 前言 实现 总结 1. 前言 这次更新有2个目的 1. 复用控件,而不是每次 ...

  5. Android自定义View 实现窗帘控件

    需求分析 这里只作简单介绍,最后会分享源码地址 窗帘分为三部分,上面的窗帘杆,杆下面的窗帘布,以及布中间的滑块,实现还是蛮简单的,我们可以通过自定义view把这个窗帘画出来 窗帘杆是一个上面是圆角,下 ...

  6. android自定义xml弹窗,Android自定义弹窗提醒控件使用详解

    Android中原生的Dialog弹窗提醒控件样式单一,有时候并不能满足我们的项目需求,而且一个工程里面有时候会在多处都用到弹窗提醒的功能,代码会出现大量的冗余,工作之余,就自己实现了这么一个弹窗提醒 ...

  7. android gridview控件使用详解_Android开发实现自定义日历、日期选择控件

    点击上方蓝字关注 ?? 来源: wenzhihao123 https://www.jianshu.com/p/a2f102c728ce 前言 最近项目需要日历效果,考虑用第三方的反而不太适合设计需求, ...

  8. android gridview控件使用详解_作为Android 开发者该如何进阶?

    经常在简书和微信上收到一些同学的私信,说自己马上毕业或者已经毕业一年,从事Android开发相关的工作,现在不知道要学习什么东西了.或者说自己也在摸索着学习,但是不知道学习的路线对不对,感觉很迷茫,想 ...

  9. 一种基于自定义View的贴纸控件Demo

    其实就是自定义一套触摸事件规则,加上对Matrix的使用即可. 首先定义基类,首先不同类型的图元,例如Bitmap或者文本,需要的缩放.移动.测量.绘制方式可能都不一致,所以做成抽象函数顶个接口规范, ...

  10. Android开发的之基本控件和详解四种布局方式

    Android中的控件的使用方式和iOS中控件的使用方式基本相同,都是事件驱动.给控件添加事件也有接口回调和委托代理的方式.今天这篇博客就总结一下Android中常用的基本控件以及布局方式.说到布局方 ...

最新文章

  1. 200行代码,一行行教你自制微信机器人
  2. 章节之外:makefile中的函数定义
  3. oracle数据库修改写入状态,【学习笔记】Oracle oradebug 使用oradebug修改数据库SCN方法案例...
  4. activemqcpp编译及可能的错误处理
  5. 提高开源项目逼格-为你的github项目添加Travis CI
  6. VMware文件扩展名
  7. 逆向Android软件的步骤
  8. 【信息系统项目管理师】第10章 下篇-项目干系人管理 知识点详细整理
  9. Keras 学习笔记
  10. 【Codeforces Round #516_div2_E】【二分交互题】Dwarves, Hats and Extrasensory Abilities
  11. git合并分支,发布代码
  12. webqq linux,Ubuntu下WebQQ桌面化替代方案完美版
  13. Junit 单元测试以及断言
  14. Embergen 流体模拟工具
  15. java 线程的插队运行_java笔记--线程的插队行为
  16. 安卓或苹果IOS的APP应用如何取名字?好的名字技巧?
  17. android wifi无法连接手机号码,安卓手机无法使用WiFi连接ApowerMirror解决方法
  18. 《Web性能测试实战》性能测试用例模板
  19. 雷达信号处理&重学傅里叶变换(一):Radar cube &multidimensional Fourier transform
  20. C++中含有无符号类型的表达式——有符号数与无符号数相加

热门文章

  1. Atitit android app 最佳实践2021目录1. Android strudio,,and viruse machine need down another... 11.1. P
  2. Atitit db query op shourt code lib list 数据库查询最佳实践 JdbcTemplate spring v2 u77 .docx Atitit db query o
  3. Atitit web 之道 艾龙著 Atitit web 之道 艾龙艾提拉著v2 saa.docx 1. 第1章 Web编程基础知识 (1) 3 1.1. 1.1 什么是Web (1) 3 1.2.
  4. Atitit 工作流之道 艾提拉著 BPM,即业务流程管理 目录 1. 流程入门 思想 历史 分类 1 第二篇 第2章 初识工作流 2 1.1. 2.3 工作流技术相关规范  2.3.1 W
  5. Atitit 翻页功能的解决方案与版本历史 v4 r49
  6. Atitit.研发团队与公司绩效管理的原理概论的attilax总结
  7. atitit.编程语言 程序语言 的 工具性 和 材料性 双重性 and 语言无关性 本质
  8. PAIP.MYSQL数据库比较
  9. (转)刘巍然-关于公钥与私钥
  10. (转)每个人都应该读一读贝索斯的致股东信 1997-2016