现在很主流使用圆形头像,还要带边框的一个圆圈圈,自己做的项目里就有这样的需求,大大小小好多地方。
本着能懒则懒的原则,把使用详解总结出来,(PS:不是控件详解)。
基本上可以拿来就用,注释我也添加的比较详细了。


文末有Demo包下载

先上个简单的效果图:


搞起:
在CircleImageView自定义类内,
有一些CircleImageView,CircleImageView_border_width,CircleImageView_border_color..等styleable参数,
需要在资源文件夹res下values里面新创建attrs文件,并添加参数设置;

attrs.xml:

<?xml version="1.0" encoding="utf-8"?>
<resources><!--圆形头像的设置--><declare-styleable name="CircleImageView"><attr name="border_width" format="dimension" /><attr name="border_color" format="color" /></declare-styleable></resources>

上干货:

  • CircleImageView自定义View类:
    (其实就是直接粘贴过去用就好了)
/*** Created by zhenqi on 2017/3/20 0020.*/public class CircleImageView extends ImageView {/*** 默认边框宽度为0*/private static final int DEFAULT_BORDER_WIDTH = 0;/*** 默认边框颜色透明*/private static final int DEFAULT_BORDER_COLOR = Color.TRANSPARENT;private static final ScaleType SCALE_TYPE = ScaleType.CENTER_CROP;private static final Bitmap.Config BITMAP_CONFIG = Bitmap.Config.ARGB_8888;private static final int COLORDRAWABLE_DIMENSION = 1;private final RectF mDrawableRect = new RectF();private final RectF mBorderRect = new RectF();private final Matrix mShaderMatrix = new Matrix();private final Paint mBitmapPaint = new Paint();private final Paint mBorderPaint = new Paint();private int mBorderColor = DEFAULT_BORDER_COLOR;private int mBorderWidth = DEFAULT_BORDER_WIDTH;private Bitmap mBitmap;private BitmapShader mBitmapShader;private int mBitmapWidth;private int mBitmapHeight;private float mDrawableRadius;private float mBorderRadius;private boolean mReady;private boolean mSetupPending;public CircleImageView(Context context) {super(context);}public CircleImageView(Context context, AttributeSet attrs) {this(context, attrs, 0);}public CircleImageView(Context context, AttributeSet attrs, int defStyle) {super(context, attrs, defStyle);super.setScaleType(SCALE_TYPE);//此处的CircleImageView,CircleImageView_border_width...等styleable参数,// 需要在资源文件夹res下values里面新创建attrs文件,并添加参数设置TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CircleImageView, defStyle, 0);mBorderWidth = a.getDimensionPixelSize(R.styleable.CircleImageView_border_width, DEFAULT_BORDER_WIDTH);mBorderColor = a.getColor(R.styleable.CircleImageView_border_color, DEFAULT_BORDER_COLOR);a.recycle();mReady = true;if (mSetupPending) {setup();mSetupPending = false;}}@Overridepublic ScaleType getScaleType() {return SCALE_TYPE;}@Overridepublic void setScaleType(ScaleType scaleType) {if (scaleType != SCALE_TYPE) {throw new IllegalArgumentException(String.format("ScaleType %s not supported.", scaleType));}}@Overrideprotected void onDraw(Canvas canvas) {if (getDrawable() == null) {return;}canvas.drawCircle(getWidth() / 2, getHeight() / 2, mDrawableRadius, mBitmapPaint);canvas.drawCircle(getWidth() / 2, getHeight() / 2, mBorderRadius, mBorderPaint);}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);setup();}public int getBorderColor() {return mBorderColor;}public void setBorderColor(int borderColor) {if (borderColor == mBorderColor) {return;}mBorderColor = borderColor;mBorderPaint.setColor(mBorderColor);invalidate();}public int getBorderWidth() {return mBorderWidth;}public void setBorderWidth(int borderWidth) {if (borderWidth == mBorderWidth) {return;}mBorderWidth = borderWidth;setup();}@Overridepublic void setImageBitmap(Bitmap bm) {super.setImageBitmap(bm);mBitmap = bm;setup();}@Overridepublic void setImageDrawable(Drawable drawable) {super.setImageDrawable(drawable);mBitmap = getBitmapFromDrawable(drawable);setup();}@Overridepublic void setImageResource(int resId) {super.setImageResource(resId);mBitmap = getBitmapFromDrawable(getDrawable());setup();}private Bitmap getBitmapFromDrawable(Drawable drawable) {if (drawable == null) {return null;}if (drawable instanceof BitmapDrawable) {return ((BitmapDrawable) drawable).getBitmap();}try {Bitmap bitmap;if (drawable instanceof ColorDrawable) {bitmap = Bitmap.createBitmap(COLORDRAWABLE_DIMENSION, COLORDRAWABLE_DIMENSION, BITMAP_CONFIG);} else {bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight(), BITMAP_CONFIG);}Canvas canvas = new Canvas(bitmap);drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());drawable.draw(canvas);return bitmap;} catch (OutOfMemoryError e) {return null;}}private void setup() {if (!mReady) {mSetupPending = true;return;}if (mBitmap == null) {return;}mBitmapShader = new BitmapShader(mBitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);mBitmapPaint.setAntiAlias(true);mBitmapPaint.setShader(mBitmapShader);mBorderPaint.setStyle(Paint.Style.STROKE);mBorderPaint.setAntiAlias(true);mBorderPaint.setColor(mBorderColor);mBorderPaint.setStrokeWidth(mBorderWidth);mBitmapHeight = mBitmap.getHeight();mBitmapWidth = mBitmap.getWidth();mBorderRect.set(0, 0, getWidth(), getHeight());mBorderRadius = Math.min((mBorderRect.height() - mBorderWidth) / 2, (mBorderRect.width() - mBorderWidth) / 2);mDrawableRect.set(mBorderWidth, mBorderWidth, mBorderRect.width() - mBorderWidth, mBorderRect.height() - mBorderWidth);mDrawableRadius = Math.min(mDrawableRect.height() / 2, mDrawableRect.width() / 2);updateShaderMatrix();invalidate();}private void updateShaderMatrix() {float scale;float dx = 0;float dy = 0;mShaderMatrix.set(null);if (mBitmapWidth * mDrawableRect.height() > mDrawableRect.width() * mBitmapHeight) {scale = mDrawableRect.height() / (float) mBitmapHeight;dx = (mDrawableRect.width() - mBitmapWidth * scale) * 0.5f;} else {scale = mDrawableRect.width() / (float) mBitmapWidth;dy = (mDrawableRect.height() - mBitmapHeight * scale) * 0.5f;}mShaderMatrix.setScale(scale, scale);mShaderMatrix.postTranslate((int) (dx + 0.5f) + mBorderWidth, (int) (dy + 0.5f) + mBorderWidth);mBitmapShader.setLocalMatrix(mShaderMatrix);}}

  • -

布局文件xml:


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"xmlns:app="http://schemas.android.com/apk/res-auto"android:id="@+id/activity_main"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"tools:context="com.zhenqi.circleimageviewdemo.MainActivity"><!--头像--><!--这里注释:--><!--border_color:表示外边框的颜色--><!--border_width:表示外边框的宽度--><!--其他同正常的view设置一样:--><!--src:给了一张方形图片,显示的圆形的--><!--background:这里注意-还是整个布局的背景--><com.zhenqi.circleimageviewdemo.CircleImageView
        android:id="@+id/iv_pers_portrait"android:layout_width="150dp"android:layout_height="150dp"android:background="#ffff00"android:src="@drawable/tupian"app:border_color="#000000"app:border_width="20dp" />
</LinearLayout>

效果图:


MainActivity:

public class MainActivity extends AppCompatActivity {private CircleImageView iv_pers_portrait;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}private void initView() {iv_pers_portrait = (CircleImageView) findViewById(R.id.iv_pers_portrait);}
}

点击此处,下载Demo

Android 圆形头像/有外边框的圆形头像CircleImageView自定义控件使用详解相关推荐

  1. Android基础入门教程——8.3.1 三个绘图工具类详解

    Android基础入门教程--8.3.1 三个绘图工具类详解 标签(空格分隔): Android基础入门教程 本节引言: 上两小节我们学习了Drawable以及Bitmap,都是加载好图片的,而本节我 ...

  2. 【Android语音合成TTS】百度语音接入方法,和使用技巧详解

    请尊重他人的劳动成果,转载请注明出处:[Android语音合成TTS]百度语音接入方法,和使用技巧详解 Ps. 依托于百度开放云,百度语音为合作伙伴提供了业界领先.永久免费的语音技术服务,目前已上线的 ...

  3. 【Android游戏开发十二】(保存游戏数据 [上文])详解SharedPreference 与 FIleInputStream/FileOutputStream将数据存储到SD卡中!

     李华明Himi 原创,转载务必在明显处注明: 转载自 [黑米GameDev街区] 原文链接:  http://www.himigame.com/android-game/327.html 很多童鞋说 ...

  4. android 6.0 存储卡,Android 6.0区别U盘和SD卡设备的方法详解

    如下所示: public static boolean isSdcardExists(Context context) { StorageManager storageManager = Storag ...

  5. CSS圆角边框、盒子阴影及文字阴影详解

    文章目录 1. 圆角边框 2. 盒子阴影 3. 文字阴影 1. 圆角边框 在CSS3 中,新增了圆角边框样式,这样我们的盒子就可以变圆角了. border-radius 属性用于设置元素的外边框圆角. ...

  6. 华为Android岗面经;群面+技术面+英语面+面试题详解

    前言 昨天是我去华为面试的整整一天,早上9点开始出发,一直弄到下午5点半,终于完成了所有的测评和面试! 简单说一下,我面的是Android高级开发职位,一个华为的前辈内推的.目前5年半开发经验.面试之 ...

  7. android json mysql_Android通过json向MySQL中读写数据的方法详解【读取篇】

    本文实例讲述了Android通过json向MySQL中读取数据的方法.分享给大家供大家参考,具体如下: 首先 要定义几个解析json的方法parseJsonMulti,代码如下: private vo ...

  8. Android逆向之旅---Native层的Hook神器Cydia Substrate使用详解

    一.前言 在之前已经介绍过了Android中一款hook神器Xposed,那个框架使用非常简单,方法也就那几个,其实最主要的是我们如何找到一个想要hook的应用的那个突破点.需要逆向分析app即可.不 ...

  9. Android 驱动(12)---Linux DTS(Device Tree Source)设备树详解

    Linux DTS(Device Tree Source)设备树详解 Linux DTS(Device Tree Source)设备树详解之一(背景基础知识篇) Linux DTS(Device Tr ...

最新文章

  1. 使用python-nmap模块扫描端口脚本
  2. python中float与eval式一样的吗_用Python最原始的函数模拟eval函数的浮点数运算功能...
  3. js获取url中的参数值
  4. 第12个双11,天猫的“造新”运动
  5. java做a_Java编程实现A*算法完整代码
  6. nginx alias
  7. 前端学习(2168):url的hash和html的history
  8. 【转】PF_RING学习笔记
  9. java自学路线图_JAVA自学路线图
  10. STM32工作笔记0083---UCOSIII中断和时间管理
  11. 9.Shell 编程从入门到精通 --- 进程
  12. LabVIEW程序快速开发流程
  13. 云计算到底有哪些魅力 云计算就业前景好不好
  14. 内存占用少的linux桌面,不同桌面环境占用内存/CPU对比
  15. 对学生公寓部建议和意见_关于进一步加强学生公寓管理工作的意见
  16. linux下mysql(rpm)安装使用手册
  17. 【LINUX】一、文件安全和权限
  18. KAPPA领衔实施服装ERP软件树立行业榜样
  19. 第一节 模式识别的基本概念
  20. 刘润年度演讲2021:进化的力量(演讲全文)

热门文章

  1. 紫书:二分图匹配 最大流解决
  2. Apple_MAC_PRO_苹果工作站历代型号
  3. 计算机视觉基础__图像特征
  4. Word中NoteExpress不显示的问题
  5. 【oracle 11g】配置Oracle lsnrctl 和自动启动
  6. Springboot3.0 打包后,tomcat启动失败
  7. 飞书远程办公,流畅的协作体验
  8. blender怎样给平面或曲面自动贴图
  9. 下面是一对情侣的聊天记录(看后也许该反思下了)分享下。
  10. java itextpdf签章 根据关键字定位在pdf文件中的坐标