Android | 多种方式实现图片圆角矩形和圆形效果

打印 上一主题 下一主题

主题 996|帖子 996|积分 2988

在项目开发中,为了让图片表现得更加美观,通常 UI 会设计成圆角矩形或圆形效果。本文将介绍几种常见的实现方式,并提供对应的代码示例。
方式一:ViewOutlineProvider可以设置圆角矩形、椭圆、圆形等

ViewOutlineProvider 是 Android 5.0 引入的一个类,用来定义视图的轮廓(outline)。可以通过它来实现圆角矩形、椭圆、圆形等效果。
  1. //设置成扩展方法
  2. fun View.clipToRoundView(type: Int = RoundImgView.SHAPE_ROUND_RECT) {
  3.     if (Build.VERSION.SDK_INT >= 21) {
  4.         outlineProvider = object : ViewOutlineProvider() {
  5.             override fun getOutline(view: View?, outline: Outline?) {
  6.                 if (view == null) return
  7.                 if (type == RoundImgView.SHAPE_ROUND_RECT) {
  8.                     //设置一个矩形的轮廓,并指定其圆角半径
  9.                     outline?.setRoundRect(0, 0, view.width, view.height, 15.dp2px().toFloat())
  10.                 } else {
  11.                     //设置成椭圆或者圆形
  12.                     outline?.setOval(0, 0, view.width, view.height)
  13.                 }
  14.             }
  15.         }
  16.         //视图会根据outlineProvider提供的轮廓进行裁剪。任何超出轮廓的部分都会被裁剪掉
  17.         clipToOutline = true
  18.     }
  19. }
复制代码
可以看到通过outline中的setRoundRect/setOval方法来设置圆角矩形、椭圆等形状,尚有其他API方法可以自行了解。在Activity中使用:
  1. private val mIvTarget: AppCompatImageView by id(R.id.iv_round_img)
  2. private val mIvTarget2: AppCompatImageView by id(R.id.iv_round_img2)
  3.    
  4. val bitmap = BitmapFactory.decodeResource(resources, R.drawable.icon_cat_w)
  5. mIvTarget.clipToRoundView(RoundImgView.SHAPE_ROUND_RECT)
  6. mIvTarget.setImageBitmap(bitmap)
  7. mIvTarget2.clipToRoundView(RoundImgView.SHAPE_CIRCLE)
  8. mIvTarget2.setImageBitmap(bitmap)
复制代码
效果如下:

方式二:使用 Glide 进行图片加载和圆角处置惩罚

Glide 是一个强盛的图片加载库,通过它的 RequestOptions 可以轻松实现图片的圆角处置惩罚。
  1. //1、图片设置的不是CenterCrop
  2. Glide.with(this)
  3.      .load(R.drawable.icon_cat_w)
  4.      .transform(RoundedCorners(16.dp2px()))
  5.      .into(mIvTarget)
  6. //2、如果图片设置的是CenterCrop,可能会导致圆角效果被 CenterCrop 操作覆盖,最终看不到圆角效果,需要用下面的方式处理CenterCrop与圆角矩形冲突问题
  7. val requestOptions = RequestOptions().transform(CenterCrop(), RoundedCorners(16.dp2px()))
  8. Glide.with(this)
  9.      .load(R.drawable.icon_cat_w)
  10.      .apply(requestOptions)
  11.      .into(mIvTarget)
复制代码
方式三:Canvas.clipPath()

自定义 ImageView 并重写 onDraw() 方法,通过 Canvas.clipPath() 实现图片的圆角矩形和圆形效果。
  1. /**
  2. * 通过clipPath的方式实现圆角矩形和圆形图片
  3. *
  4. * @param context
  5. * @param attrs
  6. * @param defStyleAttr
  7. */
  8. class RoundImgView @JvmOverloads constructor(
  9.     context: Context,
  10.     attrs: AttributeSet? = null,
  11.     defStyleAttr: Int = 0
  12. ) : AppCompatImageView(context, attrs, defStyleAttr) {
  13.     companion object {
  14.         const val SHAPE_ROUND_RECT = 0
  15.         const val SHAPE_CIRCLE = 1
  16.     }
  17.     private val path = Path()
  18.     private val strokePath = Path()
  19.     private var cornerRadius = 10.dp2px().toFloat()
  20.     private var mStrokeWidth = 10.dp2px().toFloat()
  21.     private var mShapeType = SHAPE_ROUND_RECT
  22.     private var isHasStroke = false //是否设置描边
  23.     // 创建一个画笔
  24.     val paint = Paint(Paint.ANTI_ALIAS_FLAG).apply {
  25.         color = Color.BLUE
  26.         style = Paint.Style.STROKE
  27.         strokeWidth = mStrokeWidth
  28.     }
  29.     override fun onDraw(canvas: Canvas) {
  30.         //canvas.drawColor(Color.RED)
  31.         //设置描边
  32.         if (isHasStroke) {
  33.             processPath(strokePath) //设置path
  34.             canvas.save()
  35.             canvas.drawPath(strokePath, paint)
  36.             canvas.restore()
  37.         }
  38.         processPath(path)
  39.         canvas.clipPath(path)
  40.         super.onDraw(canvas)
  41.     }
  42.     private fun processPath(path: Path) {
  43.         if (mShapeType == SHAPE_CIRCLE) {
  44.             //圆形
  45.             val radius = (maxOf(width, height) - mStrokeWidth) / 2f
  46.             path.addCircle(width / 2f, height / 2f, radius, Path.Direction.CW)
  47.         } else {
  48.             //圆角矩形
  49.             path.addRoundRect(
  50.                 mStrokeWidth, mStrokeWidth,
  51.                 width.toFloat() - mStrokeWidth,
  52.                 height.toFloat() - mStrokeWidth,
  53.                 floatArrayOf(
  54.                     cornerRadius, cornerRadius, cornerRadius, cornerRadius,
  55.                     cornerRadius, cornerRadius, cornerRadius, cornerRadius
  56.                 ),
  57.                 Path.Direction.CW
  58.             )
  59.         }
  60.     }
  61.     /**
  62.      * 设置描边宽度
  63.      * @param width 宽度
  64.      */
  65.     fun setStrokeWidth(width: Float): RoundImgView {
  66.         this.mStrokeWidth = width
  67.         paint.strokeWidth = mStrokeWidth
  68.         isHasStroke = true
  69.         return this
  70.     }
  71.     /**
  72.      * 设置圆角半径
  73.      * @param radius 半径
  74.      */
  75.     fun setCornerRadius(radius: Float): RoundImgView {
  76.         cornerRadius = radius
  77.         return this
  78.     }
  79.     /**
  80.      * 设置图片类型
  81.      * @param type
  82.      */
  83.     fun setShapeType(type: Int): RoundImgView {
  84.         this.mShapeType = type
  85.         return this
  86.     }
  87.     override fun setImageBitmap(bm: Bitmap?) {
  88.         super.setImageBitmap(bm)
  89.     }
  90. }
复制代码
Activity中使用:
  1. private val mIvCustomImg: RoundImgView by id(R.id.iv_custom_img)
  2. private val mIvCustomImg2: RoundImgView by id(R.id.iv_custom_img2)
  3. val bitmap = BitmapFactory.decodeResource(resources, R.drawable.icon_cat_w)
  4. mIvCustomImg.setCornerRadius(15.dp2px().toFloat())
  5.             .setShapeType(RoundImgView.SHAPE_ROUND_RECT)
  6.             .setStrokeWidth(10f)
  7.             .setImageBitmap(bitmap)
  8. mIvCustomImg2.setCornerRadius(15.dp2px().toFloat())
  9.             .setShapeType(RoundImgView.SHAPE_CIRCLE)
  10.             .setStrokeWidth(10f)
  11.             .setImageBitmap(bitmap)
复制代码
效果图:

方式四:CardView

通过 CardView 的 app:cardCornerRadius 属性,可以非常方便地实现圆角效果。
  1. <androidx.cardview.widget.CardView
  2.         android:layout_width="wrap_content"
  3.         android:layout_height="wrap_content"
  4.         android:layout_marginTop="50dp"
  5.         app:cardCornerRadius="10dp">
  6.    <androidx.appcompat.widget.AppCompatImageView
  7.             android:layout_width="150dp"
  8.             android:layout_height="150dp"
  9.             android:scaleType="centerCrop"
  10.             android:src="@drawable/icon_cat_w" />
  11. </androidx.cardview.widget.CardView>
复制代码
方式五:BitmapShader

BitmapShader 是 Android 中的一种着色器,通过它可以实现自定义的圆角和圆形图片。
  1.     /**
  2.      * @param bitmap  bitmap
  3.      * @param outWidth 输出的宽
  4.      * @param outHeight 输出的高
  5.      * @param radius 半径
  6.      * @param border 描边
  7.      * @param shapeType 图形类型:圆角矩形 or 圆形
  8.      */
  9.     private fun getBitmapByShader(
  10.         bitmap: Bitmap?,
  11.         outWidth: Int,
  12.         outHeight: Int,
  13.         radius: Int,
  14.         border: Int,
  15.         shapeType: Int = RoundImgView.SHAPE_ROUND_RECT
  16.     ): Bitmap? {
  17.         if (bitmap == null || bitmap.height <= 0 || bitmap.width <= 0) {
  18.             return null
  19.         }
  20.         //缩放比例
  21.         val scale = minOf(outWidth.toFloat() / bitmap.width, outHeight.toFloat() / bitmap.height)
  22.         //创建矩阵
  23.         val matrix = Matrix().apply {
  24.             setScale(scale, scale)
  25.         }
  26.         //创建shader
  27.         val shader = BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP).apply {
  28.             setLocalMatrix(matrix)
  29.         }
  30.         //通过shader着色器来绘制图像
  31.         val paint = Paint(Paint.ANTI_ALIAS_FLAG).apply {
  32.             this.shader = shader
  33.         }
  34.         return Bitmap.createBitmap(outWidth, outHeight, Bitmap.Config.ARGB_8888).also { output ->
  35.             Canvas(output).apply {
  36.                 val rect = RectF(
  37.                     border.toFloat(),
  38.                     border.toFloat(),
  39.                     (outWidth - border).toFloat(),
  40.                     (outHeight - border).toFloat()
  41.                 )
  42.                 if (shapeType == RoundImgView.SHAPE_ROUND_RECT) {
  43.                     //绘制圆角矩形
  44.                     drawRoundRect(rect, radius.toFloat(), radius.toFloat(), paint)
  45.                 } else {
  46.                     //绘制圆形
  47.                     drawCircle(outWidth / 2f, outHeight / 2f, outWidth / 2f, paint)
  48.                 }
  49.                 if (border > 0) {
  50.                     //如果有描边,绘制描边
  51.                     val strokePaint = Paint(Paint.ANTI_ALIAS_FLAG).apply {
  52.                         color = Color.RED
  53.                         style = Paint.Style.STROKE
  54.                         strokeWidth = border.toFloat()
  55.                     }
  56.                     if (shapeType == RoundImgView.SHAPE_ROUND_RECT) {
  57.                         //绘制圆角矩形
  58.                         drawRoundRect(rect, radius.toFloat(), radius.toFloat(), strokePaint)
  59.                     } else {
  60.                         //绘制圆形
  61.                         drawCircle(outWidth / 2f, outHeight / 2f, (outWidth - border) / 2f, strokePaint)
  62.                     }
  63.                 }
  64.             }
  65.         }
  66.     }
复制代码
Activity中使用:
  1. val bitmap = BitmapFactory.decodeResource(resources, R.drawable.icon_cat_w)
  2. val targetBitmap = getBitmapByShader(bitmap, 200.dp2px(), 200.dp2px(), 20.dp2px(), 5.dp2px(), RoundImgView.SHAPE_ROUND_RECT)
  3. mIvTarget.setImageBitmap(targetBitmap)
复制代码
或者在自定义Drawable中使用BitmapShader:
  1.     class RoundDrawable(val bitmap: Bitmap, val targetWH: Float) : Drawable() {
  2.         private val paint = Paint().apply {
  3.             //缩放比例
  4.             val scale = minOf(targetWH / bitmap.width, targetWH / bitmap.height)
  5.             //创建矩阵
  6.             val matrix = Matrix().apply {
  7.                 setScale(scale, scale)
  8.             }
  9.             isAntiAlias = true
  10.             shader = BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP).apply {
  11.                 setLocalMatrix(matrix)
  12.             }
  13.         }
  14.         override fun draw(canvas: Canvas) {
  15.             val rectF = RectF(0f, 0f, targetWH, targetWH)
  16.             canvas.drawRoundRect(rectF, 20f, 20f, paint)
  17.         }
  18.         override fun setAlpha(alpha: Int) {
  19.             paint.alpha = alpha
  20.         }
  21.         override fun setColorFilter(colorFilter: ColorFilter?) {
  22.             paint.colorFilter = colorFilter
  23.         }
  24.         override fun getOpacity(): Int = PixelFormat.TRANSLUCENT
  25.     }
复制代码
Activity中:
  1. val roundDrawable = RoundDrawable(bitmap, 200.dp2px().toFloat())
  2. mIvTarget.setImageDrawable(roundDrawable)
复制代码
方式六:RoundedBitmapDrawable

RoundedBitmapDrawable 是 Android 提供的一个工具类,用于处置惩罚圆形或圆角矩形的图片表现。
  1. val bitmap = BitmapFactory.decodeResource(resources, R.drawable.icon_cat_w)
  2. val roundBitmapDrawable = RoundedBitmapDrawableFactory.create(resources, bitmap).apply {
  3.       paint.isAntiAlias = true
  4.       cornerRadius = 20.dp2px().toFloat()
  5. }
  6. mIvTarget.setImageDrawable(roundBitmapDrawable)
复制代码
结论

以上介绍了几种常见的在 Android 中实现图片圆角矩形和圆形效果的方法,每种方式都有其使用场景和特点。推荐优先使用系统已经实现好的,好比ViewOutlineProvider 或者 优秀的Glide 图片加载库,如果有额外样式,可以按需实现。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

卖不甜枣

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表