经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » C# » 查看文章
Graphics2D绘图方法总结
来源:cnblogs  作者:知了一笑  时间:2024/8/26 9:17:36  对本文有异议

一、简介

在开发中可能会遇到这样一类场景,业务复杂度不算太高,技术难度不算太深,但是做起来就很容易把人整破防,伤害很高侮辱性很强的:绘图。

绘图最怕有人挑刺:这里变形,那里不对,全图失真。

最近在处理这样一个场景,使用Java的Graphics2D类,绘制业务需要的图形模板,然后在具体流程中填充数据,并且将图形存储起来,逻辑并不复杂,由于涉及ToC和ToB两端交互,必须用点雕花的态度。

二、字体安装

在绘制具体图形时,需要先处理好本地字体,使用设计师提供的字体,才可能在图片上复制出想要的效果;安装完相关的字体包,使用Java读取验证后再直接使用。

  1. public class Typeface {
  2. public static void main(String[] args) {
  3. List<String> fontNames = new java.util.ArrayList<>();
  4. Font[] fonts = GraphicsEnvironment.getLocalGraphicsEnvironment().getAllFonts();
  5. for (Font font : fonts){
  6. fontNames.add(font.getName());
  7. }
  8. fontNames.forEach(System.out::println);
  9. }
  10. }

三、绘制图形

在制图中,会涉及一些简单的图形样式,比如线条、矩形、圆弧线等,这些都可以使用Graphics2D的语法直接生成,下面的程序创建一张500x500的图片,然后在其中绘制一些简单的图形样式,最后保存到本地。

  1. public class DrawDraft {
  2. public static void main(String[] args) throws Exception {
  3. // 1、创建图片绘图
  4. BufferedImage image = new BufferedImage(500, 500, BufferedImage.TYPE_4BYTE_ABGR);
  5. Graphics2D graphics = image.createGraphics();
  6. graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  7. graphics.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
  8. // 2、填充背景色
  9. graphics.setColor(Color.white);
  10. graphics.fillRect(0, 0, 500, 500);
  11. // 3、绘制线条
  12. graphics.setStroke(new BasicStroke(3));
  13. graphics.setColor(Color.red);
  14. graphics.drawLine(50, 50, 280, 50);
  15. graphics.setColor(Color.blue);
  16. graphics.drawLine(50, 50, 165, 200);
  17. graphics.setColor(Color.green);
  18. graphics.drawLine(280, 50, 165, 200);
  19. // 4、绘制图形
  20. graphics.setStroke(new BasicStroke(2));
  21. graphics.setColor(Color.pink);
  22. graphics.drawRect(200, 200, 80, 50);// 矩形
  23. graphics.setColor(Color.green);
  24. graphics.drawArc(280, 280, 100, 100, 0, 180);//圆弧线
  25. graphics.drawArc(300, 300, 100, 50, 0, -270);//圆弧线弧度
  26. graphics.setColor(Color.orange);
  27. graphics.drawArc(350, 350, 100, 100, 0, 180);//圆弧线
  28. graphics.fillArc(350, 350, 100, 100, 0, -270);//填充四分之三的圆形
  29. // 5、写到图片
  30. ImageIO.write(image, "png", new File("src/main/draw-draft.png"));
  31. image.flush();
  32. graphics.dispose();
  33. }
  34. }

四、绘制文本

在常规的业务场景中,一般是先绘制模版图形,然后在模板的图形上填充数据,也可以直接使用设计师提供的模板文件,这样可以避免数据填充时出现排版问题,如果有大量的动态数据内容,可以使用模板引擎,这在以前的内容中有写个类似的案例。

下面这个案例,使用上面的模板,在此模版上进行文本添加,绘制文本主要就是一些动态对齐和排版等问题,最后制图生效时添加签章即可。

  1. import java.awt.Color;
  2. import java.awt.Font;
  3. import java.awt.Graphics2D;
  4. import java.awt.RenderingHints;
  5. import java.awt.image.BufferedImage;
  6. import java.io.File;
  7. import javax.imageio.ImageIO;
  8. public class DrawImage {
  9. public static void main(String[] args) throws Exception {
  10. // 1、基础样式
  11. Font yhFont = new Font("Microsoft Yahei UI", Font.PLAIN, 15);
  12. Font yhBoldFont = new Font("Microsoft Yahei UI Bold", Font.BOLD, 25);
  13. Font tailFont = new Font("Microsoft Yahei UI Bold", Font.PLAIN, 12);
  14. // 2、基于底图绘制
  15. BufferedImage backImg = ImageIO.read(new File("src/main/draw-draft.png"));
  16. int canvasWidth = backImg.getWidth();
  17. int canvasHeight = backImg.getHeight();
  18. // 3、创建画笔
  19. Graphics2D graphics = backImg.createGraphics();
  20. graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  21. graphics.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
  22. // 4、绘制居中标题
  23. graphics.setFont(yhBoldFont);
  24. graphics.setColor(Color.BLACK);
  25. String title = "2D绘图";
  26. int titleWidth = graphics.getFontMetrics().stringWidth(title);
  27. int titleX = canvasWidth/2-titleWidth/2;
  28. int titleY = 50;
  29. graphics.drawString(title, titleX, titleY);
  30. // 5、绘制长文本,左对齐和换行
  31. graphics.setFont(yhFont);
  32. graphics.setColor(Color.BLACK);
  33. String blackText = "\u3000组织需要重新审视项目的核心价值主张,以便更好地与利益相关者对齐目标,协同共创。";
  34. String[] textWord = blackText.split("");
  35. // 文本最大宽度和行高
  36. int textMaxWidth = 200;
  37. int textLineHeight = 18;
  38. // 文本字符输出起始坐标
  39. int textWordX = 20;
  40. int textWordY = 350;
  41. // 通过计算控制单行文本长度
  42. StringBuilder textLine = new StringBuilder();
  43. for (String word : textWord){
  44. graphics.drawString(word, textWordX, textWordY);
  45. if (graphics.getFontMetrics().stringWidth(textLine + word) <= textMaxWidth) {
  46. // 不需要换行,记录单行内容,移动X坐标
  47. textLine.append(word);
  48. textWordX = textWordX + graphics.getFontMetrics().stringWidth(word);
  49. } else {
  50. // 需要换行,重置当行文本内容,移动X坐标和Y坐标
  51. textLine.setLength(0);
  52. textWordX = 20 ;
  53. textWordY = textWordY+textLineHeight;
  54. }
  55. }
  56. // 6、绘制短文本,右对齐
  57. graphics.setFont(tailFont);
  58. graphics.setColor(Color.BLUE);
  59. String author = "制图方:白天睡不着";
  60. int authorWidth = canvasWidth-30-graphics.getFontMetrics().stringWidth(author);
  61. graphics.drawString(author, authorWidth, 180);
  62. String drawDate = "时间:2024年8月28日";
  63. int drawDateWidth = canvasWidth-30-graphics.getFontMetrics().stringWidth(drawDate);
  64. graphics.drawString(drawDate, drawDateWidth, 200);
  65. // 7、添加水印图片
  66. BufferedImage watermarkImg = ImageIO.read(new File("src/main/watermark.png"));
  67. graphics.drawImage(watermarkImg, 350, 120,120,120, null);
  68. // 8、写到图片
  69. ImageIO.write(backImg, "png", new File("src/main/draw-img.png"));
  70. backImg.flush();
  71. watermarkImg.flush();
  72. graphics.dispose();
  73. }
  74. }

彩蛋:这里blackText文本是让大模型随机写的,就冲这个输出和味道,大家猜猜出自哪个国产大模型,(放水提示词:国产)。最后关于文件管理就不赘述了,哪个文件服务器方便,就随地存着。

五、源码参考

  1. 文档仓库:
  2. https://gitee.com/cicadasmile/butte-java-note
  3. 源码仓库:
  4. https://gitee.com/cicadasmile/butte-spring-parent

原文链接:https://www.cnblogs.com/cicada-smile/p/18379075

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号