mirror of
https://github.com/Doodle3D/Doodle3D-Slicer.git
synced 2025-04-04 10:53:40 +02:00
when a ‘single walled’, ‘closed’ shape is sliced the shape will appear to be closed by the slicer. This happens because the start and endpoint of the 2d shape are connected. This commits fixes this. In the new approach al parts of the geometry are split up into shapes. Then the 3d shapes are places into two categories (closed and open geometries). Based on weather the 3d geometry is open or closed the 2d shape will be open or closed @mith @peteruithoven
124 lines
3.6 KiB
JavaScript
124 lines
3.6 KiB
JavaScript
import * as THREE from 'three.js';
|
|
import Shape from 'Doodle3D/clipper-js';
|
|
|
|
export default function intersectionsToShapes(layerIntersectionIndexes, layerIntersectionPoints, lines, settings) {
|
|
console.log('generating slices');
|
|
|
|
const layers = [];
|
|
|
|
for (let layer = 1; layer < layerIntersectionIndexes.length; layer ++) {
|
|
const intersectionIndexes = layerIntersectionIndexes[layer];
|
|
const intersectionPoints = layerIntersectionPoints[layer];
|
|
|
|
if (intersectionIndexes.length === 0) continue;
|
|
|
|
const closedShapes = [];
|
|
const openShapes = [];
|
|
for (let i = 0; i < intersectionIndexes.length; i ++) {
|
|
let index = intersectionIndexes[i];
|
|
|
|
if (intersectionPoints[index] === undefined) continue;
|
|
|
|
const shape = [];
|
|
|
|
const firstPoints = [index];
|
|
const { open: openGeometry } = lines[index];
|
|
let isFirstPoint = true;
|
|
let openShape = true;
|
|
|
|
while (index !== -1) {
|
|
const intersection = intersectionPoints[index];
|
|
// uppercase X and Y because clipper vector
|
|
shape.push(intersection);
|
|
|
|
delete intersectionPoints[index];
|
|
|
|
const connects = lines[index].connects;
|
|
const faceNormals = lines[index].normals;
|
|
|
|
for (let i = 0; i < connects.length; i ++) {
|
|
index = connects[i];
|
|
|
|
if (firstPoints.includes(index) && shape.length > 2) {
|
|
openShape = false;
|
|
index = -1;
|
|
break;
|
|
}
|
|
|
|
// Check if index has an intersection or is already used
|
|
if (intersectionPoints[index] !== undefined) {
|
|
const faceNormal = faceNormals[Math.floor(i / 2)];
|
|
|
|
const a = new THREE.Vector2(intersection.x, intersection.y);
|
|
const b = new THREE.Vector2(intersectionPoints[index].x, intersectionPoints[index].y);
|
|
|
|
// can't calculate normal between points if distance is smaller as 0.0001
|
|
if ((faceNormal.x === 0 && faceNormal.y === 0) || a.distanceTo(b) < 0.0001) {
|
|
if (isFirstPoint) {
|
|
firstPoints.push(index);
|
|
}
|
|
|
|
delete intersectionPoints[index];
|
|
|
|
connects.push(...lines[index].connects);
|
|
faceNormals.push(...lines[index].normals);
|
|
index = -1;
|
|
} else {
|
|
// make sure the path goes the right direction
|
|
// THREE.Vector2.normal is not yet implimented
|
|
// const normal = a.sub(b).normal().normalize();
|
|
const normal = a.sub(b);
|
|
normal.set(-normal.y, normal.x).normalize();
|
|
|
|
if (normal.dot(faceNormal) > 0) {
|
|
break;
|
|
} else {
|
|
index = -1;
|
|
}
|
|
}
|
|
} else {
|
|
index = -1;
|
|
}
|
|
}
|
|
isFirstPoint = false;
|
|
}
|
|
|
|
if (openShape) {
|
|
index = firstPoints[0];
|
|
|
|
while (index !== -1) {
|
|
if (firstPoints.indexOf(index) === -1) {
|
|
const intersection = intersectionPoints[index];
|
|
shape.unshift(intersection);
|
|
|
|
delete intersectionPoints[index];
|
|
}
|
|
|
|
const connects = lines[index].connects;
|
|
|
|
for (let i = 0; i < connects.length; i ++) {
|
|
index = connects[i];
|
|
|
|
if (intersectionPoints[index] !== undefined) {
|
|
break;
|
|
} else {
|
|
index = -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (openGeometry) {
|
|
if (!openShape) shape.push(shape[0].clone());
|
|
openShapes.push(shape);
|
|
} else {
|
|
closedShapes.push(shape);
|
|
}
|
|
}
|
|
|
|
layers.push({ closedShapes, openShapes });
|
|
}
|
|
|
|
return layers;
|
|
}
|