mirror of
https://github.com/revarbat/BOSL2.git
synced 2025-01-16 13:50:23 +01:00
plane_intersection() bugfix. plane_line_intersection() and polygon_line_intersection() can now detect on-plane intersections.
This commit is contained in:
parent
9257c74236
commit
412dd9e260
@ -592,23 +592,19 @@ function plane3pt_indexed(points, i1, i2, i3) =
|
|||||||
// is returned as a list of two points on the line of intersection. If any of the input planes are parallel
|
// is returned as a list of two points on the line of intersection. If any of the input planes are parallel
|
||||||
// then returns undef.
|
// then returns undef.
|
||||||
function plane_intersection(plane1,plane2,plane3) =
|
function plane_intersection(plane1,plane2,plane3) =
|
||||||
is_def(plane3) ?
|
is_def(plane3)? let(
|
||||||
let (
|
matrix = [for(p=[plane1,plane2,plane3]) select(p,0,2)],
|
||||||
matrix = [for(p=[plane1,plane2,plane3]) select(p,0,2)],
|
rhs = [for(p=[plane1,plane2,plane3]) p[3]]
|
||||||
rhs = [for(p=[plane1,plane2,plane3]) p[3]]
|
) linear_solve(matrix,rhs) :
|
||||||
)
|
let(
|
||||||
linear_solve(matrix,rhs)
|
normal = cross(plane_normal(plane1), plane_normal(plane2))
|
||||||
:
|
) approx(norm(normal),0) ? undef :
|
||||||
let(
|
let(
|
||||||
normal = cross(plane_normal(plane1), plane_normal(plane2))
|
matrix = [for(p=[plane1,plane2]) select(p,0,2)],
|
||||||
)
|
rhs = [for(p=[plane1,plane2]) p[3]],
|
||||||
approx(normal,0) ? undef :
|
point = linear_solve(matrix,rhs)
|
||||||
let(
|
) is_undef(point)? undef :
|
||||||
matrix = [for(p=[plane1,plane2]) select(p,0,2)],
|
[point, point+normal];
|
||||||
rhs = [for(p=[plane1,plane2]) p[3]],
|
|
||||||
point = linear_solve(matrix,rhs)
|
|
||||||
)
|
|
||||||
[point, point+normal];
|
|
||||||
|
|
||||||
|
|
||||||
// Function: plane_from_normal()
|
// Function: plane_from_normal()
|
||||||
@ -653,8 +649,8 @@ function plane_from_pointslist(points, fast=false, eps=EPSILON) =
|
|||||||
// Usage:
|
// Usage:
|
||||||
// plane_normal(plane);
|
// plane_normal(plane);
|
||||||
// Description:
|
// Description:
|
||||||
// Returns the normal vector for the given plane.
|
// Returns the unit length normal vector for the given plane.
|
||||||
function plane_normal(plane) = [for (i=[0:2]) plane[i]];
|
function plane_normal(plane) = unit([for (i=[0:2]) plane[i]]);
|
||||||
|
|
||||||
|
|
||||||
// Function: distance_from_plane()
|
// Function: distance_from_plane()
|
||||||
@ -690,6 +686,9 @@ function closest_point_on_plane(plane, point) =
|
|||||||
) point - n*d;
|
) point - n*d;
|
||||||
|
|
||||||
|
|
||||||
|
// Returns [POINT, U] if line intersects plane at one point.
|
||||||
|
// Returns [LINE, undef] if the line is on the plane.
|
||||||
|
// Returns undef if line is parallel to, but not on the given plane.
|
||||||
function _general_plane_line_intersection(plane, line, eps=EPSILON) =
|
function _general_plane_line_intersection(plane, line, eps=EPSILON) =
|
||||||
let(
|
let(
|
||||||
p0 = line[0],
|
p0 = line[0],
|
||||||
@ -698,6 +697,7 @@ function _general_plane_line_intersection(plane, line, eps=EPSILON) =
|
|||||||
u = p1 - p0,
|
u = p1 - p0,
|
||||||
d = n * u
|
d = n * u
|
||||||
) abs(d)<eps? (
|
) abs(d)<eps? (
|
||||||
|
coplanar(plane, p0)? [line,undef] : // Line on plane
|
||||||
undef // Line parallel to plane
|
undef // Line parallel to plane
|
||||||
) : let(
|
) : let(
|
||||||
v0 = closest_point_on_plane(plane, [0,0,0]),
|
v0 = closest_point_on_plane(plane, [0,0,0]),
|
||||||
@ -726,20 +726,24 @@ function plane_line_angle(plane, line) =
|
|||||||
// pt = plane_line_intersection(plane, line, [eps]);
|
// pt = plane_line_intersection(plane, line, [eps]);
|
||||||
// Description:
|
// Description:
|
||||||
// Takes a line, and a plane [A,B,C,D] where the equation of that plane is `Ax+By+Cz=D`.
|
// Takes a line, and a plane [A,B,C,D] where the equation of that plane is `Ax+By+Cz=D`.
|
||||||
// Returns the coordinates of the where the given `line` intersects the given `plane`.
|
// If `line` intersects `plane` at one point, then that intersection point is returned.
|
||||||
// Returns `undef` if the line is parallel to the plane.
|
// If `line` lies on `plane`, then the original given `line` is returned.
|
||||||
|
// If `line` is parallel to, but not on `plane`, then `undef` is returned.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// plane = The [A,B,C,D] values for the equation of the plane.
|
// plane = The [A,B,C,D] values for the equation of the plane.
|
||||||
// line = A list of two 3D points that are on the line.
|
// line = A list of two 3D points that are on the line.
|
||||||
// bounded = If false, the line is considered unbounded. If true, it is treated as a bounded line segment. If given as `[true, false]` or `[false, true]`, the boundedness of the points are specified individually, allowing the line to be treated as a half-bounded ray. Default: false (unbounded)
|
// bounded = If false, the line is considered unbounded. If true, it is treated as a bounded line segment. If given as `[true, false]` or `[false, true]`, the boundedness of the points are specified individually, allowing the line to be treated as a half-bounded ray. Default: false (unbounded)
|
||||||
// eps = The epsilon error value to determine whether the line is too close to parallel to the plane. Default: `EPSILON` (1e-9)
|
// eps = The epsilon error value to determine whether the line is too close to parallel to the plane. Default: `EPSILON` (1e-9)
|
||||||
function plane_line_intersection(plane, line, bounded=false, eps=EPSILON) =
|
function plane_line_intersection(plane, line, bounded=false, eps=EPSILON) =
|
||||||
assert(is_vector(plane)&&len(plane)==4)
|
assert(is_vector(plane)&&len(plane)==4, "Invalid plane value.")
|
||||||
assert(is_path(line)&&len(line)==2)
|
assert(is_path(line)&&len(line)==2, "Invalid line value.")
|
||||||
|
assert(!approx(line[0],line[1]), "The two points defining the line must not be the same point.")
|
||||||
let(
|
let(
|
||||||
bounded = is_list(bounded)? bounded : [bounded, bounded],
|
bounded = is_list(bounded)? bounded : [bounded, bounded],
|
||||||
res = _general_plane_line_intersection(plane, line, eps=eps)
|
res = _general_plane_line_intersection(plane, line, eps=eps)
|
||||||
)
|
)
|
||||||
|
is_undef(res)? undef :
|
||||||
|
is_undef(res[1])? res[0] :
|
||||||
bounded[0]&&res[1]<0? undef :
|
bounded[0]&&res[1]<0? undef :
|
||||||
bounded[1]&&res[1]>1? undef :
|
bounded[1]&&res[1]>1? undef :
|
||||||
res[0];
|
res[0];
|
||||||
@ -750,7 +754,10 @@ function plane_line_intersection(plane, line, bounded=false, eps=EPSILON) =
|
|||||||
// pt = polygon_line_intersection(poly, line, [bounded], [eps]);
|
// pt = polygon_line_intersection(poly, line, [bounded], [eps]);
|
||||||
// Description:
|
// Description:
|
||||||
// Takes a possibly bounded line, and a 3D planar polygon, and finds their intersection point.
|
// Takes a possibly bounded line, and a 3D planar polygon, and finds their intersection point.
|
||||||
// Returns the 3D coordinates of the intersection point, or `undef` if they do not intersect.
|
// If the line is on the plane as the polygon, and intersects, then a list of 3D line
|
||||||
|
// segments is returned, one for each section of the line that is inside the polygon.
|
||||||
|
// If the line is not on the plane of the polygon, but intersects, then the 3D intersection
|
||||||
|
// point is returned. If the line does not intersect the polygon, then `undef` is returned.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// poly = The 3D planar polygon to find the intersection with.
|
// poly = The 3D planar polygon to find the intersection with.
|
||||||
// line = A list of two 3D points that are on the line.
|
// line = A list of two 3D points that are on the line.
|
||||||
@ -770,6 +777,27 @@ function polygon_line_intersection(poly, line, bounded=false, eps=EPSILON) =
|
|||||||
res = _general_plane_line_intersection(plane, line, eps=eps)
|
res = _general_plane_line_intersection(plane, line, eps=eps)
|
||||||
)
|
)
|
||||||
is_undef(res)? undef :
|
is_undef(res)? undef :
|
||||||
|
is_undef(res[1])? (
|
||||||
|
let(
|
||||||
|
// Line is on polygon plane.
|
||||||
|
linevec = unit(line[1] - line[0]),
|
||||||
|
lp1 = line[0] + (bounded[0]? 0 : -1000000) * linevec,
|
||||||
|
lp2 = line[1] + (bounded[1]? 0 : 1000000) * linevec,
|
||||||
|
poly2d = clockwise_polygon(project_plane(poly, p1, p2, p3)),
|
||||||
|
line2d = project_plane([lp1,lp2], p1, p2, p3),
|
||||||
|
parts = split_path_at_region_crossings(line2d, [poly2d], closed=false),
|
||||||
|
inside = [
|
||||||
|
for (part = parts)
|
||||||
|
if (point_in_polygon(mean(part), poly2d)>0) part
|
||||||
|
]
|
||||||
|
) !inside? undef :
|
||||||
|
let(
|
||||||
|
isegs = [
|
||||||
|
for (seg = inside)
|
||||||
|
lift_plane(seg, p1, p2, p3)
|
||||||
|
]
|
||||||
|
) isegs
|
||||||
|
) :
|
||||||
bounded[0]&&res[1]<0? undef :
|
bounded[0]&&res[1]<0? undef :
|
||||||
bounded[1]&&res[1]>1? undef :
|
bounded[1]&&res[1]>1? undef :
|
||||||
let(
|
let(
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
//////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
BOSL_VERSION = [2,0,196];
|
BOSL_VERSION = [2,0,197];
|
||||||
|
|
||||||
|
|
||||||
// Section: BOSL Library Version Functions
|
// Section: BOSL Library Version Functions
|
||||||
|
Loading…
x
Reference in New Issue
Block a user