查看: 79|回复: 0

鸿蒙原生绘图API:从基础到高阶的绘制之旅(进阶版)

[复制链接]

1

主题

1

回帖

14

积分

新手上路

积分
14
发表于 2025-4-14 07:42:12 | 显示全部楼层 |阅读模式
家人们,还记得上次一起探索的鸿蒙绘图 API 基础用法吗?上手是不是特别容易!今天,咱们就接着深入,开启进阶版的学习,解锁更多复杂又炫酷的绘图技能,让你的鸿蒙应用界面直接“出圈”!
我将结合实际开发场景,丰富绘制路径、圆角矩形、绘制图片、画笔与画刷、裁剪区域设置这几个方面的案例,让示例更具实操性和参考价值。
一、复杂图形绘制

(一)路径绘制

当绘制不规则图形,比如独特的图标、创意的装饰元素时,Path类就成了得力助手,它能帮我们创建并操控各种复杂路径。这些路径可以由直线、弧线、贝塞尔曲线等组合而成。
创建路径对象很简单:
  1. let path = new drawing.Path();
复制代码
复制代码
接着设置路径起始点:
  1. path.moveTo(10, 10);
复制代码
复制代码
然后添加线段:
  1. path.lineTo(100, 100);
复制代码
复制代码
绘制弧线也不难:
  1. path.arcTo(100, 100, 200, 200, 0, 90);
复制代码
复制代码
除了弧线,还能使用二阶贝塞尔曲线(quadTo)和三阶贝塞尔曲线(cubicTo) 。
案例:绘制一个五角星五角星的绘制需要结合直线和贝塞尔曲线,利用 Path类的相关方法,通过精确计算每个顶点的坐标来实现。
  1. let path = new drawing.Path();
  2. // 五角星的半径
  3. const radius = 100;
  4. // 计算五角星顶点坐标的辅助函数
  5. function calculateVertex(angle) {
  6.     return {
  7.         x: Math.sin(angle) * radius + 200,
  8.         y: -Math.cos(angle) * radius + 200
  9.     };
  10. }
  11. // 五角星的内角角度
  12. const angles = [
  13.     0,
  14.     Math.PI * 2 / 5,
  15.     Math.PI * 4 / 5,
  16.     Math.PI * 6 / 5,
  17.     Math.PI * 8 / 5
  18. ];
  19. // 移动到第一个顶点
  20. path.moveTo(calculateVertex(angles[0]).x, calculateVertex(angles[0]).y);
  21. for (let i = 1; i < 5; i++) {
  22.     const vertex = calculateVertex(angles[i]);
  23.     path.lineTo(vertex.x, vertex.y);
  24. }
  25. // 闭合路径
  26. path.close();
  27. // 附加画笔并绘制路径
  28. let pen = new drawing.Pen();
  29. pen.setStrokeWidth(4);
  30. pen.setColor({ alpha: 255, red: 255, green: 0, blue: 0 });
  31. canvas.attachPen(pen);
  32. canvas.drawPath(path);
  33. canvas.detachPen();
复制代码
复制代码
在这个案例中,通过数学计算得到五角星每个顶点的坐标,利用 moveTo和 lineTo方法依次连接各个顶点,最后使用 close方法闭合路径,实现五角星的绘制。
(二)圆角矩形绘制

在追求界面美观与柔和的设计中,圆角矩形很常用。使用 drawRoundRect方法可以实现:
  1. drawRoundRect(roundRect: RoundRect): void
复制代码
复制代码
其中,RoundRect对象包含了矩形的位置、大小以及圆角半径信息。
案例:绘制一个卡片式布局在很多应用中,卡片式布局很常见,使用圆角矩形可以轻松实现。
  1. import { common2D, drawing } from '@kit.ArkGraphics2D';
  2. // 卡片的位置和大小
  3. let rect = { left: 50, top: 50, right: 250, bottom: 150 };
  4. // 圆角半径
  5. let roundRect = new drawing.RoundRect(rect, 15, 15);
  6. // 附加画刷设置背景颜色
  7. let brush = new drawing.Brush();
  8. brush.setColor({ alpha: 255, red: 240, green: 240, blue: 240 });
  9. canvas.attachBrush(brush);
  10. canvas.drawRoundRect(roundRect);
  11. canvas.detachBrush();
  12. // 附加画笔绘制边框
  13. let pen = new drawing.Pen();
  14. pen.setStrokeWidth(2);
  15. pen.setColor({ alpha: 255, red: 128, green: 128, blue: 128 });
  16. canvas.attachPen(pen);
  17. canvas.drawRoundRect(roundRect);
  18. canvas.detachPen();
复制代码
复制代码
在这个案例中,先定义了卡片的位置、大小和圆角半径,创建 RoundRect对象。然后分别使用画刷设置背景颜色,使用画笔绘制边框,实现了一个简单的卡片式布局。
二、图像绘制

(一)绘制图片

在应用开发中,在画布上绘制图片很常见。drawImage系列方法可以满足需求:
  1. drawImage(pixelmap: image.PixelMap, left: number, top: number, samplingOptions?: SamplingOptions): void
复制代码
复制代码
pixelmap是要绘制的图片对象,left和 top指定图片在画布上的位置,samplingOptions用于设置采样选项(API version 12+支持)。
从 API version 12 开始,新增了更灵活的方法。drawImageRect可以将图片绘制到指定区域:
  1. drawImageRect(pixelmap: image.PixelMap, dstRect: common2D.Rect, samplingOptions?: SamplingOptions): void
复制代码
复制代码
drawImageRectWithSrc则能将图片的指定区域绘制到画布的指定区域:
  1. drawImageRectWithSrc(pixelmap: image.PixelMap, srcRect: common2D.Rect, dstRect: common2D.Rect, samplingOptions?: SamplingOptions, constraint?: SrcRectConstraint): void
复制代码
复制代码
案例:制作一个图片画廊假设我们要制作一个简单的图片画廊,展示三张图片,并且对其中一张图片进行局部放大展示。
  1. import { image, drawing, common2D } from '@kit.ArkUI';
  2. // 加载三张图片
  3. Promise.all([
  4.     image.createPixelMap(/* 图片1数据 */),
  5.     image.createPixelMap(/* 图片2数据 */),
  6.     image.createPixelMap(/* 图片3数据 */)
  7. ]).then(([pixelMap1, pixelMap2, pixelMap3]) => {
  8.     // 绘制第一张图片
  9.     canvas.drawImage(pixelMap1, 20, 20);
  10.     // 绘制第二张图片
  11.     canvas.drawImage(pixelMap2, 150, 20);
  12.     // 绘制第三张图片,并对其局部放大展示
  13.     const srcRect = { left: 50, top: 50, right: 150, bottom: 150 };
  14.     const dstRect = { left: 280, top: 20, right: 400, bottom: 140 };
  15.     canvas.drawImageRectWithSrc(pixelMap3, srcRect, dstRect);
  16. });
复制代码
复制代码
在这个案例中,使用 Promise.all同时加载三张图片,然后分别使用 drawImage和 drawImageRectWithSrc方法将图片绘制到画布上,实现了一个简单的图片画廊效果,并且对第三张图片进行了局部放大展示。
三、绘制状态与属性设置

(一)画笔与画刷

在绘制图形时,Pen类和 Brush类能帮我们设置线条和填充属性,让图形更具个性。
Pen类可以设置线条颜色、宽度、是否抗锯齿等。
  1. let pen = new drawing.Pen();
  2. pen.setStrokeWidth(5);
  3. pen.setColor({ alpha: 255, red: 255, green: 0, blue: 0 });
  4. pen.setAntiAlias(true);
  5. canvas.attachPen(pen);
  6. // 绘制图形
  7. canvas.drawRect(10, 10, 110, 110);
  8. canvas.detachPen();
复制代码
复制代码
Brush类用于设置填充属性,比如填充颜色。
  1. let brush = new drawing.Brush();
  2. brush.setColor({ alpha: 255, red: 0, green: 255, blue: 0 });
  3. canvas.attachBrush(brush);
  4. // 绘制图形
  5. canvas.drawCircle(100, 100, 50);
  6. canvas.detachBrush();
复制代码
复制代码
案例:绘制一个立体按钮通过画笔和画刷的配合,可以绘制出具有立体感的按钮。
  1. // 绘制按钮背景
  2. let brush = new drawing.Brush();
  3. brush.setColor({ alpha: 255, red: 100, green: 100, blue: 255 });
  4. canvas.attachBrush(brush);
  5. let rect = { left: 100, top: 100, right: 200, bottom: 150 };
  6. canvas.drawRoundRect(new drawing.RoundRect(rect, 10, 10));
  7. canvas.detachBrush();
  8. // 绘制按钮边框,模拟立体效果
  9. let pen = new drawing.Pen();
  10. pen.setStrokeWidth(2);
  11. // 上边框和左边框颜色较浅
  12. pen.setColor({ alpha: 255, red: 200, green: 200, blue: 255 });
  13. canvas.attachPen(pen);
  14. canvas.drawLine(100, 100, 200, 100);
  15. canvas.drawLine(100, 100, 100, 150);
  16. canvas.detachPen();
  17. // 下边框和右边框颜色较深
  18. pen.setColor({ alpha: 255, red: 50, green: 50, blue: 150 });
  19. canvas.attachPen(pen);
  20. canvas.drawLine(100, 150, 200, 150);
  21. canvas.drawLine(200, 100, 200, 150);
  22. canvas.detachPen();
复制代码
复制代码
在这个案例中,先使用画刷绘制按钮的背景,然后通过画笔分别设置不同颜色绘制边框,模拟出立体效果。
(二)裁剪区域设置

有时候,我们只想在画布的特定区域进行绘制,clipRect方法可以实现:
  1. canvas.clipRect({ left: 50, top: 50, right: 150, bottom: 150 }, drawing.ClipOp.DIFFERENCE, true);
复制代码
复制代码
第一个参数指定裁剪区域的矩形,第二个参数是裁剪操作类型,ClipOp.DIFFERENCE表示取差集,即只保留画布上不在裁剪区域内的部分;第三个参数表示是否反转裁剪区域。
案例:制作一个图片蒙版效果通过设置裁剪区域,可以制作图片蒙版效果。
  1. import { image, drawing, common2D } from '@kit.ArkUI';
  2. image.createPixelMap(/* 图片数据 */).then((pixelMap) => {
  3.     // 设置裁剪区域为圆形
  4.     const circleRect = { left: 100, top: 100, right: 200, bottom: 200 };
  5.     const circlePath = new drawing.Path();
  6.     circlePath.addCircle(150, 150, 50);
  7.     canvas.clipPath(circlePath, drawing.ClipOp.INTERSECT);
  8.     // 绘制图片
  9.     canvas.drawImage(pixelMap, 0, 0);
  10.     // 清除裁剪区域
  11.     canvas.restore();
  12. });
复制代码
复制代码
在这个案例中,先创建一个圆形路径,使用 clipPath方法设置裁剪区域为圆形,然后绘制图片,这样图片就只会显示在圆形区域内,实现了图片蒙版效果。最后使用 restore方法清除裁剪区域,以便后续正常绘制。
家人们,到这里,鸿蒙绘图 API 的进阶内容就学完啦!是不是感觉自己离大神又近了一步?赶紧动手实践,把这些知识运用到实际开发中。要是遇到问题,别担心,去官方文档里找找答案,或者和技术交流群里的小伙伴们一起讨论。下一次,咱们将挑战高阶版,一起探索更高级的绘图技巧,期待与大家共同进步!

在开发鸿蒙原生应用的时候,需要用到图表组件的也可以尝试使用我们封装的。图表的官网地址:meichuangit.net.cn/
如果你对案例还有其他修改意见,比如增减特定场景案例、调整代码注释等,欢迎随时提出。

本帖子中包含更多资源

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

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

本版积分规则

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