UNPKG

@xtor/cga.js

Version:

Xtor Compute Geometry Algorithm Libary 计算几何算法库

333 lines (291 loc) 10.2 kB
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); }