基于DeepSeek实现PDF嵌入SVG图片无损放大

打印 上一主题 下一主题

主题 1031|帖子 1031|积分 3103

1. PDF中效果图


2. 询问Deepseek进行代码誊写,不停优化后结果

  1. /**
  2. * SVG工具类,用于生成价格趋势的SVG图表
  3. */
  4. public class SvgUtils {
  5.     // SVG画布尺寸
  6.     private static final int WIDTH = 800;
  7.     private static final int HEIGHT = 500;
  8.     private static final int PADDING = 60;
  9.     // 生成SVG折线图
  10.     public static String generatePriceTrendSvg(List<PriceData> data, String title, String subtitle) {
  11.         if (data.isEmpty()) {
  12.             return "<svg xmlns='http://www.w3.org/2000/svg' width='" + WIDTH + "' height='" + HEIGHT + "'></svg>";
  13.         }
  14.         // 计算价格和时间的范围
  15.         double minPrice = data.stream().mapToDouble(PriceData::getPrice).min().orElse(0) * 0.8;
  16.         double maxPrice = data.stream().mapToDouble(PriceData::getPrice).max().orElse(1) * 1.2;
  17.         // 解析最小和最大日期
  18.         String minYearMonth = data.stream().map(PriceData::getYearmonth).min(String::compareTo).orElse("2024-06");
  19.         String maxYearMonth = data.stream().map(PriceData::getYearmonth).max(String::compareTo).orElse("2025-01");
  20.         // 构建SVG内容
  21.         StringBuilder svg = new StringBuilder();
  22.         svg.append(String.format("<svg xmlns='http://www.w3.org/2000/svg' width='%d' height='%d'>", WIDTH, HEIGHT));
  23.         // 画背景
  24.         svg.append(String.format("<rect width='%d' height='%d' fill='#ffffff'/>", WIDTH, HEIGHT));
  25.         // 添加标题和副标题(居中显示)
  26.         svg.append(String.format("<text x='%d' y='%d' font-size='20' font-family='Arial' fill='#333333' font-weight='bold' text-anchor='middle'>%s</text>",
  27.                 WIDTH / 2, 30, title));
  28.         svg.append(String.format("<text x='%d' y='%d' font-size='14' font-family='Arial' fill='#666666' text-anchor='middle'>%s</text>",
  29.                 WIDTH / 2, 50, subtitle));
  30.         // 画虚线网格线
  31.         svg.append(drawGrid(minYearMonth, maxYearMonth, minPrice, maxPrice));
  32.         // 画深色坐标轴
  33.         svg.append(drawAxis(minYearMonth, maxYearMonth, minPrice, maxPrice));
  34.         // 画浅色平滑折线
  35.         svg.append(drawSmoothLine(data, minYearMonth, maxYearMonth, minPrice, maxPrice));
  36.         svg.append("</svg>");
  37.         return svg.toString();
  38.     }
  39.     // 画虚线网格线
  40.     private static String drawGrid(String minYearMonth, String maxYearMonth, double minPrice, double maxPrice) {
  41.         StringBuilder grid = new StringBuilder();
  42.         // 水平网格线(价格)
  43.         int numHorizontalLines = 5;
  44.         for (int i = 0; i <= numHorizontalLines; i++) {
  45.             double price = minPrice + (maxPrice - minPrice) * i / numHorizontalLines;
  46.             int y = HEIGHT - PADDING - (int) ((price - minPrice) * (HEIGHT - 2 * PADDING) / (maxPrice - minPrice));
  47.             grid.append(String.format("<line x1='%d' y1='%d' x2='%d' y2='%d' stroke='#e0e0e0' stroke-width='1' stroke-dasharray='5,5'/>",
  48.                     PADDING, y, WIDTH - PADDING, y));
  49.         }
  50.         // 垂直网格线(时间)
  51.         List<String> yearMonths = generateYearMonths(minYearMonth, maxYearMonth);
  52.         for (String yearMonth : yearMonths) {
  53.             int x = PADDING + (int) ((getMonthIndex(yearMonth) - getMonthIndex(minYearMonth)) * (WIDTH - 2 * PADDING) / (yearMonths.size() - 1));
  54.             grid.append(String.format("<line x1='%d' y1='%d' x2='%d' y2='%d' stroke='#e0e0e0' stroke-width='1' stroke-dasharray='5,5'/>",
  55.                     x, PADDING, x, HEIGHT - PADDING));
  56.         }
  57.         return grid.toString();
  58.     }
  59.     // 画深色坐标轴
  60.     private static String drawAxis(String minYearMonth, String maxYearMonth, double minPrice, double maxPrice) {
  61.         StringBuilder axis = new StringBuilder();
  62.         // X轴(时间)
  63.         axis.append(String.format("<line x1='%d' y1='%d' x2='%d' y2='%d' stroke='#333333' stroke-width='2'/>",
  64.                 PADDING, HEIGHT - PADDING, WIDTH - PADDING, HEIGHT - PADDING));
  65.         // Y轴(价格)
  66.         axis.append(String.format("<line x1='%d' y1='%d' x2='%d' y2='%d' stroke='#333333' stroke-width='2'/>",
  67.                 PADDING, PADDING, PADDING, HEIGHT - PADDING));
  68.         // X轴标签
  69.         List<String> yearMonths = generateYearMonths(minYearMonth, maxYearMonth);
  70.         for (String yearMonth : yearMonths) {
  71.             int x = PADDING + (int) ((getMonthIndex(yearMonth) - getMonthIndex(minYearMonth)) * (WIDTH - 2 * PADDING) / (yearMonths.size() - 1));
  72.             axis.append(String.format("<text x='%d' y='%d' font-size='12' font-family='Arial' fill='#000000' text-anchor='middle'>%s</text>",
  73.                     x, HEIGHT - PADDING + 20, yearMonth));
  74.         }
  75.         // Y轴标签
  76.         int numYLabels = 5;
  77.         for (int i = 0; i <= numYLabels; i++) {
  78.             double price = minPrice + (maxPrice - minPrice) * i / numYLabels;
  79.             int y = HEIGHT - PADDING - (int) ((price - minPrice) * (HEIGHT - 2 * PADDING) / (maxPrice - minPrice));
  80.             axis.append(String.format("<text x='%d' y='%d' font-size='12' font-family='Arial' fill='#000000' text-anchor='end'>%.2f</text>",
  81.                     PADDING - 10, y + 5, price));
  82.         }
  83.         return axis.toString();
  84.     }
  85.     // 画浅色平滑折线
  86.     private static String drawSmoothLine(List<PriceData> data, String minYearMonth, String maxYearMonth, double minPrice, double maxPrice) {
  87.         if (data.size() < 2) {
  88.             return "";
  89.         }
  90.         StringBuilder path = new StringBuilder("<path d='M");
  91.         for (int i = 0; i < data.size(); i++) {
  92.             PriceData point = data.get(i);
  93.             int x = PADDING + (int) ((getMonthIndex(point.getYearmonth()) - getMonthIndex(minYearMonth)) * (WIDTH - 2 * PADDING) / (getMonthIndex(maxYearMonth) - getMonthIndex(minYearMonth)));
  94.             int y = HEIGHT - PADDING - (int) ((point.getPrice() - minPrice) * (HEIGHT - 2 * PADDING) / (maxPrice - minPrice));
  95.             if (i == 0) {
  96.                 path.append(x).append(",").append(y);
  97.             } else {
  98.                 // 计算控制点以实现平滑效果
  99.                 PriceData prevPoint = data.get(i - 1);
  100.                 int prevX = PADDING + (int) ((getMonthIndex(prevPoint.getYearmonth()) - getMonthIndex(minYearMonth)) * (WIDTH - 2 * PADDING) / (getMonthIndex(maxYearMonth) - getMonthIndex(minYearMonth)));
  101.                 int prevY = HEIGHT - PADDING - (int) ((prevPoint.getPrice() - minPrice) * (HEIGHT - 2 * PADDING) / (maxPrice - minPrice));
  102.                 int ctrlX1 = (prevX + x) / 2;
  103.                 int ctrlY1 = prevY;
  104.                 int ctrlX2 = (prevX + x) / 2;
  105.                 int ctrlY2 = y;
  106.                 path.append(" C").append(ctrlX1).append(",").append(ctrlY1)
  107.                         .append(" ").append(ctrlX2).append(",").append(ctrlY2)
  108.                         .append(" ").append(x).append(",").append(y);
  109.             }
  110.         }
  111.         path.append("' fill='none' stroke='#66B3FF' stroke-width='2'/>");
  112.         return path.toString();
  113.     }
  114.     // 生成从 minYearMonth 到 maxYearMonth 的连续月份列表
  115.     private static List<String> generateYearMonths(String minYearMonth, String maxYearMonth) {
  116.         List<String> yearMonths = new ArrayList<>();
  117.         int year = Integer.parseInt(minYearMonth.substring(0, 4));
  118.         int month = Integer.parseInt(minYearMonth.substring(5));
  119.         while (true) {
  120.             yearMonths.add(String.format("%04d-%02d", year, month));
  121.             if (yearMonths.get(yearMonths.size() - 1).equals(maxYearMonth)) {
  122.                 break;
  123.             }
  124.             month++;
  125.             if (month > 12) {
  126.                 month = 1;
  127.                 year++;
  128.             }
  129.         }
  130.         return yearMonths;
  131.     }
  132.     // 将 yearmonth 转换为索引(从 0 开始)
  133.     private static int getMonthIndex(String yearMonth) {
  134.         int year = Integer.parseInt(yearMonth.substring(0, 4));
  135.         int month = Integer.parseInt(yearMonth.substring(5));
  136.         return (year - 2024) * 12 + (month - 1); // 假设最小年份是 2024
  137.     }
  138.     public static void main(String[] args) {
  139.         List<PriceData> data = PriceData.getSampleData();
  140.         String title = "价格走势图";
  141.         String subtitle = "2024-06 至 2025-01";
  142.         String svg = SvgUtils.generatePriceTrendSvg(data, title, subtitle);
  143.         System.out.println(svg);
  144.     }
  145. }
复制代码
  1. @Data
  2. public class PriceData {
  3.     private String yearmonth;
  4.     private Double price;
  5.     public PriceData() {}
  6.     public PriceData(String yearmonth, Double price) {
  7.         this.yearmonth = yearmonth;
  8.         this.price = price;
  9.     }
  10.     public static List<PriceData> getSampleData() {
  11.         List<PriceData> data = new ArrayList<>();
  12.         data.add(new PriceData("2025-01", 100.0)); // 2023-01-01
  13.         data.add(new PriceData("2025-02", 105.0)); // 2023-01-02
  14.         data.add(new PriceData("2025-03", 102.0)); // 2023-01-03
  15.         data.add(new PriceData("2025-04", 110.0)); // 2023-01-04
  16.         data.add(new PriceData("2025-05", 108.0)); // 2023-01-05
  17.         return data;
  18.     }
  19. }
复制代码
OK, 生成的SVG嵌入到html网页中进行渲染即可

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

本帖子中包含更多资源

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

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

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

本版积分规则

宁睿

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表