1、背景javascript
在采用Java配合xml布局编写鸿蒙app页面的时候,发现sdk自带的Image组件并不能将图片设置成圆形,反复了翻阅了官方API手册(主要查阅了Compont和Image相关的API),起初发现了一个setCornerRadius方法,因而想着将图片宽度和高度设置为同样,而后调用该方法将radios设置为宽度或者高度的一半,觉得能够实现圆形图片的效果,后来发现不行。因而乎想着能不能经过继承原有的Image本身来动手从新自定义一个支持圆形的图片组件。java
2、思路:ios
一、对比以前本身在其余程序开发中自定义组件的思路,首先寻找父组件Image和Component相关的Api,看看是否具有OnDraw方法。json
二、了解Canvas相关Api操做,特别是涉及到位图的操做。canvas
经过翻阅大量资料,发现了两个关键的api,分别是Component的addDrawTask方法和其内部静态接口DrawTaskapi
3、自定义组件模块app
一、新建一个工程以后,建立一个独立的Java FA模块,而后删除掉里面全部布局以及自动生成的java代码,而后本身建立一个class继承ImageViewiview
二、写一个类继承ImageView,在其中暴露出public的设置圆形图片的api方法以供后面调用;ide
三、在原有的Image组件获取到位图以后,利用该位图数据利用addDrawTask方法配合Canvas进行位图输出形状的从新绘制,这里须要使用Canvas的一个函数
关键api方法drawPixelMapHolderRoundRectShape;
四、注意,为了让Canvas最后输出的图片为圆形,须要将图片在布局中的宽度和高度设置成同样,不然输出的为圆角矩形或者椭圆形。
最后封装后的详细代码以下:
package com.xdw.customview; import ohos.agp.components.AttrSet; import ohos.agp.components.Image; import ohos.agp.render.PixelMapHolder; import ohos.agp.utils.RectFloat; import ohos.app.Context; import ohos.hiviewdfx.HiLog; import ohos.hiviewdfx.HiLogLabel; import ohos.media.image.ImageSource; import ohos.media.image.PixelMap; import ohos.media.image.common.PixelFormat; import ohos.media.image.common.Rect; import ohos.media.image.common.Size; import java.io.InputStream; /** * Created by 夏德旺 on 2021/1/1 11:00 */ public class RoundImage extends Image { private static final HiLogLabel LABEL = new HiLogLabel(HiLog.DEBUG, 0, "RoundImage"); private PixelMapHolder pixelMapHolder;//像素图片持有者 private RectFloat rectDst;//目标区域 private RectFloat rectSrc;//源区域 public RoundImage(Context context) { this(context,null); } public RoundImage(Context context, AttrSet attrSet) { this(context,attrSet,null); } /** * 加载包含该控件的xml布局,会执行该构造函数 * @param context * @param attrSet * @param styleName */ public RoundImage(Context context, AttrSet attrSet, String styleName) { super(context, attrSet, styleName); HiLog.error(LABEL,"RoundImage"); } public void onRoundRectDraw(int radius){ //添加绘制任务 this.addDrawTask((view, canvas) -> { if (pixelMapHolder == null){ return; } synchronized (pixelMapHolder) { //给目标区域赋值,宽度和高度取自xml配置文件中的属性 rectDst = new RectFloat(0,0,getWidth(),getHeight()); //绘制圆角图片 canvas.drawPixelMapHolderRoundRectShape(pixelMapHolder, rectSrc, rectDst, radius, radius); pixelMapHolder = null; } }); } //使用canvas绘制圆形 private void onCircleDraw(){ //添加绘制任务,自定义组件的核心api调用,该接口的参数为Component下的DrawTask接口 this.addDrawTask((view, canvas) -> { if (pixelMapHolder == null){ return; } synchronized (pixelMapHolder) { //给目标区域赋值,宽度和高度取自xml配置文件中的属性 rectDst = new RectFloat(0,0,getWidth(),getHeight()); //使用canvas绘制输出圆角矩形的位图,该方法第4个参数和第5个参数为radios参数, // 绘制图片,必须把图片的宽度和高度先设置成同样,而后把它们设置为图片宽度或者高度一半时则绘制的为圆形 canvas.drawPixelMapHolderRoundRectShape(pixelMapHolder, rectSrc, rectDst, getWidth()/2, getHeight()/2); pixelMapHolder = null; } }); } /** *获取原有Image中的位图资源后从新检验绘制该组件 * @param pixelMap */ private void putPixelMap(PixelMap pixelMap){ if (pixelMap != null) { rectSrc = new RectFloat(0, 0, pixelMap.getImageInfo().size.width, pixelMap.getImageInfo().size.height); pixelMapHolder = new PixelMapHolder(pixelMap); invalidate();//从新检验该组件 }else{ pixelMapHolder = null; setPixelMap(null); } } /** * 经过资源ID获取位图对象 **/ private PixelMap getPixelMap(int resId) { InputStream drawableInputStream = null; try { drawableInputStream = getResourceManager().getResource(resId); ImageSource.SourceOptions sourceOptions = new ImageSource.SourceOptions(); sourceOptions.formatHint = "image/png"; ImageSource imageSource = ImageSource.create(drawableInputStream, null); ImageSource.DecodingOptions decodingOptions = new ImageSource.DecodingOptions(); decodingOptions.desiredSize = new Size(0, 0); decodingOptions.desiredRegion = new Rect(0, 0, 0, 0); decodingOptions.desiredPixelFormat = PixelFormat.ARGB_8888; PixelMap pixelMap = imageSource.createPixelmap(decodingOptions); return pixelMap; } catch (Exception e) { e.printStackTrace(); } finally { try{ if (drawableInputStream != null){ drawableInputStream.close(); } }catch (Exception e) { e.printStackTrace(); } } return null; } /** * 对外调用的api,设置圆形图片方法 * @param resId */ public void setPixelMapAndCircle(int resId){ PixelMap pixelMap = getPixelMap(resId); putPixelMap(pixelMap); onCircleDraw(); } /** * 对外调用的api,设置圆角图片方法 * @param resId * @param radius */ public void setPixelMapAndRoundRect(int resId,int radius){ PixelMap pixelMap = getPixelMap(resId); putPixelMap(pixelMap); onRoundRectDraw(radius); } }
五、修改config.json文件,代码以下
{ "app": { "bundleName": "com.xdw.customview", "vendor": "xdw", "version": { "code": 1, "name": "1.0" }, "apiVersion": { "compatible": 4, "target": 4, "releaseType": "Beta1" } }, "deviceConfig": {}, "module": { "package": "com.xdw.customview", "deviceType": [ "phone", "tv", "tablet", "car", "wearable" ], "reqPermissions": [ { "name": "ohos.permission.INTERNET" } ], "distro": { "deliveryWithInstall": true, "moduleName": "roundimage", "moduleType": "har" } } }
这样该模块就能够导出后续给其余全部工程引用了,后面还能够编译以后发布到gradle上直接经过添加依赖来进行使用(这个是后话),下面咱们先经过本地依赖导入的方式来调用这个自定义组件模块吧。
4、其余工程调用该自定义组件并测试效果
一、再来新建一个工程,而后将以前的模块导入到新建的工程中(DevEco暂时不支持自动导入外部模块的操做,须要手动导入操做,请关注个人另一篇博客)
二、在gradle中引用导入的模块的组件,代码以下:
文章后续内容和相关附件能够点击下面的原文连接前往学习
原文连接:https://harmonyos.51cto.com/posts/2423#bkwz