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
76 lines • 18.8 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.clipToPolyTree = exports.clipToPaths = exports.clipToPathsOrPolyTree = void 0;
var Clipper_1 = require("./Clipper");
var ClipperError_1 = require("./ClipperError");
var enums_1 = require("./enums");
var devMode = typeof "process" !== "undefined" && process.env && process.env.NODE_ENV !== "production";
var addPathOrPaths = function (clipper, inputDatas, polyType) {
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;
}
var closed_1 = inputData.closed === undefined ? true : inputData.closed;
// is it a path or paths?
if (Array.isArray(pathOrPaths[0])) {
// paths
if (!clipper.addPaths(pathOrPaths, polyType, closed_1)) {
throw new ClipperError_1.ClipperError("invalid paths");
}
}
else {
// path
if (!clipper.addPath(pathOrPaths, polyType, closed_1)) {
throw new ClipperError_1.ClipperError("invalid path");
}
}
}
};
function clipToPathsOrPolyTree(polyTreeMode, nativeClipperLib, params) {
if (devMode) {
if (!polyTreeMode && params.subjectInputs && params.subjectInputs.some(function (si) { return !si.closed; })) {
throw new Error("clip to a PolyTree (not to a Path) when using open paths");
}
}
var clipper = new Clipper_1.Clipper(nativeClipperLib, params);
//noinspection UnusedCatchParameterJS
try {
addPathOrPaths(clipper, params.subjectInputs, enums_1.PolyType.Subject);
addPathOrPaths(clipper, params.clipInputs, enums_1.PolyType.Clip);
var result = void 0;
var clipFillType = params.clipFillType === undefined ? params.subjectFillType : params.clipFillType;
if (!polyTreeMode) {
result = clipper.executeToPaths(params.clipType, params.subjectFillType, clipFillType, params.cleanDistance);
}
else {
if (params.cleanDistance !== undefined) {
throw new ClipperError_1.ClipperError("cleaning is not available for poly tree results");
}
result = clipper.executeToPolyTee(params.clipType, params.subjectFillType, clipFillType);
}
if (result === undefined) {
throw new ClipperError_1.ClipperError("error while performing clipping task");
}
return result;
}
finally {
clipper.dispose();
}
}
exports.clipToPathsOrPolyTree = clipToPathsOrPolyTree;
function clipToPaths(nativeClipperLib, params) {
return clipToPathsOrPolyTree(false, nativeClipperLib, params);
}
exports.clipToPaths = clipToPaths;
function clipToPolyTree(nativeClipperLib, params) {
return clipToPathsOrPolyTree(true, nativeClipperLib, params);
}
exports.clipToPolyTree = clipToPolyTree;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"clipFunctions.js","sourceRoot":"","sources":["../src/clipFunctions.ts"],"names":[],"mappings":"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","sourcesContent":["import { Clipper } from \"./Clipper\";\r\nimport { ClipperError } from \"./ClipperError\";\r\nimport { ClipType, PolyFillType, PolyType } from \"./enums\";\r\nimport { NativeClipperLibInstance } from \"./native/NativeClipperLibInstance\";\r\nimport { Path, ReadonlyPath } from \"./Path\";\r\nimport { Paths, ReadonlyPaths } from \"./Paths\";\r\nimport { PolyTree } from \"./PolyTree\";\r\n\r\nconst devMode =\r\n  typeof \"process\" !== \"undefined\" && process.env && process.env.NODE_ENV !== \"production\";\r\n\r\n/**\r\n * A single subject input (of multiple possible inputs) for the clipToPaths / clipToPolyTree operations\r\n *\r\n * 'Subject' paths may be either open (lines) or closed (polygons) or even a mixture of both.\r\n * With closed paths, orientation should conform with the filling rule that will be passed via Clipper's execute method.\r\n */\r\nexport interface SubjectInput {\r\n  /**\r\n   * Path / Paths data.\r\n   *\r\n   * Path Coordinate range:\r\n   * Path coordinates must be between ± 9007199254740991, otherwise a range error will be thrown when attempting to add the path to the Clipper object.\r\n   * If coordinates can be kept between ± 0x3FFFFFFF (± 1.0e+9), a modest increase in performance (approx. 15-20%) over the larger range can be achieved by\r\n   * avoiding large integer math.\r\n   *\r\n   * The function operation will throw an error if the path is invalid for clipping. A path is invalid for clipping when:\r\n   * - it has less than 2 vertices\r\n   * - it has 2 vertices but is not an open path\r\n   * - the vertices are all co-linear and it is not an open path\r\n   */\r\n  data: ReadonlyPath | ReadonlyPaths;\r\n\r\n  /**\r\n   * If the path/paths is closed or not.\r\n   */\r\n  closed: boolean;\r\n}\r\n\r\n/**\r\n * A single clip input (of multiple possible inputs) for the clipToPaths / clipToPolyTree operations.\r\n *\r\n * Clipping paths must always be closed. Clipper allows polygons to clip both lines and other polygons, but doesn't allow lines to clip either lines or polygons.\r\n * With closed paths, orientation should conform with the filling rule that will be passed via Clipper's execute method.\r\n */\r\nexport interface ClipInput {\r\n  /**\r\n   * Path / Paths data.\r\n   *\r\n   * Path Coordinate range:\r\n   * Path coordinates must be between ± 9007199254740991, otherwise a range error will be thrown when attempting to add the path to the Clipper object.\r\n   * If coordinates can be kept between ± 0x3FFFFFFF (± 1.0e+9), a modest increase in performance (approx. 15-20%) over the larger range can be achieved by\r\n   * avoiding large integer math.\r\n   *\r\n   * The function operation will throw an error if the path is invalid for clipping. A path is invalid for clipping when:\r\n   * - it has less than 2 vertices\r\n   * - it has 2 vertices but is not an open path\r\n   * - the vertices are all co-linear and it is not an open path\r\n   */\r\n  data: ReadonlyPath | ReadonlyPaths;\r\n}\r\n\r\n/**\r\n * Params for the clipToPaths / clipToPolyTree operations.\r\n *\r\n * Any number of subject and clip paths can be added to a clipping task.\r\n *\r\n * Boolean (clipping) operations are mostly applied to two sets of Polygons, represented in this library as subject and clip polygons. Whenever Polygons\r\n * are added to the Clipper object, they must be assigned to either subject or clip polygons.\r\n *\r\n * UNION operations can be performed on one set or both sets of polygons, but all other boolean operations require both sets of polygons to derive\r\n * meaningful solutions.\r\n */\r\nexport interface ClipParams {\r\n  /**\r\n   * Clipping operation type (Intersection, Union, Difference or Xor).\r\n   */\r\n  clipType: ClipType;\r\n\r\n  /**\r\n   * Winding (fill) rule for subject polygons.\r\n   */\r\n  subjectFillType: PolyFillType;\r\n\r\n  /**\r\n   * Subject inputs.\r\n   */\r\n  subjectInputs: SubjectInput[];\r\n\r\n  /**\r\n   * Winding (fill) rule for clipping polygons. If missing it will use the same one as subjectFillType.\r\n   */\r\n  clipFillType?: PolyFillType;\r\n\r\n  /**\r\n   * Clipping inputs. Not required for union operations, required for others.\r\n   */\r\n  clipInputs?: ClipInput[];\r\n\r\n  /**\r\n   * When this property is set to true, polygons returned in the solution parameter of the clip method will have orientations opposite to their normal\r\n   * orientations.\r\n   */\r\n  reverseSolution?: boolean;\r\n\r\n  /**\r\n   * Terminology:\r\n   * - A simple polygon is one that does not self-intersect.\r\n   * - A weakly simple polygon is a simple polygon that contains 'touching' vertices, or 'touching' edges.\r\n   * - A strictly simple polygon is a simple polygon that does not contain 'touching' vertices, or 'touching' edges.\r\n   *\r\n   * Vertices 'touch' if they share the same coordinates (and are not adjacent). An edge touches another if one of its end vertices touches another edge\r\n   * excluding its adjacent edges, or if they are co-linear and overlapping (including adjacent edges).\r\n   *\r\n   * Polygons returned by clipping operations (see Clipper.execute()) should always be simple polygons. When the StrictlySimply property is enabled,\r\n   * polygons returned will be strictly simple, otherwise they may be weakly simple. It's computationally expensive ensuring polygons are strictly simple\r\n   * and so this property is disabled by default.\r\n   *\r\n   * Note: There's currently no guarantee that polygons will be strictly simple since 'simplifying' is still a work in progress.\r\n   */\r\n  strictlySimple?: boolean;\r\n\r\n  /**\r\n   * By default, when three or more vertices are collinear in input polygons (subject or clip), the Clipper object removes the 'inner' vertices before\r\n   * clipping. When enabled the preserveCollinear property prevents this default behavior to allow these inner vertices to appear in the solution.\r\n   */\r\n  preserveCollinear?: boolean;\r\n\r\n  /**\r\n   * If this is not undefined then cleaning of the result polygon will be performed.\r\n   * This operation is only available when the output format is not a poly tree.\r\n   */\r\n  cleanDistance?: number;\r\n}\r\n\r\nconst addPathOrPaths = (\r\n  clipper: Clipper,\r\n  inputDatas: (SubjectInput | ClipInput)[] | undefined,\r\n  polyType: PolyType\r\n) => {\r\n  if (inputDatas === undefined) {\r\n    return;\r\n  }\r\n\r\n  // add each input\r\n  for (let i = 0, maxi = inputDatas.length; i < maxi; i++) {\r\n    const inputData = inputDatas[i];\r\n\r\n    // add the path/paths\r\n    const pathOrPaths = inputData.data;\r\n    if (!pathOrPaths || pathOrPaths.length <= 0) {\r\n      continue;\r\n    }\r\n\r\n    const closed =\r\n      (inputData as SubjectInput).closed === undefined ? true : (inputData as SubjectInput).closed;\r\n\r\n    // is it a path or paths?\r\n    if (Array.isArray(pathOrPaths[0])) {\r\n      // paths\r\n      if (!clipper.addPaths(pathOrPaths as Paths, polyType, closed)) {\r\n        throw new ClipperError(\"invalid paths\");\r\n      }\r\n    } else {\r\n      // path\r\n      if (!clipper.addPath(pathOrPaths as Path, polyType, closed)) {\r\n        throw new ClipperError(\"invalid path\");\r\n      }\r\n    }\r\n  }\r\n};\r\n\r\nexport function clipToPathsOrPolyTree(\r\n  polyTreeMode: boolean,\r\n  nativeClipperLib: NativeClipperLibInstance,\r\n  params: ClipParams\r\n): Paths | PolyTree {\r\n  if (devMode) {\r\n    if (!polyTreeMode && params.subjectInputs && params.subjectInputs.some((si) => !si.closed)) {\r\n      throw new Error(\"clip to a PolyTree (not to a Path) when using open paths\");\r\n    }\r\n  }\r\n\r\n  const clipper = new Clipper(nativeClipperLib, params);\r\n\r\n  //noinspection UnusedCatchParameterJS\r\n  try {\r\n    addPathOrPaths(clipper, params.subjectInputs, PolyType.Subject);\r\n    addPathOrPaths(clipper, params.clipInputs, PolyType.Clip);\r\n    let result;\r\n    const clipFillType =\r\n      params.clipFillType === undefined ? params.subjectFillType : params.clipFillType;\r\n    if (!polyTreeMode) {\r\n      result = clipper.executeToPaths(\r\n        params.clipType,\r\n        params.subjectFillType,\r\n        clipFillType,\r\n        params.cleanDistance\r\n      );\r\n    } else {\r\n      if (params.cleanDistance !== undefined) {\r\n        throw new ClipperError(\"cleaning is not available for poly tree results\");\r\n      }\r\n      result = clipper.executeToPolyTee(params.clipType, params.subjectFillType, clipFillType);\r\n    }\r\n    if (result === undefined) {\r\n      throw new ClipperError(\"error while performing clipping task\");\r\n    }\r\n    return result;\r\n  } finally {\r\n    clipper.dispose();\r\n  }\r\n}\r\n\r\nexport function clipToPaths(nativeClipperLib: NativeClipperLibInstance, params: ClipParams): Paths {\r\n  return clipToPathsOrPolyTree(false, nativeClipperLib, params) as Paths;\r\n}\r\n\r\nexport function clipToPolyTree(\r\n  nativeClipperLib: NativeClipperLibInstance,\r\n  params: ClipParams\r\n): PolyTree {\r\n  return clipToPathsOrPolyTree(true, nativeClipperLib, params) as PolyTree;\r\n}\r\n"]}