Android开发使用自定义View将圆角矩形绘制在Canvas上的方法

作者:伟雪无痕 时间:2021-06-08 01:03:17 

本文实例讲述了Android开发使用自定义View将圆角矩形绘制在Canvas上的方法。分享给大家供大家参考,具体如下:

前几天,公司一个项目中,头像图片需要添加圆角,这样UI效果会更好看,于是写了一个小的demo进行圆角的定义,该处主要是使用BitmapShader进行了渲染(如果要将一张图片裁剪成椭圆或圆形显示在屏幕上,也可以使用BitmapShader来完成).

BitmapShader类完成渲染图片的基本步骤如下:

1、创建BitmapShader类的对象


/**
  * Call this to create a new shader that will draw with a bitmap.
  *
  * @param bitmap      The bitmap to use inside the shader
  * @param tileX       The tiling mode for x to draw the bitmap in.
  * @param tileY       The tiling mode for y to draw the bitmap in.
  */
 public BitmapShader(Bitmap bitmap, TileMode tileX, TileMode tileY) {
......
}

其中,Shader.TitleMode类型有三种,CALMP、MIRROR、REPEAT

CALMP:使用边界颜色来填充剩余空间
MIRROR:使用镜像方式
REPEAT:使用重复方式

2、通过Paint的setShader(bitmapShafer)来设置画笔

3、使用已经setShader(bitmapShafer)的画笔来绘制图形

下面展示绘制圆角图片的demo

1、自定义RounderCornerImageView.java类


package com.example.test;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapShader;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.RectF;
import android.graphics.Shader;
import android.util.AttributeSet;
import android.view.View;
public class RounderCornerImageView extends View {
 private Bitmap mImage;// source bitmap
 private Paint mBitmapPaint;//paint
 private RectF mBrounds;//rect
 private float mRadius=20.0f;//round
 public RounderCornerImageView(Context context) {
   this(context, null);
 }
 public RounderCornerImageView(Context context, AttributeSet attrs) {
   this(context, attrs, 0);
 }
 public RounderCornerImageView(Context context, AttributeSet attrs,
     int defStyleAttr) {
   super(context, attrs, defStyleAttr);
   init();
 }
 private void init() {
   mBitmapPaint=new Paint(Paint.ANTI_ALIAS_FLAG);
   mBrounds=new RectF();
 }
 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
   // TODO Auto-generated method stub
   int height,width;
   height=width=0;
   //obtain bitmap size
   int imageHeight,imageWidth;
   if (null!=mImage) {
     imageHeight=imageWidth=0;
   }else
   {
     imageHeight=mImage.getHeight();
     imageWidth=mImage.getWidth();
   }
   //obtain best measure data and set on View
   width=getMeasurement(widthMeasureSpec,imageWidth);
   height=getMeasurement(heightMeasureSpec, imageHeight);
   //set View last size
   setMeasuredDimension(width, height);
 }
 /**
  * measure width and height by specMode
  **/
 private int getMeasurement(int measureSpec, int contentSize) {
   int specSize=MeasureSpec.getSize(measureSpec);
   switch (MeasureSpec.getMode(measureSpec)) {
   case MeasureSpec.AT_MOST:
     return Math.min(specSize, contentSize);
   case MeasureSpec.UNSPECIFIED:
     return contentSize;
   case MeasureSpec.EXACTLY:
     return specSize;
   default:
     return 0;
   }//switch
 }
 @Override
 protected void onSizeChanged(int w, int h, int oldw, int oldh) {
   if (w!=oldw || h!=oldh) {
     int imageWidth,imageHeight;
     if (null==mImage) {
       imageWidth=imageHeight=0;
     }else
     {
       imageWidth=mImage.getWidth();
       imageHeight=mImage.getHeight();
     }
     //center point
     int left=(w-imageWidth)/2;
     int top=(h-imageHeight)/2;
     mBrounds.set(left, top, left+imageWidth, top+imageHeight);
     if (null!=mBitmapPaint.getShader()) {
       Matrix m=new Matrix();
       m.setTranslate(left, top);
       mBitmapPaint.getShader().setLocalMatrix(m);
     }
   }
 }
 public void setImage(Bitmap bitmap) {
   if (mImage!=bitmap) {
     mImage=bitmap;
     if (null!=mImage) {
       BitmapShader shader=new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);
       mBitmapPaint.setShader(shader);
     }else {
       mBitmapPaint.setShader(null);
     }
     requestLayout();//invalidated the layout of this view by onDraw()
   }
 }
 @Override
 protected void onDraw(Canvas canvas) {
   super.onDraw(canvas);
   if (null!=mBitmapPaint) {
     //draw Round Rect
     canvas.drawRoundRect(mBrounds, mRadius, mRadius, mBitmapPaint);
   }
 }
}

2、显示圆角图片的RoundActivity.java类


package com.example.test;
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Bundle;
public class RoundActivity extends Activity{
 @Override
 protected void onCreate(Bundle savedInstanceState) {
   super.onCreate(savedInstanceState);
   RounderCornerImageView view=new RounderCornerImageView(this);
   Bitmap souBitmap=BitmapFactory.decodeResource(getResources(), R.drawable.sun);
   view.setImage(souBitmap);
   setContentView(view);
 }
}

另外,附注下自定义View的一些基本步骤和必须实现的方法

1、继承view

2、重写自定义View的构造方法

3、如需要对view进行位置进行测量和重写布局,则需要重写onMeasure()onLayout()onDraw()方法

onMeasure():view本身大小多少,可以测量出来
onLayout():view在ViewGroup中的位置可以决定
onDraw():定义了如何绘制该view

希望本文所述对大家Android程序设计有所帮助。

来源:http://blog.csdn.net/j086924/article/details/49868543

标签:Android,自定义View,Canvas
0
投稿

猜你喜欢

  • java冷知识:javac AbstractProcessor详解

    2022-08-01 19:32:09
  • Android仿通话来电界面效果

    2022-01-02 07:48:51
  • 基于Springboot实现JWT认证的示例代码

    2023-04-01 09:46:40
  • JDK 7U15在 Windows x86平台下的安装方法

    2023-04-09 07:31:08
  • C#使用Word中的内置对话框实例

    2023-08-11 08:04:20
  • 图解Java经典算法冒泡排序的原理与实现

    2023-03-14 21:41:23
  • springboot自定义Starter过程解析

    2023-07-24 22:24:55
  • 详解如何继承Mybatis中Mapper.xml文件

    2022-12-22 14:48:34
  • spring依赖注入知识点分享

    2023-11-26 15:32:32
  • Spring Security添加验证码的两种方式小结

    2021-08-05 17:24:25
  • IDEA快速搭建spring boot项目教程(Spring initializr)

    2023-08-17 21:11:16
  • 解决spring boot启动扫描不到自定义注解的问题

    2023-10-29 14:31:48
  • spring @Conditional的使用与扩展源码分析

    2022-01-09 05:52:57
  • java中常见的死锁以及解决方法代码

    2023-04-07 19:47:30
  • Java实现的求逆矩阵算法示例

    2023-05-02 03:02:56
  • Maven继承与聚合详解及作用介绍

    2023-03-08 00:14:36
  • IDEA 2022 中的Lombok 使用基础教程

    2023-04-09 21:57:09
  • Proxy实现AOP切面编程案例

    2023-07-23 06:44:52
  • Spring代理对象导致的获取不到原生对象注解的解决

    2021-12-05 11:44:19
  • Java递归寻路实现,你真的理解了吗

    2022-09-17 02:24:34
  • asp之家 软件编程 m.aspxhome.com