Android CircleImageView自定义圆形与边框(圆形头像)详解

背景:

当前我们在使用图片过程中,都知道正常的ImageView是一个矩形RectF,RectF只需要知道两个对角坐标就可以确定该图形的大小

RectF rect=new RectF(int left,int top,int right,int bottom);

四个参数可以确定四个角的坐标和矩形的长以及宽。

但是我们在使用过程中进行会看见圆形或者不规则图片,比如三角形等图片展示,这是如何实现的呢?

其实这些图片的基础都是矩形图片,如果我们用剪刀能在这边图片进行任意裁剪,那最后的图片就是我们想要显示的嘛?其实这个思路是正确的。

那我们如何来实现这个呢?这就需要我们重写画布(即对画布的裁剪)

说明:我们在使用过程中,只要是对画布的裁剪,如果你不放心,可以把裁剪过的ImageView的drawable区回来,设置在正常的显示即可。

这个时候我们可以测试,发现。最终裁剪的是画布

1.思路:裁剪

裁剪这边其实需要我们用一个路径,即Canvas裁剪需要有一个路径,只有知道尺寸,裁缝才会沿着路径剪出精美的画布。

这时,我们需要借助一个工具类Path

Path path = new Path();
path.addCircle(width / 2, height / 2, width / 2 , Path.Direction.CW);

我们通过path路径,绘制出一个圆出来

path.addCircle(cx,cy,radius , Path.Direction.CW);

cx,cy:代表坐标

radius :圆的半径

Path.Direction.CW:代表绘制放心,是顺时针还是逆时针

注意:在做半径的时候,我们应该取宽或者高的最小值做半径

int radius=Math.min(width,height);

这时候圆已出来了,接下来就是裁剪

canvas.clipPath(path);

一定在绘制之前进行裁剪,否则裁剪会不生效。

这样,一个简单的裁剪就生成了 

2.边框颜色

借用上图,左上角有一个圆,并有一个红色的边框。

我们知道作图是在画布上的,所以需要在画布sper以后,生成新画布我们再进行绘图。

绘图我们需要画笔Paint

Paint paint = new Paint();
paint.setColor(Color.RED);//画笔颜色
paint.setStrokeWidth(10);//画笔的笔尖宽
paint.setStyle(Paint.Style.STROKE);//样式
paint.setAntiAlias(true);//锯齿
path.addCircle(width / 2, height / 2, width / 2, Path.Direction.CCW);
canvas.drawPath(path, paint);

1.初始化一个画笔,设置好画笔的参数

2.我们依旧用path画圆,

3.最后通过画布进行绘制。

完整代码如下:

@Override
protected void onDraw(Canvas canvas) {

    int width = getMeasuredWidth();
    int height = getMeasuredHeight();
    Path path = new Path();
    path.addCircle(width / 2, height / 2, width / 2 , Path.Direction.CCW);
    canvas.clipPath(path);
    super.onDraw(canvas);

    //如果需要在图形上进行二次绘画,需要调用super后,也就是剪切后的画布已生成,再进行创作
    Paint paint = new Paint();
    paint.setColor(Color.RED);
    paint.setStrokeWidth(10);
    paint.setStyle(Paint.Style.STROKE);
    paint.setAntiAlias(true);
    path.addCircle(width / 2, height / 2, width / 2, Path.Direction.CCW);
    canvas.drawPath(path, paint);


}

解释:

有人会发现其他demo中会出现大量的代码,或者画笔的出现,以及BitmapShader来管理,我这边只是简单的处理,自身的熟悉依旧依赖ImageView去调试,你也可以通过BitmapShader生成你们业务需要的样式,拉伸或者填满等效果。这只是介绍入门

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个实现圆形边框图片裁剪自定义View的完整代码: ``` public class CircleImageView extends ImageView { private Paint mBorderPaint; private int mBorderColor; private int mBorderWidth; private Bitmap mBitmap; private BitmapShader mBitmapShader; private int mBitmapWidth; private int mBitmapHeight; private float mRadius; private RectF mBorderRect; public CircleImageView(Context context) { super(context); init(); } public CircleImageView(Context context, AttributeSet attrs) { super(context, attrs); init(); } public CircleImageView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } private void init() { mBorderPaint = new Paint(); mBorderPaint.setAntiAlias(true); mBorderRect = new RectF(); } @Override protected void onDraw(Canvas canvas) { if (mBitmapShader == null) { mBitmap = getBitmap(); if (mBitmap != null) { mBitmapShader = new BitmapShader(mBitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP); mBitmapWidth = mBitmap.getWidth(); mBitmapHeight = mBitmap.getHeight(); mRadius = Math.min(mBitmapWidth, mBitmapHeight) / 2; updateShaderMatrix(); } } if (mBitmapShader != null) { mBorderPaint.setShader(mBitmapShader); canvas.drawCircle(mRadius, mRadius, mRadius, mBorderPaint); mBorderPaint.setShader(null); mBorderPaint.setColor(mBorderColor); mBorderPaint.setStrokeWidth(mBorderWidth); mBorderRect.set(0, 0, getWidth(), getHeight()); canvas.drawArc(mBorderRect, 0, 360, false, mBorderPaint); } } private void updateShaderMatrix() { float scale; float dx = 0; float dy = 0; if (mBitmapWidth * getHeight() > getWidth() * mBitmapHeight) { scale = getHeight() / (float) mBitmapHeight; dx = (getWidth() - mBitmapWidth * scale) * 0.5f; } else { scale = getWidth() / (float) mBitmapWidth; dy = (getHeight() - mBitmapHeight * scale) * 0.5f; } Matrix matrix = new Matrix(); matrix.setScale(scale, scale); matrix.postTranslate((int) (dx + 0.5f), (int) (dy + 0.5f)); mBitmapShader.setLocalMatrix(matrix); } public void setBorderColor(int borderColor) { if (borderColor == mBorderColor) { return; } mBorderColor = borderColor; invalidate(); } public void setBorderWidth(int borderWidth) { if (borderWidth == mBorderWidth) { return; } mBorderWidth = borderWidth; invalidate(); } private Bitmap getBitmap() { Drawable drawable = getDrawable(); if (drawable == null) { return null; } if (drawable instanceof BitmapDrawable) { return ((BitmapDrawable) drawable).getBitmap(); } Bitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(bitmap); drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight()); drawable.draw(canvas); return bitmap; } } ``` 使用方法: 在布局文件中添加自定义View: ``` <com.example.CircleImageView android:id="@+id/circle_image_view" android:layout_width="100dp" android:layout_height="100dp" android:src="@drawable/avatar" app:border_color="#ffffff" app:border_width="4dp" /> ``` 其中 `app:border_color` 和 `app:border_width` 分别表示边框的颜色和宽度,可以根据需要调整。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值