0

0

动态生成 SVG 多边形:基于 XML 数据驱动的坐标绑定与渲染教程

花韻仙語

花韻仙語

发布时间:2026-03-17 11:53:01

|

748人浏览过

|

来源于php中文网

原创

动态生成 SVG 多边形:基于 XML 数据驱动的坐标绑定与渲染教程

本文详解如何在原生 JavaScript 中解析外部 XML 数据,将其字段映射为 SVG <polygon> 的动态顶点坐标,并通过规范化字符串拼接与坐标计算实现可维护、可扩展的矢量图形渲染。

本文详解如何在原生 javascript 中解析外部 xml 数据,将其字段映射为 svg `` 的动态顶点坐标,并通过规范化字符串拼接与坐标计算实现可维护、可扩展的矢量图形渲染。

在构建数据可视化报告(如 JasperSoft Studio 生成的 SVG 图表)时,常需根据实时变化的配置数据(如设备尺寸、参数阈值)动态绘制几何图形。SVG 的 <polygon> 元素虽不支持模板语法或绑定表达式,但完全可通过 JavaScript 精确控制其 points 属性——关键在于正确构造符合 SVG 规范的坐标字符串:每对 (x, y) 坐标须以空格分隔,且 x 与 y 之间必须用英文逗号连接(如 "10,20 30,40 50,60"),缺一不可。

以下是一个完整、健壮的实现方案,涵盖 XML 解析、变量提取、坐标计算与 SVG 渲染全流程:

✅ 正确构造 points 字符串:数组 + join(' ')

错误写法(缺少逗号、无分隔):

// ❌ 错误:坐标未用逗号分隔,且无空格分隔各点
brjambshape.setAttribute("points", 
  (bx + ...) (by + ...) (bx + ...) (by + ...) // 语法错误,JS 报错
);

正确写法(推荐):

皮卡智能
皮卡智能

AI驱动高效视觉设计平台

下载
// ✅ 正确:将所有 x、y 坐标按顺序推入数组,再用空格连接
const points = [
  bx + (depth - Cstmr - 4.5) * scale, by + dy + Height * scale,
  bx + (depth - Cstmr) * scale,      by + dy + Height * scale,
  bx + (depth - Cstmr) * scale,      by + dy + (Height - 2.5) * scale,
  bx + (depth - Cstmr - 0.75) * scale, by + dy + (Height - 2.5) * scale,
  bx + (depth - Cstmr - 0.75) * scale, by + dy + (Height - 1.75) * scale,
  bx + (depth - Cstmr - 4.5) * scale, by + dy + (Height - 1.75) * scale,
  bx + (depth - Cstmr - 4.5) * scale, by + dy + Height * scale
];
brjambshape.setAttribute("points", points.join(' '));

? 解析 XML 数据为 JavaScript 对象

使用标准 DOMParser 解析 XML 字符串,并递归转换为扁平化 JS 对象(含自动类型推断):

function xmlStringToJSO(xmlString) {
  const parser = new DOMParser();
  const xmlDoc = parser.parseFromString(`<root>${xmlString}</root>`, "text/xml");
  const root = xmlDoc.querySelector('root');

  const xml2Jso = (node) => {
    const obj = {};
    if (node.children.length > 0) {
      for (let i = 0; i < node.children.length; i++) {
        const child = node.children[i];
        const key = child.nodeName;
        const value = xml2Jso(child);
        if (obj[key] === undefined) {
          obj[key] = value;
        } else {
          if (!Array.isArray(obj[key])) obj[key] = [obj[key]];
          obj[key].push(value);
        }
      }
    } else {
      const text = node.textContent.trim();
      return isFinite(text) ? parseFloat(text) : text;
    }
    return obj;
  };

  return xml2Jso(root);
}

// 示例 XML 数据(实际中来自 JasperSoft 的 instanceData 或 API)
const xml = `
  <Depth>12</Depth>
  <Cstmr>3.5</Cstmr>
  <Vndr>4</Vndr>
  <JH>1.75</JH>
  <Width>12</Width>
  <Height>34</Height>
  <BHeight>2</BHeight>
  <BWidth>2</BWidth>
  <TYPE>4</TYPE>
`;

const instanceData = xmlStringToJSO(xml);
const { Width, Height, Depth, Cstmr, TYPE } = instanceData; // 解构赋值,提升可读性

⚙️ 构建 SVG 容器与图形组

const svgns = "http://www.w3.org/2000/svg";
const svg = document.createElementNS(svgns, "svg");
svg.setAttribute("width", Width);
svg.setAttribute("height", Height);

// 自适应缩放逻辑(保持比例居中)
const b = Math.min(Width, Height) / 10;
const dx = Width - b * 2;
const dy = Height - b * 2;
const scale = Math.min((dx / Width) * 0.75, (dy / Height) * 0.75);
const bx = b + (dx - Width * scale) / 2;
const by = b - (dy - Height * scale) / 2;

// 创建图形组
const brJambGroup = document.createElementNS(svgns, "g");
brJambGroup.id = "mygroup";

// 创建多边形
const brjambshape = document.createElementNS(svgns, "polygon");
brjambshape.setAttribute("stroke", "red");
brjambshape.setAttribute("fill", "none");

// 绑定动态坐标(核心步骤)
const points = [
  bx + (Depth - Cstmr - 4.5) * scale, by + dy + Height * scale,
  bx + (Depth - Cstmr) * scale,      by + dy + Height * scale,
  bx + (Depth - Cstmr) * scale,      by + dy + (Height - 2.5) * scale,
  bx + (Depth - Cstmr - 0.75) * scale, by + dy + (Height - 2.5) * scale,
  bx + (Depth - Cstmr - 0.75) * scale, by + dy + (Height - 1.75) * scale,
  bx + (Depth - Cstmr - 4.5) * scale, by + dy + (Height - 1.75) * scale,
  bx + (Depth - Cstmr - 4.5) * scale, by + dy + Height * scale
];
brjambshape.setAttribute("points", points.join(' '));
brJambGroup.appendChild(brjambshape);

// 条件渲染
if (TYPE === 4) {
  svg.appendChild(brJambGroup);
}
document.body.appendChild(svg);

// 可选:自动适配 viewBox(提升响应式表现)
const bb = svg.getBBox();
svg.setAttribute("viewBox", `${bb.x} ${bb.y} ${bb.width} ${bb.height}`);

⚠️ 注意事项与最佳实践

  • 命名一致性:确保 XML 字段名(如 <Depth>)与 JS 变量名(Depth)严格匹配,区分大小写;
  • 数值安全:始终使用 parseFloat() 或解构时的自动类型转换,避免字符串参与运算;
  • 坐标单位:SVG 坐标系原点在左上角,y 增加方向向下,注意 by -= ... 类调整是否符合预期;
  • 性能考量:若需频繁重绘(如动画),建议复用 <polygon> 元素并仅更新 points 属性,而非反复创建/销毁;
  • 替代方案优先级:若后端支持 JSON 输出,应优先选用 JSON.parse() 替代 XML 解析,减少解析开销与容错复杂度;
  • JasperSoft 集成提示:在 .jrxml 中,确保 instanceData 已正确注入全局作用域;若使用 <![CDATA[...]]> 块,需确保 XML 片段被完整传递。

通过以上结构化方法,你不仅能可靠地将 XML 参数驱动 SVG 图形,还能轻松扩展至多边形、路径(<path>)、甚至组合图形(<g> 嵌套),真正实现“一份配置,处处渲染”的工程目标。

相关文章

驱动精灵
驱动精灵

驱动精灵基于驱动之家十余年的专业数据积累,驱动支持度高,已经为数亿用户解决了各种电脑驱动问题、系统故障,是目前有效的驱动软件,有需要的小伙伴快来保存下载体验吧!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

458

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

549

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

337

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

84

2025.09.10

pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1950

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2119

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1177

2024.11.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

c++ 字符处理
c++ 字符处理

本专题整合了c++字符处理教程、字符串处理函数相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.17

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
svg中文手册
svg中文手册

共0课时 | 0.3万人学习

SVG 教程
SVG 教程

共20课时 | 13.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号