Android中圆形图的几种实现方式
在Android开发中,圆形图片是很常见的,例如淘宝的宝贝,QQ的联系人头像等都是圆形的图片,
但是Android原生的ImageView又不能显示圆形的图片,这就需要我们自己去实现一个圆形图了
一、自定义View实现圆形图
1.1继承ImageView
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.widget.ImageView;/*** Created by ChenFengYao on 16/3/15.*/
public class RoundImageView extends ImageView {private Paint paint;public RoundImageView(Context context) {super(context);paint = new Paint();//初始化画笔对象}public RoundImageView(Context context, AttributeSet attrs) {super(context, attrs);paint = new Paint();//初始化画笔对象}public RoundImageView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);paint = new Paint();//初始化画笔对象}
}
继承ImageView复写其中的构造方法,并在构造方法里对画笔对象进行初始化
1.2复写onDraw方法
@Overrideprotected void onDraw(Canvas canvas) {Drawable drawable = getDrawable();if (null != drawable) {Bitmap bitmap = ((BitmapDrawable) drawable).getBitmap();//核心代码Bitmap b = getCircleBitmap(bitmap);paint.reset();canvas.drawBitmap(b,0,0,paint);} else {super.onDraw(canvas);}}
onDraw方法即图片绘制的时候系统所调用的方法,在该方法内部首先去判断是否设置了图片的src,如果能拿到改View设置的图片,则将它转换成圆形图片,如果没有设置的话,则不做任何操作,直接调用父类的onDraw方法
1.2getCircleBitmap
private Bitmap getCircleBitmap(Bitmap bitmap){Bitmap output = Bitmap.createBitmap(bitmap.getWidth(),bitmap.getHeight(), Bitmap.Config.ARGB_8888);Canvas canvas = new Canvas(output);Rect rect = new Rect(0,0,bitmap.getWidth(),bitmap.getHeight());paint.setAntiAlias(true);canvas.drawCircle(bitmap.getWidth() / 2, bitmap.getHeight() / 2, bitmap.getHeight() / 2, paint);paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));canvas.drawBitmap(bitmap, rect, rect, paint);//将图片画出来return output;
}
这是获取圆形图的方法,目的是将我们自定义的View中的图片,转化成圆形的Bitmap,这里需要Canvas对象,首先画一个圆形的底层,再在其上放上我们的图片,通过设置画笔的paint的Xfermode属性,该属性是用来设置前后图层的显示关系的,这是设置Mode.SRC_IN,的意思是输出的范围是底层图形的范围,而显示的内容是上层的内容
1.2.1paint.setXfermode()
PorterDuff.Mode.CLEAR |
所绘制不会提交到画布上。 |
PorterDuff.Mode.SRC |
显示上层绘制图片 |
PorterDuff.Mode.DST |
显示下层绘制图片 |
PorterDuff.Mode.SRC_OVER |
正常绘制显示,上下层绘制叠盖 |
PorterDuff.Mode.DST_OVER |
上下层都显示。下层居上显示 |
PorterDuff.Mode.SRC_IN |
取两层绘制交集。显示上层 |
PorterDuff.Mode.DST_IN |
取两层绘制交集。显示下层 |
PorterDuff.Mode.SRC_OUT |
取上层绘制非交集部分 |
PorterDuff.Mode.DST_OUT |
取下层绘制非交集部分 |
PorterDuff.Mode.SRC_ATOP |
取下层非交集部分与上层交集部分 |
PorterDuff.Mode.DST_ATOP |
取上层非交集部分与下层交集部分 |
PorterDuff.Mode.XOR |
取两层绘制非交集。两层绘制非交集 |
PorterDuff.Mode.DARKEN |
上下层都显示。变暗 |
PorterDuff.Mode.LIGHTEN |
上下层都显示。变亮 |
PorterDuff.Mode.MULTIPLY |
取两层绘制交集 |
PorterDuff.Mode.SCREEN |
上下层都显示 |
1.2.2Canvas
1.3测试
<com.lanou.chenfengyao.temproundimageview.RoundImageViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:src="@mipmap/test_img"/>
1.4 添加功能
<resources><declare-styleable name="RoundImageView"><attr name="is_round" format="boolean" /></declare-styleable>
</resources>
private boolean isRound;
然后在构造方法里提取这条属性,如果没有提取到,默认值设置为true,让它默认就可以显示圆角图片
public RoundImageView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);paint = new Paint();//初始化画笔对象TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.RoundImageView);isRound = typedArray.getBoolean(R.styleable.RoundImageView_is_round, true);
}
为了增加实用性 我们改造一下剩下的构造方法
public RoundImageView(Context context) {this(context, null);
}public RoundImageView(Context context, AttributeSet attrs) {this(context, attrs, 0);
}
@Overrideprotected void onDraw(Canvas canvas) {Drawable drawable = getDrawable();if (null != drawable && isRound) {Bitmap bitmap = ((BitmapDrawable) drawable).getBitmap();//核心代码Bitmap b = getCircleBitmap(bitmap);paint.reset();canvas.drawBitmap(b, 0, 0, paint);b.recycle();} else {super.onDraw(canvas);}}
<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);">测试一下,我们将我们的组件的is_round属性改成false看看效果</span>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"xmlns:app="http://schemas.android.com/apk/res-auto"android:paddingBottom="@dimen/activity_vertical_margin"android:paddingLeft="@dimen/activity_horizontal_margin"android:paddingRight="@dimen/activity_horizontal_margin"android:paddingTop="@dimen/activity_vertical_margin"tools:context="com.lanou.chenfengyao.temproundimageview.MainActivity"><com.lanou.chenfengyao.temproundimageview.RoundImageViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:src="@mipmap/test_img"app:is_round="false"/>
</RelativeLayout>
注意,我们自定义的属性,命名空间不要忘记了
可以看到,现在的RoundImageView就和正常的ImageView的效果是一样的啦.
public void setIsRound(boolean isRound) {this.isRound = isRound;invalidate();}
现在我们在Activity里放上一个按钮,点击它切换显示模式,来测试一下
可以看到 我们的图片可以通过Java代码来动态的切换正常模式和圆形图片啦
二、使用Fresco显示圆形图片
compile 'com.facebook.fresco:fresco:0.9.0+'
另外Fresco在使用的时候需要对其进行初始化,可以在需要的Activity里的onCreate方法里添加
Fresco.initialize(this);
建议将这行代码添加到Application里进行全局的初始化
<com.facebook.drawee.view.SimpleDraweeViewfresco:roundAsCircle = "true"fresco:actualImageScaleType="centerInside"fresco:roundingBorderColor="@color/colorAccent"fresco:roundingBorderWidth="1dp"android:layout_width="match_parent"android:layout_height="match_parent"fresco:backgroundImage="@mipmap/test_img"/>
并且不要忘记命名空间
xmlns:fresco="http://schemas.android.com/apk/res-auto"
看一下效果
可以看到出现了圆形图片,并且还有1dp的红色边框
2.1一些坑
三、利用sharp做圆形图
<shapeandroid:innerRadius="0dp"android:shape="ring"android:thicknessRatio="1"android:useLevel="false"xmlns:android="http://schemas.android.com/apk/res/android"><solid android:color="@android:color/transparent" /><strokeandroid:width="68dp"android:color="#FFFFFFFF" />
</shape>
让shape是ring即环形,并且设置填充颜色为白色,线宽需要根据图片自己调整
<layer-list xmlns:android="http://schemas.android.com/apk/res/android"><item android:drawable="@mipmap/test_img" /><item android:drawable="@drawable/circle" />
</layer-list>
layer-list的意思是让ImageView中显示的图片产生一个叠加的效果,越靠下写的,在显示的时候就越靠上层
<ImageViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:src="@drawable/round_layers"/>
看一下效果吧
实际上,该方法的原理就是在正常的ImageView上再叠加了一个白色的环形图形,这样显示的效果就是圆形图片了.但是缺点也是显而易见的,即需要手动的调节环形的粗细,比较难控制
Android中圆形图的几种实现方式相关推荐
- Android中夜间模式的三种实现方式
参考:https://www.jianshu.com/p/f3aaed57fa15 在本篇文章中给出了三种实现日间/夜间模式切换的方案: 使用 setTheme 的方法让 Activity 重新设置主 ...
- android 软件 加密方法,Android中WIFI常见的几种加密方式(详细)
系统设置中WIFI热点的加密方式,添加新的WIFI和连接WIFI的时候,需要在程序中配置相应的参数,可以打开自己的手机对着看 包:android.net.wifi.WifiConfiguration ...
- Android中WIFI常见的几种加密方式(详细)
系统设置中WIFI热点的加密方式,添加新的WIFI和连接WIFI的时候,需要在程序中配置相应的参数,可以打开自己的手机对着看 包:android.net.wifi.WifiConfiguration ...
- [译] 绘制路径:Android 中矢量图渲染
原文地址:Draw a Path: Rendering Android VectorDrawables 原文作者:Nick Butcher 译文出自:掘金翻译计划 本文永久链接:github.com/ ...
- Android 圆形头像的两种实现方式
Android 圆形头像的两种实现方式 前言 这篇博客只是为了做一个记录而已,方便而后查询,核心代码都是直接采用鸿洋博客里面的代码的. 圆形头像在实际开发中实际很常见,一般来说,主要有两种实现方式: ...
- Android实现圆形图像的两种方法(Glide和Picasso)
Android实现圆形图像的两种方法 先上效果图 Glide Picasso CircleTransform.java(圆形图片工具类) 先上效果图 Glide 在app的build.gradle中引 ...
- lisp遍历表中所有顶点_三十张图片让你彻底弄明白图的两种遍历方式:DFS和BFS...
1 引言 遍历是指从某个节点出发,按照一定的的搜索路线,依次访问对数据结构中的全部节点,且每个节点仅访问一次. 在二叉树基础中,介绍了对于树的遍历.树的遍历是指从根节点出发,按照一定的访问规则 ...
- android xpath解析xml,Android 中处理 XML 的四种方式-DOM
Android 中处理 XML 的几种方式连载中,我们就不介绍合成 XML 了,因为合成 XML 可以直接拼接字符串,虽然看起很不高大上,但却很有效.我们主要介绍如何取 XML 中的值. 适用 DOM ...
- Android中播放音乐的几种方式
Android中播放音乐的几种方式 前言 前几天一直在研究RxJava2,也写了记录了几篇博客,但因为工作任务原因,需要研究音频相关的知识,暂时放下Rxjava,本文的demo中,MediaPalye ...
最新文章
- 工作中MySql的了解到的小技巧
- 设计模式 之 建造者
- Python_03-数据类型
- Arduino可穿戴教程之第一个程序——选择端口(三)
- 用户控件(.ascx)调用网页(.aspx)的方法
- 下载达 10 万次的 IDEA 插件,K8s 一键部署了解一下?
- 认识因特网络(小学计算机课件),小学信息技术认识因特网ppt课件.ppt
- (王道408考研数据结构)第五章树-第四节2:平衡二叉树(AVL)及其旋转
- 【面试】面试文章积累
- 概率图模型(03): 模板模型(动态贝叶斯, 隐马尔可夫和Plate模型)
- Linux/Ubuntu20 安装 TP-link(RTL8812AU) 无线网卡驱动
- LabVIEW 2021 工具包
- Ubuntu登录界面键盘鼠标失灵
- Mac 安装LaTeX教程 mactex+sublime+skim
- Unity项目美术资源优化篇
- 【Data URL】【RE】【bugku】逆向入门writeup
- Openwrt Kernel panic - not syncing: Fatal exception 解决方案
- Spring JDBC与事务管理
- 如何进入游戏行业成为一个程序员
- SLM328美格4G模组SDK开发笔记