@xtor/cga.js
Version:
Xtor Compute Geometry Algorithm Libary 计算几何算法库
333 lines (291 loc) • 10.2 kB
JavaScript
import { v3, Vector3 } from "../../math/Vector3";
import { gPrecision } from "../../math/Math";
import { Segment } from "./Segment";
import { Orientation } from "./type";
export class Line {
constructor(origin, end) {
this.origin = origin !== undefined ? origin : v3();
this.end = end !== undefined ? end : v3();
this.direction = this.end
.clone()
.sub(this.origin)
.normalize();
}
distancePoint(point) {
return point.distanceLine(this);
}
//---距离-------------
/**
* 直线到直线的距离
* 参数与最近点顺序一致
* @param {Line} line
*/
distanceLine(line) {
var result = {
parameters: [],
closests: []
};
var diff = this.origin.clone().sub(line.origin);
var a01 = -this.direction.dot(line.direction);
var b0 = diff.dot(this.direction);
var s0, s1;
if (Math.abs(a01) < 1) {
var det = 1 - a01 * a01;
var b1 = -diff.dot(line.direction);
s0 = (a01 * b1 - b0) / det;
s1 = (a01 * b0 - b1) / det;
} else {
s0 = -b0;
s1 = 0;
}
result.parameters[0] = s0;
result.parameters[1] = s1;
result.closests[0] = this.direction
.clone()
.multiplyScalar(s0)
.add(this.origin);
result.closests[1] = line.direction
.clone()
.multiplyScalar(s1)
.add(line.origin);
diff = result.closests[0].clone().sub(result.closests[1]);
result.sqrDistance = diff.dot(diff);
result.distance = Math.sqrt(result.sqrDistance);
return result;
}
/**
* 直线与射线的距离
* @param {Ray} ray
*/
distanceRay(ray) {
const result = {
parameters: [],
closests: [],
sqrDistance: 0,
distance: 0
};
var diff = this.origin.clone().sub(ray.origin);
var a01 = - this.direction.dot(ray.direction);
var b0 = diff.dot(this.direction);
var s0, s1;
if (Math.abs(a01) < 1) {
var b1 = -diff.dot(ray.direction);
s1 = a01 * b0 - b1;
if (s1 >= 0) {
//在最近点在射线上,相当于直线与直线最短距离
var det = 1 - a01 * a01;
s0 = (a01 * b1 - b0) / det;
s1 /= det;
}
else {
// 射线的起始点是离直线的最近点
s0 = -b0;
s1 = 0;
}
} else {
s0 = -b0;
s1 = 0;
}
result.parameters[0] = s0;
result.parameters[1] = s1;
result.closests[0] = this.direction.clone().multiplyScalar(s0).add(this.origin);
result.closests[1] = ray.direction.clone().multiplyScalar(s1).add(ray.origin);
diff = result.closests[0].clone().sub(result.closests[1]);
result.sqrDistance = diff.dot(diff);
result.distance = Math.sqrt(result.sqrDistance);
return result;
}
/**
* 直线与线段的距离
* @param {Segment} segment
*/
distanceSegment(segment) {
var result = {
parameters: [],
closests: []
};
var segCenter = segment.center;
var segDirection = segment.direction;
var segExtent = segment.len * 0.5;
var diff = this.origin.clone().sub(segCenter);
var a01 = - this.direction.dot(segDirection);
var b0 = diff.dot(this.direction);
var s0, s1;
if (Math.abs(a01) < 1) {
// 判断是否平行
var det = 1 - a01 * a01;
var extDet = segExtent * det;
var b1 = -diff.dot(segDirection);
s1 = a01 * b0 - b1;
if (s1 >= -extDet) {
if (s1 <= extDet) {
// Two interior points are closest, one on the this
// and one on the segment.
s0 = (a01 * b1 - b0) / det;
s1 /= det;
}
else {
// The endpoint e1 of the segment and an interior
// point of the this are closest.
s1 = segExtent;
s0 = -(a01 * s1 + b0);
}
}
else {
// The endpoint e0 of the segment and an interior point
// of the this are closest.
s1 = -segExtent;
s0 = -(a01 * s1 + b0);
}
}
else {
// The this and segment are parallel. Choose the closest pair
// so that one point is at segment origin.
s1 = 0;
s0 = -b0;
}
result.parameters[0] = s0;
result.parameters[1] = s1;
result.closests[0] = this.direction.clone().multiplyScalar(s0).add(this.origin);
result.closests[1] = segDirection.clone().multiplyScalar(s1).add(segCenter);
diff = result.closests[0].clone().sub(result.closests[1]);
result.sqrDistance = diff.dot(diff);
result.distance = Math.sqrt(result.sqrDistance);
return result;
}
distancePlane(plane) {
}
distanceTriangle(triangle) {
function Orthonormalize(numInputs, v, robust = false) {
if (v && 1 <= numInputs && numInputs <= 3) {
var minLength = v[0].length();
v[0].normalize();
for (var i = 1; i < numInputs; ++i) {
for (var j = 0; j < i; ++j) {
var dot = v[i].dot(v[j]);
v[i].sub(v[j].clone().multiplyScalar(dot));
}
var length = v[i].length();
v[i].normalize();
if (length < minLength) {
minLength = length;
}
}
return minLength;
}
return 0;
}
function ComputeOrthogonalComplement(numInputs, v, robust = false) {
if (numInputs === 1) {
if (Math.abs(v[0][0]) > Math.abs(v[0][1])) {
v[1] = v3(- v[0].z, 0, +v[0].x)
}
else {
v[1] = v3(0, + v[0].z, -v[0].y)
};
numInputs = 2;
}
if (numInputs == 2) {
v[2] = v[0].clone().cross(v[1]);
return Orthonormalize(3, v, robust);
}
return 0;
}
const result = {
closests: [],
parameters: [],
triangleParameter: [],
};
// Test if line intersects triangle. If so, the squared distance
// is zero.
var edge0 = triangle.p1.clone().sub(triangle.p0);
var edge1 = triangle.p2.clone().sub(triangle.p0);
var normal = edge0.clone().cross(edge1).normalize();
var NdD = normal.dot(this.direction);
if (Math.abs(NdD) >= gPrecision) {
// The line and triangle are not parallel, so the line
// intersects/ the plane of the triangle.
var diff = this.origin.clone().sub(triangle.p0);
var basis = new Array(3); // {D, U, V}
basis[0] = this.direction;
ComputeOrthogonalComplement(1, basis);
var UdE0 = basis[1].dot(edge0);
var UdE1 = basis[1].dot(edge1);
var UdDiff = basis[1].dot(diff);
var VdE0 = basis[2].dot(edge0);
var VdE1 = basis[2].dot(edge1);
var VdDiff = basis[2].dot(diff);
var invDet = 1 / (UdE0 * VdE1 - UdE1 * VdE0);
// Barycentric coordinates for the point of intersection.
var b1 = (VdE1 * UdDiff - UdE1 * VdDiff) * invDet;
var b2 = (UdE0 * VdDiff - VdE0 * UdDiff) * invDet;
var b0 = 1 - b1 - b2;
if (b0 >= 0 && b1 >= 0 && b2 >= 0) {
// Line parameter for the point of intersection.
var DdE0 = this.direction.dot(edge0);
var DdE1 = this.direction.dot(edge1);
var DdDiff = this.direction.dot(diff);
result.lineParameter = b1 * DdE0 + b2 * DdE1 - DdDiff;
// Barycentric coordinates for the point of intersection.
result.triangleParameter[0] = b0;
result.triangleParameter[1] = b1;
result.triangleParameter[2] = b2;
// The intersection point is inside or on the triangle.
result.closests[0] = this.direction.clone().multiplyScalar(result.lineParameter).add(this.origin);
result.closests[1] = edge0.multiplyScalar(b1).add(edge1.multiplyScalar(b2)).add(triangle.p0);
result.distance = 0;
result.sqrDistance = 0;
return result;
}
}
// Either (1) the line is not parallel to the triangle and the
// point of intersection of the line and the plane of the triangle
// is outside the triangle or (2) the line and triangle are
// parallel. Regardless, the closest point on the triangle is on
// an edge of the triangle. Compare the line to all three edges
// of the triangle.
result.distance = +Infinity;
result.sqrDistance = +Infinity;
for (var i0 = 2, i1 = 0; i1 < 3; i0 = i1++) {
var segCenter = triangle[i0].clone().add(triangle[i1]).multiplyScalar(0.5);
var segDirection = triangle[i1].clone().sub(triangle[i0]);
var segExtent = 0.5 * segDirection.length();
segDirection.normalize();
var segment = new Segment(triangle[i0], triangle[i1]);
var lsResult = this.distanceSegment(segment);
if (lsResult.sqrDistance < result.sqrDistance) {
result.sqrDistance = lsResult.sqrDistance;
result.distance = lsResult.distance;
result.lineParameter = lsResult.parameters[0];
result.triangleParameter[i0] = 0.5 * (1 -
lsResult.parameters[0] / segExtent);
result.triangleParameter[i1] = 1 -
result.triangleParameter[i0];
result.triangleParameter[3 - i0 - i1] = 0;
result.closests[0] = lsResult.closests[0];
result.closests[1] = lsResult.closests[1];
}
}
return result;
}
//---相交-------------
intersectLine(line) {
}
//---平行-------------
parallelLine(line) { }
//---方位---------------------
orientationPoint(point, normal = Vector3.UnitY) {
var binormal = this.direction.clone().cross(normal);
if (this.distancePoint(point).distance < gPrecision)
return Orientation.Common;
return point.clone().sub(this.origin).dot(binormal) > 0 ? Orientation.Positive : Orientation.Negative;
}
orientationSegment(segment, normal = Vector3.UnitY) {
var or0 = this.orientationPoint(segment.p0, normal);
var or1 = this.orientationPoint(segment.p1, normal);
return or0 | or1;
}
}
export function line(start, end) {
return new Line(start, end);
}