js-angusj-clipper
Version:
Polygon and line clipping and offsetting library for Javascript / Typescript - a port of Angus Johnson's clipper to WebAssembly / Asm.JS
70 lines • 14 kB
JavaScript
;
var __assign = (this && this.__assign) || function () {
__assign = Object.assign || function(t) {
for (var s, i = 1, n = arguments.length; i < n; i++) {
s = arguments[i];
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
t[p] = s[p];
}
return t;
};
return __assign.apply(this, arguments);
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.offsetToPolyTree = exports.offsetToPaths = void 0;
var ClipperOffset_1 = require("./ClipperOffset");
var addPathOrPaths = function (offset, inputDatas) {
if (inputDatas === undefined) {
return;
}
// add each input
for (var i = 0, maxi = inputDatas.length; i < maxi; i++) {
var inputData = inputDatas[i];
// add the path/paths
var pathOrPaths = inputData.data;
if (!pathOrPaths || pathOrPaths.length <= 0) {
continue;
}
// is it a path or paths?
if (Array.isArray(pathOrPaths[0])) {
// paths
offset.addPaths(pathOrPaths, inputData.joinType, inputData.endType);
}
else {
// path
offset.addPath(pathOrPaths, inputData.joinType, inputData.endType);
}
}
};
function offsetToPathsOrPolyTree(polyTreeMode, nativeClipperLib, params) {
var filledData = __assign({ arcTolerance: 0.25, miterLimit: 2 }, params);
var offset = new ClipperOffset_1.ClipperOffset(nativeClipperLib, filledData.miterLimit, filledData.arcTolerance);
//noinspection UnusedCatchParameterJS
try {
addPathOrPaths(offset, params.offsetInputs);
if (!polyTreeMode) {
return offset.executeToPaths(params.delta, params.cleanDistance);
}
else {
if (params.cleanDistance !== undefined) {
return undefined; // cleaning is not available for poly tree results
}
return offset.executeToPolyTree(params.delta);
}
}
catch (err) {
return undefined;
}
finally {
offset.dispose();
}
}
function offsetToPaths(nativeClipperLib, params) {
return offsetToPathsOrPolyTree(false, nativeClipperLib, params);
}
exports.offsetToPaths = offsetToPaths;
function offsetToPolyTree(nativeClipperLib, params) {
return offsetToPathsOrPolyTree(true, nativeClipperLib, params);
}
exports.offsetToPolyTree = offsetToPolyTree;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"offsetFunctions.js","sourceRoot":"","sources":["../src/offsetFunctions.ts"],"names":[],"mappings":";;;;;;;;;;;;;;AAAA,iDAAgD;AA0FhD,IAAM,cAAc,GAAG,UAAC,MAAqB,EAAE,UAAqC;IAClF,IAAI,UAAU,KAAK,SAAS,EAAE;QAC5B,OAAO;KACR;IAED,iBAAiB;IACjB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,IAAI,GAAG,UAAU,CAAC,MAAM,EAAE,CAAC,GAAG,IAAI,EAAE,CAAC,EAAE,EAAE;QACvD,IAAM,SAAS,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;QAEhC,qBAAqB;QACrB,IAAM,WAAW,GAAG,SAAS,CAAC,IAAI,CAAC;QACnC,IAAI,CAAC,WAAW,IAAI,WAAW,CAAC,MAAM,IAAI,CAAC,EAAE;YAC3C,SAAS;SACV;QAED,yBAAyB;QACzB,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,EAAE;YACjC,QAAQ;YACR,MAAM,CAAC,QAAQ,CAAC,WAA4B,EAAE,SAAS,CAAC,QAAQ,EAAE,SAAS,CAAC,OAAO,CAAC,CAAC;SACtF;aAAM;YACL,OAAO;YACP,MAAM,CAAC,OAAO,CAAC,WAA2B,EAAE,SAAS,CAAC,QAAQ,EAAE,SAAS,CAAC,OAAO,CAAC,CAAC;SACpF;KACF;AACH,CAAC,CAAC;AAEF,SAAS,uBAAuB,CAC9B,YAAqB,EACrB,gBAA0C,EAC1C,MAAoB;IAEpB,IAAM,UAAU,cACd,YAAY,EAAE,IAAI,EAClB,UAAU,EAAE,CAAC,IACV,MAAM,CACV,CAAC;IAEF,IAAM,MAAM,GAAG,IAAI,6BAAa,CAC9B,gBAAgB,EAChB,UAAU,CAAC,UAAU,EACrB,UAAU,CAAC,YAAY,CACxB,CAAC;IAEF,qCAAqC;IACrC,IAAI;QACF,cAAc,CAAC,MAAM,EAAE,MAAM,CAAC,YAAY,CAAC,CAAC;QAC5C,IAAI,CAAC,YAAY,EAAE;YACjB,OAAO,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,aAAa,CAAC,CAAC;SAClE;aAAM;YACL,IAAI,MAAM,CAAC,aAAa,KAAK,SAAS,EAAE;gBACtC,OAAO,SAAS,CAAC,CAAC,kDAAkD;aACrE;YACD,OAAO,MAAM,CAAC,iBAAiB,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;SAC/C;KACF;IAAC,OAAO,GAAG,EAAE;QACZ,OAAO,SAAS,CAAC;KAClB;YAAS;QACR,MAAM,CAAC,OAAO,EAAE,CAAC;KAClB;AACH,CAAC;AAED,SAAgB,aAAa,CAC3B,gBAA0C,EAC1C,MAAoB;IAEpB,OAAO,uBAAuB,CAAC,KAAK,EAAE,gBAAgB,EAAE,MAAM,CAAsB,CAAC;AACvF,CAAC;AALD,sCAKC;AAED,SAAgB,gBAAgB,CAC9B,gBAA0C,EAC1C,MAAoB;IAEpB,OAAO,uBAAuB,CAAC,IAAI,EAAE,gBAAgB,EAAE,MAAM,CAAyB,CAAC;AACzF,CAAC;AALD,4CAKC","sourcesContent":["import { ClipperOffset } from \"./ClipperOffset\";\nimport { EndType, JoinType } from \"./enums\";\nimport { NativeClipperLibInstance } from \"./native/NativeClipperLibInstance\";\nimport { ReadonlyPath } from \"./Path\";\nimport { Paths, ReadonlyPaths } from \"./Paths\";\nimport { PolyTree } from \"./PolyTree\";\n\n/**\n * A single input (of multiple possible inputs) for the offsetToPaths / offsetToPolyTree operation.\n */\nexport interface OffsetInput {\n  /**\n   * Join type.\n   */\n  joinType: JoinType;\n\n  /**\n   * End type.\n   */\n  endType: EndType;\n\n  /**\n   * Data of one of the Path or Paths to be used in preparation for offsetting.\n   *\n   * All 'outer' Paths must have the same orientation, and any 'hole' paths must have reverse orientation. Closed paths must have at least 3 vertices.\n   * Open paths may have as few as one vertex. Open paths can only be offset with positive deltas.\n   */\n  data: ReadonlyPath | ReadonlyPaths;\n}\n\n/**\n * Params for the polygon offset operation.\n */\nexport interface OffsetParams {\n  /**\n   * Firstly, this field/property is only relevant when JoinType = Round and/or EndType = Round.\n   *\n   * Since flattened paths can never perfectly represent arcs, this field/property specifies a maximum acceptable imprecision ('tolerance') when arcs are\n   * approximated in an offsetting operation. Smaller values will increase 'smoothness' up to a point though at a cost of performance and in creating more\n   * vertices to construct the arc.\n   *\n   * The default ArcTolerance is 0.25 units. This means that the maximum distance the flattened path will deviate from the 'true' arc will be no more\n   * than 0.25 units (before rounding).\n   *\n   * Reducing tolerances below 0.25 will not improve smoothness since vertex coordinates will still be rounded to integer values. The only way to achieve\n   * sub-integer precision is through coordinate scaling before and after offsetting (see example below).\n   *\n   * It's important to make ArcTolerance a sensible fraction of the offset delta (arc radius). Large tolerances relative to the offset delta will produce\n   * poor arc approximations but, just as importantly, very small tolerances will substantially slow offsetting performance while providing unnecessary\n   * degrees of precision. This is most likely to be an issue when offsetting polygons whose coordinates have been scaled to preserve floating point precision.\n   *\n   * Example: Imagine a set of polygons (defined in floating point coordinates) that is to be offset by 10 units using round joins, and the solution is to\n   * retain floating point precision up to at least 6 decimal places.\n   * To preserve this degree of floating point precision, and given that Clipper and ClipperOffset both operate on integer coordinates, the polygon\n   * coordinates will be scaled up by 108 (and rounded to integers) prior to offsetting. Both offset delta and ArcTolerance will also need to be scaled\n   * by this same factor. If ArcTolerance was left unscaled at the default 0.25 units, every arc in the solution would contain a fraction of 44 THOUSAND\n   * vertices while the final arc imprecision would be 0.25 × 10-8 units (ie once scaling was reversed). However, if 0.1 units was an acceptable imprecision\n   * in the final unscaled solution, then ArcTolerance should be set to 0.1 × scaling_factor (0.1 × 108 ). Now if scaling is applied equally to both\n   * ArcTolerance and to Delta Offset, then in this example the number of vertices (steps) defining each arc would be a fraction of 23.\n   *\n   * The formula for the number of steps in a full circular arc is ... Pi / acos(1 - arc_tolerance / abs(delta))\n   */\n  arcTolerance?: number; // defaults to 0.25\n\n  /**\n   * This property sets the maximum distance in multiples of delta that vertices can be offset from their original positions before squaring is applied.\n   * (Squaring truncates a miter by 'cutting it off' at 1 × delta distance from the original vertex.)\n   *\n   * The default value for MiterLimit is 2 (ie twice delta). This is also the smallest MiterLimit that's allowed. If mitering was unrestricted (ie without\n   * any squaring), then offsets at very acute angles would generate unacceptably long 'spikes'.\n   */\n  miterLimit?: number; // defaults to 2 (twice delta)\n\n  /**\n   * Negative delta values shrink polygons and positive delta expand them.\n   */\n  delta: number;\n\n  /**\n   * One or more inputs to use for the offset operation.\n   */\n  offsetInputs: OffsetInput[];\n\n  /**\n   * If this is not undefined then cleaning of the result polygon will be performed.\n   * This operation is only available when the output format is not a poly tree.\n   */\n  cleanDistance?: number;\n}\n\nconst addPathOrPaths = (offset: ClipperOffset, inputDatas: OffsetInput[] | undefined) => {\n  if (inputDatas === undefined) {\n    return;\n  }\n\n  // add each input\n  for (let i = 0, maxi = inputDatas.length; i < maxi; i++) {\n    const inputData = inputDatas[i];\n\n    // add the path/paths\n    const pathOrPaths = inputData.data;\n    if (!pathOrPaths || pathOrPaths.length <= 0) {\n      continue;\n    }\n\n    // is it a path or paths?\n    if (Array.isArray(pathOrPaths[0])) {\n      // paths\n      offset.addPaths(pathOrPaths as ReadonlyPaths, inputData.joinType, inputData.endType);\n    } else {\n      // path\n      offset.addPath(pathOrPaths as ReadonlyPath, inputData.joinType, inputData.endType);\n    }\n  }\n};\n\nfunction offsetToPathsOrPolyTree(\n  polyTreeMode: boolean,\n  nativeClipperLib: NativeClipperLibInstance,\n  params: OffsetParams\n): Paths | PolyTree | undefined {\n  const filledData = {\n    arcTolerance: 0.25,\n    miterLimit: 2,\n    ...params,\n  };\n\n  const offset = new ClipperOffset(\n    nativeClipperLib,\n    filledData.miterLimit,\n    filledData.arcTolerance\n  );\n\n  //noinspection UnusedCatchParameterJS\n  try {\n    addPathOrPaths(offset, params.offsetInputs);\n    if (!polyTreeMode) {\n      return offset.executeToPaths(params.delta, params.cleanDistance);\n    } else {\n      if (params.cleanDistance !== undefined) {\n        return undefined; // cleaning is not available for poly tree results\n      }\n      return offset.executeToPolyTree(params.delta);\n    }\n  } catch (err) {\n    return undefined;\n  } finally {\n    offset.dispose();\n  }\n}\n\nexport function offsetToPaths(\n  nativeClipperLib: NativeClipperLibInstance,\n  params: OffsetParams\n): Paths | undefined {\n  return offsetToPathsOrPolyTree(false, nativeClipperLib, params) as Paths | undefined;\n}\n\nexport function offsetToPolyTree(\n  nativeClipperLib: NativeClipperLibInstance,\n  params: OffsetParams\n): PolyTree | undefined {\n  return offsetToPathsOrPolyTree(true, nativeClipperLib, params) as PolyTree | undefined;\n}\n"]}