HarmonyOS Sample 之 PixelMap 图像功能开发

打印 上一主题 下一主题

主题 822|帖子 822|积分 2466

1.介绍

HarmonyOS图像模块支持图像业务的开发,常见功能如图像解码、图像编码、基本的位图操作、图像编辑等。当然,也支持通过接口组合来实现更复杂的图像处理逻辑。
那么什么叫图像解码,什么叫图像编码,什么叫位图?
PixelMap是图像解码后无压缩的位图格式
图像解码就是差别的存档格式图片(如JPEG、PNG等)解码为无压缩的位图格式
图像编码就是将无压缩的位图格式,编码成差别格式的存档格式图片(JPEG、PNG等)
不管是编码照旧解码的目标是方便在应用或者系统中进行相应的处理。
2.搭建环境

安装DevEco Studio,​ 详情请参考DevEco Studio下载。
设置DevEco Studio开发环境,DevEco Studio开发环境必要依赖于网络环境,必要连接上网络才能确保工具的正常使用,可以根据如下两种环境来配置开发环境:
如果可以直接访问Internet,​ 只需进行下载HarmonyOS SDK操作。
如果网络不能直接访问Internet,必要通过代理服务器才可以访问,​ 请参考配置开发环境。
下载源码后,使用DevEco Studio 打开项目,模仿器运行即可。
真机运行必要将config.json中的bundleName修改为自己的,​ 如果没有请到AGC上进行配置,拜见 使用模仿器进行调试 。
3.代码结构



4.实例讲解

4.1.界面结构

结构只有几个操作按钮,见反面的效果图,不是本文的重点。
4.2.后台代码

4.2.1 图像解码功能

图像解码就是将所支持格式的存档图片解码成同一的PixelMap图像,用于后续图像表现或其他处理,比如旋转、缩放、裁剪等。当前支持格式包罗JPEG、PNG、GIF、HEIF、WebP、BMP。
重要用到 ImageSource等
ImageSource.SourceOptions 指定命据源的格式信息,非必填
ImageSource.DecodingOptions 用来支持在解码过程中的图像处理操作,比方缩放、裁剪、旋转,非必填。
接下来看个最简朴的图像解码的例子:
  1. /**
  2. * 图形解码
  3. * png -->ImageSource -->PixelMap
  4. *
  5. * @param component
  6. */
  7. private void commonDecode(Component component) {
  8.     cleanComponents();
  9.     //1.获取应用程序在设备内部存储器上存放文件的目录。
  10.     String pathName = new File(getFilesDir(), "test.png").getPath();
  11.     /*
  12.     InputStream inputStream=null;
  13.     try {
  14.         //2.读取 media目录的图片
  15.         inputStream = getContext().getResourceManager().getResource(ResourceTable.Media_icon);
  16.         ImageSource imageSource = ImageSource.create(inputStream, null);
  17.         inputStream.close();
  18.     } catch (IOException e) {
  19.         e.printStackTrace();
  20.     } catch (NotExistException e) {
  21.         e.printStackTrace();
  22.     }*/
  23.     //--------------------最简单的方式-----------------------
  24.     ImageSource imageSource = ImageSource.create(pathName, null);
  25.     //解码为位图格式
  26.     PixelMap pixelMap = imageSource.createPixelmap(null);
  27.     //设置图片显示
  28.     showFirstImage.setPixelMap(pixelMap);
  29.     //--------------------使用SourceOptions、DecodingOptions 选项-------------
  30.     //指定数据源的格式信息,提高解码效率
  31.     ImageSource.SourceOptions sourceOptions = new ImageSource.SourceOptions();
  32.     sourceOptions.formatHint = "image/png";
  33.     //image 源
  34.     imageSource = ImageSource.create(pathName, sourceOptions);
  35.     //提供图像解码选项
  36.     ImageSource.DecodingOptions decodingOptions = new ImageSource.DecodingOptions();
  37.     //裁剪,如果设置为全0,则不进行裁剪。358/227-448/279
  38.     decodingOptions.desiredRegion = new Rect(358, 227, 90, 52);
  39.     //缩放,如果选择的尺寸与原始图像尺寸不同,则将图像放大或缩小后输出。如果设置为全0,则不进行缩放
  40.     decodingOptions.desiredSize = new Size(90 * 3, 52 * 3);
  41.     //旋转角度,取值范围为0~360。以原图为旋转中心,顺时针旋转图像。
  42.     decodingOptions.rotateDegrees = 180;
  43.     //解码为位图格式
  44.     pixelMap = imageSource.createPixelmap(decodingOptions);
  45.     //设置图片先死
  46.     showSecondImage.setPixelMap(pixelMap);
  47.     //记得释放资源
  48.     imageSource.release();
  49.     pixelMap.release();
  50. }
复制代码
效果:

  1. 10-26 19:02:01.594 13148-13148/? D 00000/=>MainAbilitySlice:  pngCachePath:/data/user/0/com.buty.samples/files/test.png
  2. 10-26 19:02:01.594 13148-13148/? D 00000/=>MainAbilitySlice:  jpgCachePath:/data/user/0/com.buty.samples/files/test.jpg
复制代码
还有一个渐进式解码
官方文档的形貌:在未获取到全部图像时,支持先更新部分数据来实行解码,调用updateData更新数据,将参数isFinal设置为false;当获取到全部数据后,最后一次更新数据时设置isFinal为true,表示数据更新完毕。
又去网上搜刮了一下关于渐进式解码的内容,是这么说的
“渐进式解码提供在整个图像完成下载之前以增量方式解码和出现图像部分的本领。 此功能极大地改进了从 Internet 检察图像时用户的体验,因为用户无需等待整个图像下载,就可以开始解码。 在下载整个映像之前,用户可以检察包含可用数据的映像预览。 此功能对于用于从 Internet 或带宽有限的数据源检察图像的任何应用步伐都至关重要。”
“渐进式解码是一种从不完整的图像文件中以增量方式解码图像部分的本领。 传统解码必要完整的图像文件才能开始解码。 渐进式解码在图像的渐进式级别完成下载后开始。 解码器对图像的当前渐进式级别执行解码传递。 然后,它会在下载每个渐进式级别时对图像执行多个解码传递。 每次解码传递都表现更多图像,直到完全下载并解码图像。 解码完整图像所需的传递数取决于图像文件格式和用于创建图像的编码过程。
使用渐进式解码的要求:
1.图像文件必须支持渐进式解码。 大多数图像格式不支持渐进式解码,尽管常用图像格式为 JPEG、PNG 和 GIF。
2.图像文件必须编码为渐进式图像。 未使用渐进式图像编码创建的图像文件无法实现渐进式解码,即使文件格式会支持渐进式解码。
3.支持渐进式解码的编解码器必须可用。 如果编解码器不支持渐进式解码,则编码为渐进式图像的图像将被解码为传统图像。”

一起看个渐进式解码的例子,
点击 “渐进式解码” 按钮,执行一次imageSource.updateData
  1. /**
  2. * 渐进式解码
  3. *
  4. * @param component
  5. */
  6. private void regionDecode(Component component) {
  7.     if (buttonClickNum < 10) {
  8.         // 获取到一定的数据时尝试解码,
  9.         imageSource.updateData(fileByte, fileByte.length * buttonClickNum / 10, fileByte.length / 10,
  10.                 buttonClickNum==9?true:false);
  11.         pixelMap = imageSource.createPixelmap(null);
  12.         showResultText.setText( (buttonClickNum + 1) + "/10");
  13.         showSecondImage.setPixelMap(pixelMap);
  14.         buttonClickNum++;
  15.     }else{
  16.         pixelMap.release();
  17.         imageSource.release();
  18.         readFileForInitFileByte();
  19.         buttonClickNum=0;
  20.         cleanComponents();
  21.     }
  22. }
  23. /**
  24. * 读取文件初始化 fileByte变量
  25. * 初始化 imageSource
  26. */
  27. private void readFileForInitFileByte() {
  28.     File file = new File(jpgCachePath);
  29.     try {
  30.         FileInputStream fileInputStream = new FileInputStream(file);
  31.         ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
  32.         byte[] data = new byte[1024];
  33.         int len = -1;
  34.         while ((len = fileInputStream.read(data)) != -1) {
  35.             byteArrayOutputStream.write(data, 0, len);
  36.         }
  37.         fileByte = byteArrayOutputStream.toByteArray();
  38.         ImageSource.SourceOptions srcOpts = new ImageSource.SourceOptions();
  39.         srcOpts.formatHint = "image/jpeg";
  40.         //增量源选项 IncrementalSourceOptions
  41.         ImageSource.IncrementalSourceOptions incrementalSourceOptions = new ImageSource.IncrementalSourceOptions();
  42.         incrementalSourceOptions.opts = srcOpts;
  43.         //表示只输入增量数据来更新源。
  44.         incrementalSourceOptions.mode = ImageSource.UpdateMode.INCREMENTAL_DATA;
  45.         //创建增量数据源
  46.         imageSource = ImageSource.createIncrementalSource(incrementalSourceOptions);
  47.     } catch (IOException e) {
  48.         e.printStackTrace();
  49.     }
  50. }
复制代码
效果:


4.2.2 图像编码功能

图像编码重要用到ImagePacker
ImagePacker.PackingOptions 设置编码选项,目前没有太多可选。
  1. /**
  2. * 使用 ImagePacker  来打包图片
  3. * 表示将压缩图像打包成文件或其他对象的图像打包器。
  4. * 可以调用create创建图片打包器,调用initializePacking设置打包选项,调用addImage添加要打包的图片数据,调用finalizePacking完成打包输出目标对象。
  5. *
  6. * @param component
  7. */
  8. private void encode(Component component) {
  9.     cleanComponents();
  10.     //创建ImagePacker对象
  11.     ImagePacker imagePacker = ImagePacker.create();
  12.     //设置编码选项
  13.     ImagePacker.PackingOptions packingOptions = new ImagePacker.PackingOptions();
  14.     //图像质量,范围从0-100,100为最佳质量。
  15.     packingOptions.quality = 90;
  16.     //文件输出流, test_encode.jpg
  17.     try (FileOutputStream outputStream = new FileOutputStream(encodeOutPath)) {
  18.         //初始化将结果输出到 OutputStream 对象的打包任务。
  19.         imagePacker.initializePacking(outputStream, packingOptions);
  20.         //图像数据源,test.png
  21.         ImageSource imageSource = ImageSource.create(pngCachePath, null);
  22.         //转为位图格式
  23.         PixelMap pixelMap = imageSource.createPixelmap(null);
  24.         //将pixelMap添加到编码器,进行编码
  25.         boolean result = imagePacker.addImage(pixelMap);
  26.         showResultText.setText(
  27.                 "Encode result : " + result + System.lineSeparator() + "OutputFilePath:" + encodeOutPath);
  28.         //释放图像数据源
  29.         imageSource.release();
  30.         pixelMap.release();
  31.     } catch (IOException e) {
  32.         HiLog.info(LABEL_LOG, "%{public}s", "encode IOException ");
  33.     }
  34.     //释放imagePacker
  35.     imagePacker.release();
  36. }
复制代码
4.2.3 编辑位图功能

a.通过imageSource创建缩略图,createThumbnailPixelmap
  1. //解码 ImageSource 实例中包含的缩略图数据以生成缩略图并创建缩略图像素图。
  2. //allowFromImage - 如果 ImageSource 不包含缩略图数据,则指定是否允许基于原始图像创建。
  3. PixelMap thumbnailPixelMap = imageSource.createThumbnailPixelmap(decodingOpts, true);
复制代码
b.读写位图像素数据,画个像素小人
  1. /**
  2. * 编辑位图
  3. * @param component
  4. */
  5. private void edit(Component component) {
  6.     cleanComponents();
  7.     int colorsWidth = 552;
  8.     int colorsHeight = 310;
  9.     //PixelMap 选项
  10.     PixelMap.InitializationOptions initializationOptions = new PixelMap.InitializationOptions();
  11.     //指示要创建的像素图的预期大小。
  12.     initializationOptions.size = new Size(colorsWidth, colorsHeight);
  13.     initializationOptions.pixelFormat = PixelFormat.ARGB_8888;
  14.     //PixelMap是否允许修改
  15.     initializationOptions.editable = true;
  16.     //表示像素颜色的 int 数组。 数组中的每个元素都是 PixelFormat#ARGB_8888 格式。
  17.     int[] colors = new int[colorsWidth * colorsHeight];
  18.     Arrays.fill(colors, Color.RED.getValue());
  19.     //创建PixelMap
  20.     PixelMap pixelMap = PixelMap.create(colors, initializationOptions);
  21.     //显示图片
  22.     showFirstImage.setPixelMap(pixelMap);
  23.     // 以另外一个PixelMap作为数据源创建
  24.     PixelMap pixelMap2 = PixelMap.create(pixelMap, initializationOptions);
  25.     //读取指定位置的颜色值。
  26.     int color = pixelMap2.readPixel(new Position(1, 1));
  27.     HiLog.info(LABEL_LOG, "%{public}s", "pixelMapEdit readPixel color :" + color);
  28.     //背上一把宝剑
  29.     for(int i=130;i<180 ;i++){
  30.         //在指定位置写入像素
  31.         pixelMap2.writePixel(new Position(i, i), 0xFF112233);
  32.     }
  33.     //在指定区域写入像素
  34.     int[] pixelArray = new int[1024*1024];
  35.     Arrays.fill(pixelArray, Color.BLACK.getValue());
  36.     //头
  37.     Rect region = new Rect(160, 110, 50, 50);
  38.     //stride - 表示数组每行中的像素数。 该值必须大于或等于此 PixelMap 中目标区域的宽度。
  39.     pixelMap2.writePixels(pixelArray, 0, 100, region);
  40.     Arrays.fill(pixelArray, Color.GREEN.getValue());
  41.     //身体
  42.     Rect region2= new Rect(150, 160, 70, 60);
  43.     //stride - 表示数组每行中的像素数。 该值必须大于或等于此 PixelMap 中目标区域的宽度。
  44.     pixelMap2.writePixels(pixelArray, 0, 100, region2);
  45.     Arrays.fill(pixelArray, Color.YELLOW.getValue());
  46.     //胳膊
  47.     Rect region3= new Rect(130, 160, 20, 70);
  48.     //stride - 表示数组每行中的像素数。 该值必须大于或等于此 PixelMap 中目标区域的宽度。
  49.     pixelMap2.writePixels(pixelArray, 0, 100, region3);
  50.     //胳膊
  51.     Rect region4= new Rect(220, 160, 20, 70);
  52.     //stride - 表示数组每行中的像素数。 该值必须大于或等于此 PixelMap 中目标区域的宽度。
  53.     pixelMap2.writePixels(pixelArray, 0, 100, region4);
  54.     Arrays.fill(pixelArray, Color.GRAY.getValue());
  55.     //腿
  56.     Rect region5= new Rect(160, 200, 20, 70);
  57.     //stride - 表示数组每行中的像素数。 该值必须大于或等于此 PixelMap 中目标区域的宽度。
  58.     pixelMap2.writePixels(pixelArray, 0, 100, region5);
  59.     //腿
  60.     Rect region6= new Rect(190, 200, 20, 70);
  61.     //stride - 表示数组每行中的像素数。 该值必须大于或等于此 PixelMap 中目标区域的宽度。
  62.     pixelMap2.writePixels(pixelArray, 0, 100, region6);
  63.     showSecondImage.setPixelMap(pixelMap2);
  64.     //从位图对象中获取信息
  65.     long capacity = pixelMap.getPixelBytesCapacity();
  66.     long bytesNumber = pixelMap.getPixelBytesNumber();
  67.     int rowBytes = pixelMap.getBytesNumberPerRow();
  68.     byte[] ninePatchData = pixelMap.getNinePatchChunk();
  69.     showResultText.setText(
  70.             "This pixelMap detail info :" + System.lineSeparator() + "capacity = " + capacity + System.lineSeparator()
  71.                     + "bytesNumber = " + bytesNumber + System.lineSeparator() + "rowBytes = " + rowBytes
  72.                     + System.lineSeparator() + "ninePatchData = " + Arrays.toString(ninePatchData) + System.lineSeparator());
  73.     pixelMap.release();
  74.     pixelMap2.release();
  75. }
复制代码
效果是这样的

4.2.4 获取图像属性

图像属性解码就是获取图像中包含的属性信息,PropertyKey 存储了常用的属性KEY信息。
看代码:
  1. /**
  2. * 获取图片的缩略图和位置信息
  3. *
  4. * @param component
  5. */
  6. private void attribute(Component component) {
  7.     cleanComponents();
  8.     ImageSource.SourceOptions srcOpts = new ImageSource.SourceOptions();
  9.     srcOpts.formatHint = "image/jpeg";
  10.     HiLog.debug(LABEL_LOG,"jpgCachePath="+jpgCachePath);
  11.     ImageSource imageSource = ImageSource.create(jpgCachePath, srcOpts);
  12.     ImageSource.DecodingOptions decodingOpts = new ImageSource.DecodingOptions();
  13.     //解码 ImageSource 实例中包含的缩略图数据以生成缩略图并创建缩略图像素图。
  14.     //allowFromImage - 如果 ImageSource 不包含缩略图数据,则指定是否允许基于原始图像创建。
  15.     PixelMap thumbnailPixelMap = imageSource.createThumbnailPixelmap(decodingOpts, true);
  16.     //位置信息
  17.     String location = imageSource.getImagePropertyString(PropertyKey.Exif.SUBJECT_LOCATION);
  18.     HiLog.info(LABEL_LOG, "%{public}s", "imageExif location : " + location);
  19.     showResultText.setText("ImageSource attribute : createThumbnailPixelMap");
  20.     showSecondImage.setPixelMap(thumbnailPixelMap);
  21.     //
  22.     imageSource.release();
  23.     thumbnailPixelMap.release();
  24. }
复制代码
效果:

5.思索总结

通过这次实践,可以学到的内容:
1.位图的概念,怎样进行图像的编码、解码、编辑。
2.渐进式解码的概念,怎样天生缩略图,获取其他图像属性等。
最后

如果你想快速提升鸿蒙技术,那么可以直接领取这份包含了:【OpenHarmony多媒体技术、Stage模子、ArkUI多端摆设、分布式应用开发、音频、视频、WebGL、Napi组件、OpenHarmony内核、Harmony南向开发、鸿蒙项目实战】等技术知识点。
鸿蒙Next全套VIP学习资料←点击领取!(安全链接,放心点击
1.鸿蒙焦点技术学习路线


2.大厂面试必问面试题


3.鸿蒙南向开发技术


 4.鸿蒙APP开发必备


 5.HarmonyOS Next 最新全套视频教程


 6.鸿蒙生态应用开发白皮书V2.0PDF


这份全套完整版的学习资料已经全部打包好,朋友们如果必要可以点击→鸿蒙Next全套VIP学习资料免费领取(安全链接,放心点击


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

本帖子中包含更多资源

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

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

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

本版积分规则

南飓风

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表