How to get the nearest point outside a polygon from a point inside a polygon?

后端 未结 2 1707
夕颜
夕颜 2021-01-12 10:21

I have a map with a lot of polygons and a point inside in one of them, like this: \"polygons\"

The x and y coordina

相关标签:
2条回答
  • 2021-01-12 10:29

    To know in which polygon the point is in you can use the Ray Casting algorithm.

    For finding the closest edge you could use a naive approach computing the distance to each edge and then take the minimum. Just remember to check if the intersection with edge is outside the edge (check this). If it is outside take the distance to the closest extreme of the edge.

    Ok better explaining the intuition with some pseudo-code:

    dot(u,v) --> ((u).x * (v).x + (u).y * (v).y)
    norm(v)  --> sqrt(dot(v,v))     // norm = length of vector
    dist(u,v)--> norm(u-v)          // distance = norm of difference
    
    // Vector contains x and y
    // Point contains x and y
    // Segment contains P0 and P1 of type Point
    // Point  = Point ± Vector
    // Vector = Point - Point
    // Vector = Scalar * Vector
    Point closest_Point_in_Segment_to(Point P, Segment S)
    {
         Vector v = S.P1 - S.P0;
         Vector w = P - S.P0;
    
         double c1 = dot(w,v);
         if ( c1 <= 0 )   // the closest point is outside the segment and nearer to P0
              return S.P0;
    
         double c2 = dot(v,v);
         if ( c2 <= c1 )  // the closest point is outside the segment and nearer to P1
              return S.P1;
    
         double b = c1 / c2;
         Point Pb = S.P0 + b * v;
         return Pb;
    }
    
    [Point, Segment] get_closest_border_point_to(Point point, Polygon poly) {
    
        double bestDistance = MAX_DOUBLE;
        Segment bestSegment;
        Point bestPoint;
    
        foreach s in poly.segments {
            Point closestInS = closest_Point_in_Segment_to(point, s);
            double d = dist(point, closestInS);
            if (d < bestDistance) {
                bestDistance = d;
                bestSegment = s;
                bestPoint = closestInS; 
            }
        }
    
        return [bestPoint, bestSegment];
    }
    

    I think this pseudo-code should get you going, of course once you have the polygon your point is in!.

    0 讨论(0)
  • 2021-01-12 10:39

    My PolyCollisions class:

    public class PolyCollisions {
    
        // Call this function...
        public static Vector2 doCollisions (Vector2[] polygon, Vector2 point) {
    
            if(!pointIsInPoly(polygon, point)) {
                // The point is not colliding with the polygon, so it does not need to change location
                return point;
            }
    
            // Get the closest point of the polygon
            return closestPointOutsidePolygon(polygon, point);
    
        }
    
        // Check if the given point is within the given polygon (Vertexes)
        // 
        // If so, call on collision if required, and move the point to the
        // closest point outside of the polygon
        public static boolean pointIsInPoly(Vector2[] verts, Vector2 p) {
            int nvert = verts.length;
            double[] vertx = new double[nvert];
            double[] verty = new double[nvert];
            for(int i = 0; i < nvert; i++) {
                Vector2 vert = verts[i];
                vertx[i] = vert.x;
                verty[i] = vert.y;
            }
            double testx = p.x;
            double testy = p.y;
            int i, j;
            boolean c = false;
            for (i = 0, j = nvert-1; i < nvert; j = i++) {
                if ( ((verty[i]>testy) != (verty[j]>testy)) &&
                        (testx < (vertx[j]-vertx[i]) * (testy-verty[i]) / (verty[j]-verty[i]) + vertx[i]) )
                    c = !c;
            }
            return c;
        }
    
        // Gets the closed point that isn't inside the polygon...
        public static Vector2 closestPointOutsidePolygon (Vector2[] poly, Vector2 point) {
    
            return getClosestPointInSegment(closestSegment(poly, point), point);
    
        }
    
        public static Vector2 getClosestPointInSegment (Vector2[] segment, Vector2 point) {
    
            return newPointFromCollision(segment[0], segment[1], point);
    
        }
    
        public static Vector2 newPointFromCollision (Vector2 aLine, Vector2 bLine, Vector2 p) {
    
            return nearestPointOnLine(aLine.x, aLine.y, bLine.x, bLine.y, p.x, p.y);
    
        }
    
        public static Vector2 nearestPointOnLine(double ax, double ay, double bx, double by, double px, double py) {
    
            // https://stackoverflow.com/questions/1459368/snap-point-to-a-line-java
    
            double apx = px - ax;
            double apy = py - ay;
            double abx = bx - ax;
            double aby = by - ay;
    
            double ab2 = abx * abx + aby * aby;
            double ap_ab = apx * abx + apy * aby;
            double t = ap_ab / ab2;
            if (t < 0) {
                t = 0;
            } else if (t > 1) {
                t = 1;
            }
            return new Vector2(ax + abx * t, ay + aby * t);
        }
    
        public static Vector2[] closestSegment (Vector2[] points, Vector2 point) {
    
            Vector2[] returns = new Vector2[2];
    
            int index = closestPointIndex(points, point);
    
            returns[0] = points[index];
    
            Vector2[] neighbors = new Vector2[] {
                    points[(index+1+points.length)%points.length],
                    points[(index-1+points.length)%points.length]
            };
    
            double[] neighborAngles = new double[] {
                    getAngle(new Vector2[] {point, returns[0], neighbors[0]}),
                    getAngle(new Vector2[] {point, returns[0], neighbors[1]})
            };
            // The neighbor with the lower angle is the one to use
            if(neighborAngles[0] < neighborAngles[1]) {
                returns[1] = neighbors[0];
            } else {
                returns[1] = neighbors[1];
            }
    
            return returns;
    
        }
    
        public static double getAngle (Vector2[] abc) {
    
            // https://stackoverflow.com/questions/1211212/how-to-calculate-an-angle-from-three-points
            // atan2(P2.y - P1.y, P2.x - P1.x) - atan2(P3.y - P1.y, P3.x - P1.x)
            return Math.atan2(abc[2].y - abc[0].y, abc[2].x - abc[0].x) - Math.atan2(abc[1].y - abc[0].y, abc[1].x - abc[0].x);
    
        }
    
        public static int closestPointIndex (Vector2[] points, Vector2 point) {
    
            int leastDistanceIndex = 0;
            double leastDistance = Double.MAX_VALUE;
    
            for(int i = 0; i < points.length; i++) {
                double dist = distance(points[i], point);
                if(dist < leastDistance) {
                    leastDistanceIndex = i;
                    leastDistance = dist;
                }
            }
    
            return leastDistanceIndex;
    
        }
    
        public static double distance (Vector2 a, Vector2 b) {
            return Math.sqrt(Math.pow(Math.abs(a.x-b.x), 2)+Math.pow(Math.abs(a.y-b.y), 2));
        }
    
    }
    

    Here's a small explanation (Fun Fact: This is the first image I posted to stack overflow!)

    Sorry that it's so messy...


    Step-by-Step of the class:

    • Check if the given point is within a polygon
      • If it isn't, return the current point as no changes are needed.
    • Find the closest VERTEX of the polygon
      • This is not the closest POINT, as a point can be between vertexes
    • Grab the two neighbors of the vertex, keeping the one with a lower angle.
      • The lower angle has lower distance than the higher angle because the higher angle "goes away" faster
    • Get the closest point of the line segment, using the answer to this question on StackOverflow.

    Congrats! You survived the bad tutorial! Hopefully it helped :) + Thanks to all of the commented links to answers that helped me help you!

    Snap Point to Line

    How to calculate an angle from 3 points

    0 讨论(0)
提交回复
热议问题