mirror of
https://github.com/Doodle3D/Doodle3D-Slicer.git
synced 2025-04-03 10:23:23 +02:00
133 lines
4.0 KiB
JavaScript
133 lines
4.0 KiB
JavaScript
import Shape from 'clipper-js';
|
|
import { subtract, add, scale, normalize, dot, length, distanceTo } from './VectorUtils.js';
|
|
import { PRECISION } from '../../constants.js';
|
|
|
|
const TOLERANCE = 5 / PRECISION;
|
|
|
|
export default function comb(outline, start, end) {
|
|
if (distanceTo(start, end) < TOLERANCE) {
|
|
return [start, end];
|
|
}
|
|
|
|
let combPath = new Shape([[start, end]], false, true, false);
|
|
|
|
for (let i = 0; i < outline.paths.length; i ++) {
|
|
let outlinePart = new Shape([outline.paths[i]], true, false, false, true);
|
|
|
|
let snappedCombPaths = i === 0 ? combPath.intersect(outlinePart) : combPath.difference(outlinePart);
|
|
|
|
snappedCombPaths = snappedCombPaths.mapToLower();
|
|
outlinePart = outlinePart.mapToLower()[0];
|
|
|
|
if (distanceTo(start, outlinePart[outlinePart.length - 1]) < distanceTo(start, outlinePart[0])) {
|
|
outlinePart = outlinePart.reverse();
|
|
}
|
|
|
|
const distanceMap = new WeakMap();
|
|
|
|
for (let i = 0; i < snappedCombPaths.length; i ++) {
|
|
const snappedCombPath = snappedCombPaths[i];
|
|
|
|
const distanceStart = distanceTo(start, snappedCombPath[0]);
|
|
const distanceEnd = distanceTo(start, snappedCombPath[snappedCombPath.length - 1]);
|
|
|
|
if (distanceStart < distanceEnd) {
|
|
distanceMap.set(snappedCombPath, distanceStart);
|
|
} else {
|
|
snappedCombPath.reverse();
|
|
distanceMap.set(snappedCombPath, distanceEnd);
|
|
}
|
|
}
|
|
snappedCombPaths.sort((a, b) => distanceMap.get(a) - distanceMap.get(b));
|
|
|
|
const firstPath = snappedCombPaths[0];
|
|
const lastPath = snappedCombPaths[snappedCombPaths.length - 1];
|
|
|
|
if (snappedCombPaths.length === 0) {
|
|
continue;
|
|
// snappedCombPaths.push([start], [end]);
|
|
} else if (distanceTo(firstPath[0], start) > 1.0) {
|
|
snappedCombPaths.unshift([start]);
|
|
} else if (distanceTo(lastPath[lastPath.length - 1], end) > 1.0) {
|
|
snappedCombPaths.push([end]);
|
|
}
|
|
|
|
if (snappedCombPaths.length === 1) {
|
|
continue;
|
|
}
|
|
|
|
const startPath = snappedCombPaths[0];
|
|
const startPoint = startPath[startPath.length - 1];
|
|
|
|
const endPath = snappedCombPaths[snappedCombPaths.length - 1];
|
|
const endPoint = endPath[0];
|
|
|
|
const lineIndexStart = findClosestLineOnPath(outlinePart, startPoint);
|
|
const lineIndexEnd = findClosestLineOnPath(outlinePart, endPoint);
|
|
|
|
const path = [];
|
|
if (lineIndexEnd === lineIndexStart) {
|
|
continue;
|
|
} else if (lineIndexEnd > lineIndexStart) {
|
|
if (lineIndexStart + outlinePart.length - lineIndexEnd < lineIndexEnd - lineIndexStart) {
|
|
for (let i = lineIndexStart + outlinePart.length; i > lineIndexEnd; i --) {
|
|
path.push(outlinePart[i % outlinePart.length]);
|
|
}
|
|
} else {
|
|
for (let i = lineIndexStart; i < lineIndexEnd; i ++) {
|
|
path.push(outlinePart[i + 1]);
|
|
}
|
|
}
|
|
} else {
|
|
if (lineIndexEnd + outlinePart.length - lineIndexStart < lineIndexStart - lineIndexEnd) {
|
|
for (let i = lineIndexStart; i < lineIndexEnd + outlinePart.length; i ++) {
|
|
path.push(outlinePart[(i + 1) % outlinePart.length]);
|
|
}
|
|
} else {
|
|
for (let i = lineIndexStart; i > lineIndexEnd; i --) {
|
|
path.push(outlinePart[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
combPath = new Shape([[...startPath, ...path, ...endPath]], false, true, false, true);
|
|
}
|
|
|
|
return combPath.mapToLower()[0];
|
|
}
|
|
|
|
function findClosestLineOnPath(path, point) {
|
|
let distance = Infinity;
|
|
let lineIndex;
|
|
|
|
for (let i = 0; i < path.length; i ++) {
|
|
const pointA = path[i];
|
|
const pointB = path[(i + 1) % path.length];
|
|
|
|
const tempClosestPoint = findClosestPointOnLine(pointA, pointB, point);
|
|
const tempDistance = distanceTo(tempClosestPoint, point);
|
|
|
|
if (tempDistance < distance) {
|
|
distance = tempDistance;
|
|
lineIndex = i;
|
|
}
|
|
}
|
|
|
|
return lineIndex;
|
|
}
|
|
|
|
function findClosestPointOnLine(a, b, c) {
|
|
const b_ = subtract(b, a);
|
|
const c_ = subtract(c, a);
|
|
|
|
const lambda = dot(normalize(b_), c_) / length(b_);
|
|
|
|
if (lambda >= 1) {
|
|
return b;
|
|
} else if (lambda > 0) {
|
|
return add(a, scale(b_, lambda));
|
|
} else {
|
|
return a;
|
|
}
|
|
}
|