point_in_polygon optimization

This commit is contained in:
Adrian Mariano 2021-11-01 17:34:18 -04:00
parent 3a367c3faf
commit 7bac9a484d

View file

@ -1583,6 +1583,7 @@ function _point_above_below_segment(point, edge) =
? (edge[1].y > 0 && cross(edge[0], edge[1]-edge[0]) > 0) ? 1 : 0 ? (edge[1].y > 0 && cross(edge[0], edge[1]-edge[0]) > 0) ? 1 : 0
: (edge[1].y <= 0 && cross(edge[0], edge[1]-edge[0]) < 0) ? -1 : 0; : (edge[1].y <= 0 && cross(edge[0], edge[1]-edge[0]) < 0) ? -1 : 0;
function point_in_polygon(point, poly, nonzero=false, eps=EPSILON) = function point_in_polygon(point, poly, nonzero=false, eps=EPSILON) =
// Original algorithms from http://geomalgorithms.com/a03-_inclusion.html // Original algorithms from http://geomalgorithms.com/a03-_inclusion.html
assert( is_vector(point,2) && is_path(poly,dim=2) && len(poly)>2, assert( is_vector(point,2) && is_path(poly,dim=2) && len(poly)>2,
@ -1597,39 +1598,44 @@ function point_in_polygon(point, poly, nonzero=false, eps=EPSILON) =
: :
// Does the point lie on any edges? If so return 0. // Does the point lie on any edges? If so return 0.
let( let(
on_brd = [ segs = pair(poly,true),
for (i = [0:1:len(poly)-1]) on_border = [for (seg=segs)
let( seg = select(poly,i,i+1) ) if (norm(seg[0]-seg[1])>eps && _is_point_on_line(point, seg, SEGMENT, eps=eps)) 1]
if (!approx(seg[0],seg[1],eps) )
_is_point_on_line(point, seg, SEGMENT, eps=eps)? 1:0
]
) )
sum(on_brd) > 0? 0 : on_border != [] ? 0 :
nonzero nonzero // Compute winding number and return 1 for interior, -1 for exterior
? // Compute winding number and return 1 for interior, -1 for exterior ? let(
let( winding = [
windchk = [ for(seg=segs)
for(i=[0:1:len(poly)-1]) let(
let( seg=select(poly,i,i+1) ) p0=seg[0]-point,
if (!approx(seg[0],seg[1],eps=eps)) p1=seg[1]-point
_point_above_below_segment(point, seg) )
if (norm(p0-p1)>eps)
p0.y <=0
? p1.y > 0 && cross(p0,p1-p0)>0 ? 1 : 0
: p1.y <=0 && cross(p0,p1-p0)<0 ? -1: 0
] ]
) sum(windchk) != 0 ? 1 : -1 )
sum(winding) != 0 ? 1 : -1
: // or compute the crossings with the ray [point, point+[1,0]] : // or compute the crossings with the ray [point, point+[1,0]]
let( let(
n = len(poly),
cross = [ cross = [
for(i=[0:n-1]) for(seg=segs)
let( let(
p0 = poly[i]-point, p0 = seg[0]-point,
p1 = poly[(i+1)%n]-point p1 = seg[1]-point
) )
if ( if (
( (p1.y>eps && p0.y<=eps) || (p1.y<=eps && p0.y>eps) ) ( (p1.y>eps && p0.y<=eps) || (p1.y<=eps && p0.y>eps) )
&& -eps < p0.x - p0.y *(p1.x - p0.x)/(p1.y - p0.y) && -eps < p0.x - p0.y *(p1.x - p0.x)/(p1.y - p0.y)
) 1 )
1
] ]
) 2*(len(cross)%2)-1; )
2*(len(cross)%2)-1;
// Function: polygon_triangulate() // Function: polygon_triangulate()