ShapeUtils.java 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. package cn.com.taiji.tile.util;
  2. import cn.com.taiji.tile.model.POI;
  3. import cn.com.taiji.tile.style.ShapeStyle;
  4. import cn.hutool.core.util.BooleanUtil;
  5. import org.apache.commons.lang3.StringUtils;
  6. import org.locationtech.jts.geom.Coordinate;
  7. import org.locationtech.jts.geom.Geometry;
  8. import javax.imageio.ImageIO;
  9. import java.awt.*;
  10. import java.awt.image.BufferedImage;
  11. import java.io.ByteArrayOutputStream;
  12. import java.io.IOException;
  13. import java.util.List;
  14. import java.util.Objects;
  15. import java.util.concurrent.CopyOnWriteArrayList;
  16. /**
  17. * @author zzyx 2024/1/5
  18. */
  19. public class ShapeUtils {
  20. public static final int WIDTH = 256;
  21. public static final int HEIGHT = 256;
  22. /**
  23. * 获取瓦片字节数组
  24. *
  25. * @param bboxStr 逗号拼接的bbox字符串
  26. * @param dataList 数据集合
  27. * @param shapeStyle 样式
  28. * @return 瓦片字节数组
  29. */
  30. public static byte[] getTileByte(String bboxStr, List<POI> dataList, ShapeStyle shapeStyle) {
  31. BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_ARGB);
  32. renderAndDetectClick(image, bboxStr, dataList, shapeStyle);
  33. ByteArrayOutputStream buffer = new ByteArrayOutputStream();
  34. try {
  35. ImageIO.write(image, "png", buffer);
  36. } catch (IOException e) {
  37. e.printStackTrace();
  38. }
  39. return buffer.toByteArray();
  40. }
  41. private static void drawPolygonShape(double[] bbox, Graphics2D graphics, Geometry geometry, POI poi, ShapeStyle shapeStyle) {
  42. drawPolygonToTile(shapeStyle, geometry, graphics, bbox);
  43. //是否需要设置字名称
  44. if (BooleanUtil.isTrue(shapeStyle.getIsShowName()) && StringUtils.isNotBlank(poi.getName())) {
  45. org.locationtech.jts.geom.Point centroid = geometry.getCentroid();
  46. int[] pxy = TileUtils.toPixelXY(centroid.getX(), centroid.getY(), bbox[0], bbox[1], bbox[2], bbox[3]);
  47. //设置字体样式
  48. Font font = new Font("Arial", Font.TRUETYPE_FONT, shapeStyle.getFontSize());
  49. String s = poi.getName();
  50. graphics.setColor(shapeStyle.getFontColor());
  51. graphics.setFont(font);
  52. graphics.drawString(s, pxy[0], pxy[1]);
  53. //设置面的图标
  54. BufferedImage cpi = shapeStyle.getCenterPointImage();
  55. if (cpi != null) {
  56. graphics.drawImage(TileUtils.scaleWidthHeight(cpi, 13, 13), pxy[0], pxy[1], null);
  57. }
  58. }
  59. }
  60. public static void drawLineShape(double[] bbox, Graphics2D graphics, Geometry geometry, ShapeStyle shapeStyle) {
  61. int[] xpoint = new int[(geometry.getCoordinates()).length];
  62. int[] ypoint = new int[(geometry.getCoordinates()).length];
  63. convertCoordinatesToPixels(geometry.getCoordinates(), xpoint, ypoint, bbox);
  64. graphics.setColor(shapeStyle.getColor());
  65. graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  66. if (Objects.equals(shapeStyle.getLineType(), 2)) {
  67. graphics.setStroke(new BasicStroke(shapeStyle.getWidth(), BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER, 10.0f, shapeStyle.getDashPattern(), 0.0f));
  68. } else if (Objects.equals(shapeStyle.getLineType(), 3)) {
  69. graphics.setStroke(new BasicStroke(shapeStyle.getWidth(), BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER, 10.0f, shapeStyle.getDashPattern(), 0.0f));
  70. } else {
  71. graphics.setStroke(new BasicStroke(shapeStyle.getWidth()));
  72. }
  73. //端点的样式可以是圆形 (CAP_ROUND)、平直 (CAP_BUTT) 或方形 (CAP_SQUARE)
  74. // 连接样式:线段之间的连接可以设置为斜接 (JOIN_MITER)、圆角 (JOIN_ROUND) 或平直 (JOIN_BEVEL)。
  75. // graphics.setStroke(new BasicStroke(10f, BasicStroke.CAP_ROUND, BasicStroke.JOIN_BEVEL));
  76. graphics.drawPolyline(xpoint, ypoint, geometry.getCoordinates().length);
  77. }
  78. public static void drawPointShape(double[] bbox, Graphics2D graphics, Geometry geometry, POI poi, ShapeStyle shapeStyle) {
  79. int[] pxy = TileUtils.toPixelXY(geometry.getCoordinate().x, geometry.getCoordinate().y, bbox[0], bbox[1], bbox[2], bbox[3]);
  80. //设置边框 实现轮廓
  81. graphics.setColor(shapeStyle.getColor());
  82. graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  83. //设置字体
  84. //设置字体样式
  85. if (BooleanUtil.isTrue(shapeStyle.getIsShowName())) {
  86. Font font = new Font("Arial", Font.BOLD, shapeStyle.getFontSize());
  87. String s = poi.getName();
  88. graphics.setColor(shapeStyle.getFontColor());
  89. graphics.setFont(font);
  90. graphics.drawString(s, pxy[0], pxy[1]);
  91. }
  92. BufferedImage pointMarkerImage = shapeStyle.getPointMarkerImage();
  93. if (pointMarkerImage != null) {
  94. graphics.drawImage(TileUtils.scaleWidthHeight(pointMarkerImage, shapeStyle.getWidth(), shapeStyle.getHeight()), pxy[0], pxy[1], null);
  95. } else {
  96. drawPointShape(shapeStyle, graphics, pxy);
  97. }
  98. }
  99. public static void convertCoordinatesToPixels(Coordinate[] coordinates, int[] xPoints, int[] yPoints, double[] bbox) {
  100. // 坐标转换逻辑
  101. for (int i = 0; i < coordinates.length; i++) {
  102. Coordinate c = coordinates[i];
  103. //坐标转换:对于每个坐标 c,代码调用 TileUtils.toPixelXY 方法,将地理坐标(经纬度)转换为像素坐标。这个转换考虑了一个称为 bbox(边界框)的地理区域,这个区域定义了地理坐标到像素坐标转换的上下文。bbox 通常包含边界框的最小经度、最小纬度、最大经度和最大纬度。
  104. int[] pxy = TileUtils.toPixelXY(c.getX(), c.getY(), bbox[0], bbox[1], bbox[2], bbox[3]);
  105. xPoints[i] = pxy[0];
  106. yPoints[i] = pxy[1];
  107. }
  108. }
  109. /**
  110. * 面填充样式
  111. * 在地理信息系统(GIS)和空间数据处理领域中,Geometry 类是一个常用的概念,用于表示空间对象的形状和位置。这个类通常包含各种方法来操作和查询几何形状。getCoordinates() 方法是 Geometry 类中的一个方法,它的作用是获取构成该几何形状的所有坐标点。
  112. * 当你调用 Geometry 对象的 getCoordinates() 方法时,它会返回一个坐标数组,这个数组包含了组成该几何形状的所有顶点坐标。这些坐标通常是 Coordinate 对象,每个 Coordinate 包含 x(经度)、y(纬度)和可选的 z(高度)值。
  113. * 例如,如果你的 Geometry 对象是一个多边形,getCoordinates() 方法将返回构成该多边形边界的所有顶点坐标。如果是一条线,它将返回线的所有顶点坐标。
  114. * 这个方法在进行空间分析和几何处理时非常有用,比如当你需要遍历几何形状的所有顶点、计算形状的边界、或者在不同几何形状之间进行坐标转换时。
  115. * <p>
  116. * Coordinate 是在地理信息系统(GIS)和计算机图形学中常用的一个基本概念,用于表示空间中的点。在多种编程环境和库中,Coordinate 类或结构体通常用于存储和操作这些点的位置信息。其主要作用和特点包括:
  117. * 存储位置信息:Coordinate 通常包含至少两个维度的数据,即 x(经度)和 y(纬度)。在三维空间中,它还可以包含 z(高程或深度)值。
  118. * 空间分析和计算:在 GIS 和计算机图形学应用中,Coordinate 用于执行各种空间分析和计算,如测量距离、定义几何形状(如线和多边形)的边界等。
  119. * 数据转换和处理:Coordinate 在进行坐标系转换、地图投影、空间查询等操作中也非常重要。
  120. * 可视化和绘图:在地图制作和空间数据可视化中,Coordinate 用于确定应在地图上绘制的点的确切位置。
  121. * 在不同的编程库和框架中,Coordinate 类可能还包含了其他辅助方法,如计算两个坐标之间的距离、比较坐标点是否相等等。例如,在 Java 的 JTS Topology Suite(一个常用的空间数据处理库)中,Coordinate 类就提供了这样的功能。
  122. */
  123. public static void drawPolygonToTile(ShapeStyle shapeStyle, Geometry shape, Graphics2D graphics, double[] bbox) {
  124. //普通面
  125. if (shape != null && shape.getCoordinates().length > 0) {
  126. int[] xpoint = new int[(shape.getCoordinates()).length];
  127. int[] ypoint = new int[(shape.getCoordinates()).length];
  128. convertCoordinatesToPixels(shape.getCoordinates(), xpoint, ypoint, bbox);
  129. //设置样式
  130. //2.设置抗锯齿
  131. graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  132. //是否填充
  133. if (BooleanUtil.isTrue(shapeStyle.getIsFill())) {
  134. //fill 是将设置的颜色全部填充
  135. graphics.setColor(shapeStyle.getColor());
  136. //设置背景面
  137. setBackgroundImage(graphics, shapeStyle);
  138. Polygon polygon = new Polygon(xpoint, ypoint, shape.getNumPoints());
  139. //填充
  140. graphics.fill(polygon);
  141. //设置边框颜色
  142. setBorderStyle(graphics, shapeStyle);
  143. //drawPolygon 是透明的只会出来边框
  144. graphics.drawPolygon(polygon);
  145. } else {
  146. setBorderStyle(graphics, shapeStyle);
  147. graphics.drawPolygon(xpoint, ypoint, (shape.getCoordinates()).length);
  148. }
  149. }
  150. }
  151. private static void setBackgroundImage(Graphics2D graphics, ShapeStyle shapeStyle) {
  152. BufferedImage backgroundImage = shapeStyle.getBgImage();
  153. if (backgroundImage != null) {
  154. TexturePaint texturePaint = new TexturePaint(backgroundImage, new Rectangle(0, 0, backgroundImage.getWidth(), backgroundImage.getHeight()));
  155. graphics.setPaint(texturePaint);
  156. }
  157. }
  158. private static void setBorderStyle(Graphics2D graphics, ShapeStyle shapeStyle) {
  159. //不填充只显示边框颜色
  160. graphics.setColor(shapeStyle.getBorderColor());
  161. // 设置虚线:5像素实线,5像素空白
  162. if (BooleanUtil.isTrue(shapeStyle.getIsDashed())) {
  163. Stroke dashed = new BasicStroke(shapeStyle.getBorderSize(), BasicStroke.CAP_BUTT, BasicStroke.JOIN_BEVEL, 0, shapeStyle.getDashPattern(), 0);
  164. graphics.setStroke(dashed);
  165. } else {
  166. graphics.setStroke(new BasicStroke(shapeStyle.getBorderSize()));
  167. }
  168. }
  169. /**
  170. * Converts a hexadecimal color string to RGBA format.
  171. *
  172. * @param hexColor The hexadecimal color string (e.g., "#2860F1").
  173. * @param alpha The alpha value (0-255) for opacity.
  174. * @return An array containing the RGBA values.
  175. */
  176. public static int[] hexToRgba(String hexColor, int alpha) {
  177. // Remove the hash (#) sign if it exists
  178. hexColor = hexColor.replace("#", "");
  179. // Parse the string
  180. int red = Integer.valueOf(hexColor.substring(0, 2), 16);
  181. int green = Integer.valueOf(hexColor.substring(2, 4), 16);
  182. int blue = Integer.valueOf(hexColor.substring(4, 6), 16);
  183. // Return the RGBA values
  184. return new int[]{red, green, blue, alpha};
  185. }
  186. private static void drawPointShape(ShapeStyle shapeStyle, Graphics2D graphics, int[] pxy) {
  187. if (BooleanUtil.isTrue(shapeStyle.getIsDashed())) {
  188. //如果是
  189. Stroke dashed = new BasicStroke(2, BasicStroke.CAP_BUTT, BasicStroke.JOIN_BEVEL, 0, shapeStyle.getPointDashPattern(), 0);
  190. graphics.setStroke(dashed);
  191. }
  192. if (Objects.equals(shapeStyle.getPointType(), 1)) {
  193. //正方形
  194. graphics.drawRect(pxy[0], pxy[1], shapeStyle.getWidth(), shapeStyle.getHeight());
  195. if (BooleanUtil.isTrue(shapeStyle.getIsFill())) {
  196. graphics.fillRect(pxy[0], pxy[1], shapeStyle.getWidth(), shapeStyle.getHeight());
  197. }
  198. } else if (Objects.equals(shapeStyle.getPointType(), 2)) {
  199. //三角形
  200. int[] xPoints = {pxy[0], pxy[0] - shapeStyle.getHeight(), pxy[0] + shapeStyle.getHeight()};
  201. int[] yPoints = {pxy[1] - shapeStyle.getHeight(), pxy[1] + shapeStyle.getHeight(), pxy[1] + shapeStyle.getHeight()};
  202. int nPoints = 3; // 顶点数量
  203. graphics.drawPolygon(xPoints, yPoints, nPoints);
  204. if (BooleanUtil.isTrue(shapeStyle.getIsFill())) {
  205. graphics.fillPolygon(xPoints, yPoints, nPoints);
  206. }
  207. } else if (Objects.equals(shapeStyle.getPointType(), 3)) {
  208. //圆形
  209. graphics.drawOval(pxy[0], pxy[1], shapeStyle.getWidth(), shapeStyle.getHeight());
  210. if (BooleanUtil.isTrue(shapeStyle.getIsFill())) {
  211. graphics.fillOval(pxy[0], pxy[1], shapeStyle.getWidth(), shapeStyle.getHeight());
  212. }
  213. } else {
  214. //画一个简单的点
  215. graphics.drawArc(pxy[0], pxy[1], shapeStyle.getWidth(), shapeStyle.getHeight(), 0, 360);
  216. //是否填充
  217. if (BooleanUtil.isTrue(shapeStyle.getIsFill())) {
  218. graphics.fillArc(pxy[0], pxy[1], shapeStyle.getWidth(), shapeStyle.getHeight(), 0, 360);
  219. }
  220. }
  221. }
  222. public static List<POI> detectClick(String bboxStr, List<POI> dataList, ShapeStyle shapeStyle, int x, int y) {
  223. List<POI> clickSelectedDataList = new CopyOnWriteArrayList<>();
  224. // renderAndDetectClick(image, bboxStr, dataList, shapeStyle, x, y, clickSelectedDataList, true);
  225. String[] split = bboxStr.split(",");
  226. double[] bbox = new double[split.length];
  227. for (int i = 0; i < split.length; i++) {
  228. bbox[i] = Double.parseDouble(split[i]);
  229. }
  230. dataList.parallelStream().forEach(poi -> {
  231. BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_ARGB);
  232. Graphics2D graphics = image.createGraphics();
  233. Geometry shape = GeoUtils.getGeoUtils().createGeometry(poi.getShape());
  234. // //判断数据点线面
  235. if (StringUtils.equals(shape.getGeometryType(), "Polygon")) {
  236. drawPolygonShape(bbox, graphics, shape, poi, shapeStyle);
  237. } else if (StringUtils.equalsAny(shape.getGeometryType(), "LinearRing", "LineString", "MultiLineString")) {
  238. drawLineShape(bbox, graphics, shape, shapeStyle);
  239. } else if (StringUtils.equals(shape.getGeometryType(), "Point")) {
  240. drawPointShape(bbox, graphics, shape, poi, shapeStyle);
  241. }
  242. //如果选中
  243. int rgb = image.getRGB(x, y);
  244. int alpha = (rgb >> 24) & 0xff; // 提取alpha值
  245. if (alpha > 0) {
  246. // 像素被填充
  247. clickSelectedDataList.add(poi);
  248. }
  249. });
  250. return clickSelectedDataList;
  251. }
  252. public static void renderAndDetectClick(BufferedImage image, String bboxStr, List<POI> dataList, ShapeStyle shapeStyle) {
  253. String[] split = bboxStr.split(",");
  254. double[] bbox = new double[split.length];
  255. for (int i = 0; i < split.length; i++) {
  256. bbox[i] = Double.parseDouble(split[i]);
  257. }
  258. Graphics2D graphics = image.createGraphics();
  259. dataList.parallelStream().forEach(poi -> {
  260. Geometry shape = GeoUtils.getGeoUtils().createGeometry(poi.getShape());
  261. // //判断数据点线面
  262. if (StringUtils.equals(shape.getGeometryType(), "Polygon")) {
  263. drawPolygonShape(bbox, graphics, shape, poi, shapeStyle);
  264. } else if (StringUtils.equalsAny(shape.getGeometryType(), "LinearRing", "LineString", "MultiLineString")) {
  265. drawLineShape(bbox, graphics, shape, shapeStyle);
  266. } else if (StringUtils.equals(shape.getGeometryType(), "Point")) {
  267. drawPointShape(bbox, graphics, shape, poi, shapeStyle);
  268. }
  269. });
  270. }
  271. }