将两个基于贝塞尔曲线的形状合并为一个以创建新轮廓

Merging two bezier-based shapes into one to create a new outline

假设我有数据来渲染两个重叠的基于贝塞尔曲线的形状,它们重叠,显示在 svg 或 canvas 上(在哪里并不重要)。我想计算由两个形状合并产生的形状轮廓,以便我有一个干净的(新的)轮廓以及尽可能少的节点和手柄。 我想实现 adobe illustrator 等矢量程序通过 Pathfinder > Add 或字体程序 glyphs[= 提供的效果26=] 与 删除重叠 。 示例:https://helpx.adobe.com/illustrator/using/combining-objects.html

是否有可能用于该任务的库或概念? 我在浏览器中使用 javascript,但有关如何进行此类计算的任何其他来源也会有所帮助。

同样重要的是,此计算发生在渲染之前,与渲染结果不可知(svg/canvas)。

在下图中,左侧是输入形状。右边是预期的结果。我有数据,即所有节点和手柄(来自贝塞尔曲线),我想计算右侧的红色(节点)和绿色点(手柄)的坐标。

Paper.js 可能是完成此任务的完美库:
特别是 Boolean operations – 就像 unite() 一样合并路径元素。语法看起来像这样:

let unitedPath = items[0].unite(items[1]);  

以下示例还使用了 Jarek Foksa 的 pathData polyfill。

示例:联合路径:

const svg = document.querySelector("#svgunite");
const btnDownload = document.querySelector("#btnDownload");
const decimals = 1;
// set auto ids for processing
function setAutoIDs(svg) {
  let svgtEls = svg.querySelectorAll(
    "path, polygon, rect, circle, line, text, g"
  );
  svgtEls.forEach(function(el, i) {
    if (!el.getAttribute("id")) {
      el.id = el.nodeName + "-" + i;
    }
  });
}
setAutoIDs(svg);

function shapesToPathMerged(svg) {
  let els = svg.querySelectorAll('path, rect, circle, polygon, ellipse ');
  let pathsCombinedData = '';
  let className = els[1].getAttribute('class');
  let id = els[1].id;
  let d = els[1].getAttribute('d');
  let fill = els[1].getAttribute('fill');

  els.forEach(function(el, i) {
    let pathData = el.getPathData({
      normalize: true
    });
    if (i == 0 && el.nodeName.toLowerCase() != 'path') {
      let firstTmp = document.createElementNS("http://www.w3.org/2000/svg", 'path');
      let firstClassName = els[1].getAttribute('class');
      let firstId = el.id;
      let firstFill = el.getAttribute('fill');
      firstTmp.setPathData(pathData);
      firstTmp.id = firstId;
      firstTmp.setAttribute('class', firstClassName);
      firstTmp.setAttribute('fill', firstFill);
      svg.insertBefore(firstTmp, el);
      el.remove();
    }
    if (i > 0) {
      pathData.forEach(function(command, c) {
        pathsCombinedData += ' ' + command['type'] + '' + command['values'].join(' ');
      });
      el.remove();
    }
  })
  let pathTmp = document.createElementNS("http://www.w3.org/2000/svg", 'path');
  pathTmp.id = id;
  pathTmp.setAttribute('class', className);
  pathTmp.setAttribute('fill', fill);
  pathTmp.setAttribute('d', pathsCombinedData);
  svg.insertBefore(pathTmp, els[0].nextElementSibling);
};

shapesToPathMerged(svg);


function unite(svg) {
  // init paper.js and add mandatory canvas
  canvas = document.createElement('canvas');
  canvas.id = "canvasPaper";
  canvas.setAttribute('style', 'display:none')
  document.body.appendChild(canvas);
  paper.setup("canvasPaper");

  let all = paper.project.importSVG(svg, function(item, i) {
    let items = item.getItems();
    // remove first item not containing path data
    items.shift();
    // get id names for selecting svg elements after processing
    let ids = Object.keys(item._namedChildren);

    if (items.length) {
      let lastEl = items[items.length - 1];
      // unite paper.js objects
      let united = items[0].unite(lastEl);
      // convert united paper.js object to svg pathData
      let unitedData = united
        .exportSVG({
          precision: decimals
        })
        .getAttribute("d");
      let svgElFirst = svg.querySelector('#' + ids[0]);
      let svgElLast = svg.querySelector('#' + ids[ids.length - 1]);
      // overwrite original svg path
      svgElFirst.setAttribute("d", unitedData);
      // delete united svg path
      svgElLast.remove();
    }
  });
  // get data URL
  getdataURL(svg)

}

function getdataURL(svg) {
  let markup = svg.outerHTML;
  markupOpt = 'data:image/svg+xml;utf8,' + markup.replaceAll('"', '\'').
  replaceAll('\t', '').
  replaceAll('\n', '').
  replaceAll('\r', '').
  replaceAll('></path>', '/>').
  replaceAll('<', '%3C').
  replaceAll('>', '%3E').
  replaceAll('#', '%23').
  replaceAll(',', ' ').
  replaceAll(' -', '-').
  replace(/ +(?= )/g, '');

  let btn = document.createElement('a');
  btn.href = markupOpt;
  btn.innerText = 'Download Svg';
  btn.setAttribute('download', 'united.svg');
  document.body.insertAdjacentElement('afterbegin', btn);
  return markupOpt;
}
svg{
  display:inline-block;
  width:10em
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/paper.js/0.12.0/paper-full.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/path-data-polyfill@1.0.3/path-data-polyfill.min.js"></script>


<p>
  <button type="button" onclick="unite(svg)">Subtract Path </button>
</p>
<svg class="svgunite" id="svgunite" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100" stroke-width="1" stroke="#000">
<path fill="none" d="M50.05 23.21l-19.83 61.51h-9.27l23.6-69.44h10.82l23.7 69.44h-9.58l-20.44-61.51h1z"/>
<rect fill="none" x="35.49" y="52.75" width="28.5" height="6.17">
</rect>
</svg>

可选:路径规范化(使用 getPathData() polyfill)

您可能还需要转换 svg 图元(<rect><circle><polygon>),例如大写 A 中的横线。

pathData polyfill提供了normalizing svg elements.
的方法 此规范化将输出一个 d 属性(对于每个选定的 svg 子元素)仅包含 简化的立方路径命令集(M、C、L、Z) – 全部基于绝对坐标.

小郁闷:
我不会说 paper.js 可以吹嘘过多的教程或详细示例。但是您可以检查 reference for pathItem to see all options.

另请参阅: