Find the face-id of the face properly containing a given point.
7111 uint64_t numedges, i;
7113 const POINT2D *closestPointOnEdge = NULL;
7114 uint32_t closestSegmentIndex;
7115 int closestSegmentSide;
7116 uint32_t closestPointVertex;
7117 const POINT2D *closestSegmentP0, *closestSegmentP1;
7120 int containingFace = -1;
7130 if (numedges == UINT64_MAX)
7176 closestSegmentP0->
x,
7177 closestSegmentP0->
y,
7178 closestSegmentP1->
x,
7196 const POINT2D *p = queryPoint;
7197 const POINT2D *A = closestSegmentP0;
7198 const POINT2D *B = closestSegmentP1;
7199 double r = ( (p->
x-A->
x) * (B->
x-A->
x) + (p->
y-A->
y) * (B->
y-A->
y) )/( (B->
x-A->
x)*(B->
x-A->
x) +(B->
y-A->
y)*(B->
y-A->
y) );
7202 closestPointOnEdge = A;
7203 closestPointVertex = closestSegmentIndex;
7204 if ( closestSegmentIndex == 0 )
7211 closestPointOnEdge = B;
7212 closestPointVertex = closestSegmentIndex + 1;
7215 closestNode = closestEdge->
end_node;
7223 if ( closestNode != 0 )
7225 LWDEBUGF(1,
"Closest point is node %d", closestNode);
7228 LWDEBUGF(1,
"Query point is node %d", closestNode);
7239 lwerror(
"Two or more faces found");
7242 containingFace = closestEdge->
face_left;
7247 if (numedges == UINT64_MAX)
7254 for (i=0; i<numedges; ++i)
7256 if ( edges[i].face_left != containingFace ||
7257 edges[i].face_right != containingFace )
7261 lwerror(
"Two or more faces found");
7267 lwerror(
"Unexpected backend return: getEdgeByNode(%d) returns no edges when we previously found edge %d ending on that node",
7268 closestNode, closestEdge->
edge_id);
7273 LWDEBUGF(1,
"lwt_be_getEdgeByNode returned %d edges", numedges);
7276 return containingFace;
7284 lwerror(
"error computing azimuth of query point [%.15g %.15g,%.15g %.15g]",
7285 closestPointOnEdge->
x, closestPointOnEdge->
y,
7286 queryPoint->
x, queryPoint->
y);
7295 lwerror(
"Unexpected backend return: _lwt_FindAdjacentEdges(%d) found no edges when we previously found edge %d ending on that node",
7296 closestNode, closestEdge->
edge_id);
7306 LWDEBUG(1,
"Closest point is NOT a node");
7312 containingFace = closestEdge->
face_left;
7314 return containingFace;
7321 lwerror(
"Two or more faces found");
7329 LWDEBUGF(1,
"Closest point is vertex %d of closest segment", closestPointVertex);
7340 uint32_t prevVertexIndex = closestPointVertex > 0 ?
7341 closestPointVertex - 1u :
7349 LWDEBUGF(1,
"Previous vertex %u is POINT(%.15g %.15g)",
7355 uint32_t nextVertexIndex = closestPointVertex == closestEdge->
geom->
points->
npoints - 1u ?
7357 closestPointVertex + 1u;
7364 LWDEBUGF(1,
"Next vertex %u is POINT(%.15g %.15g)",
7375 if ( !
azimuth_pt_pt(closestPointOnEdge, prevVertex, &azS0)) {
7376 lwerror(
"error computing azimuth of segment to closest point [%.15g %.15g,%.15g %.15g]",
7377 closestPointOnEdge->
x, closestPointOnEdge->
y,
7378 prevVertex->
x, prevVertex->
y);
7382 if ( !
azimuth_pt_pt(closestPointOnEdge, nextVertex, &azS1)) {
7383 lwerror(
"error computing azimuth of segment from closest point [%.15g %.15g,%.15g %.15g]",
7384 closestPointOnEdge->
x, closestPointOnEdge->
y,
7385 nextVertex->
x, nextVertex->
y);
7389 if ( !
azimuth_pt_pt(closestPointOnEdge, queryPoint, &azSL) ) {
7390 lwerror(
"error computing azimuth of queryPoint [%.15g %.15g,%.15g %.15g]",
7391 closestPointOnEdge->
x, closestPointOnEdge->
y,
7392 queryPoint->
x, queryPoint->
y);
7397 double angle_S0_S1 = azS1 - azS0;
7398 if ( angle_S0_S1 < 0 ) angle_S0_S1 += 2 * M_PI;
7400 double angle_S0_SL = azSL - azS0;
7401 if ( angle_S0_SL < 0 ) angle_S0_SL += 2 * M_PI;
7403 LWDEBUGF(1,
"Azimuths from closest (vertex) point: P:%g, N:%g (+%g), Q:%g (+%g)",
7408 if ( angle_S0_SL < angle_S0_S1 )
7411 containingFace = closestEdge->
face_left;
7424 LWDEBUGF(1,
"Closest point is internal to closest segment, calling lw_segment_side((%g,%g),(%g,%g),(%g,%g)",
7425 closestSegmentP0->
x,
7426 closestSegmentP0->
y,
7427 closestSegmentP1->
x,
7428 closestSegmentP1->
y,
7433 closestSegmentSide =
lw_segment_side(closestSegmentP0, closestSegmentP1, queryPoint);
7434 LWDEBUGF(1,
"Side of closest segment query point falls on: %d", closestSegmentSide);
7436 if ( closestSegmentSide == -1 )
7438 containingFace = closestEdge->
face_left;
7440 else if ( closestSegmentSide == 1 )
7446 lwerror(
"Unexpected collinearity reported from lw_segment_side");
7454 return containingFace;
LWGEOM * lwline_as_lwgeom(const LWLINE *obj)
int ptarray_closest_segment_2d(const POINTARRAY *pa, const POINT2D *qp, double *dist)
int azimuth_pt_pt(const POINT2D *p1, const POINT2D *p2, double *ret)
Compute the azimuth of segment AB in radians.
int lw_segment_side(const POINT2D *p1, const POINT2D *p2, const POINT2D *q)
lw_segment_side()
#define LWT_COL_EDGE_FACE_RIGHT
LWT_INT64 LWT_ELEMID
Identifier of topology element.
#define LWT_COL_EDGE_START_NODE
#define LWT_COL_EDGE_FACE_LEFT
#define LWT_COL_EDGE_EDGE_ID
Edge fields.
#define LWT_COL_EDGE_END_NODE
#define LWT_COL_EDGE_GEOM
#define LWDEBUG(level, msg)
#define LWDEBUGF(level, msg,...)
void lwerror(const char *fmt,...)
Write a notice out to the error handler.
#define LWDEBUGGF(level, geom, fmt,...)
const char * lwt_be_lastErrorMessage(const LWT_BE_IFACE *be)
static LWT_ISO_EDGE * lwt_be_getEdgeByNode(LWT_TOPOLOGY *topo, const LWT_ELEMID *ids, uint64_t *numelems, int fields)
static LWT_ISO_EDGE * lwt_be_getClosestEdge(const LWT_TOPOLOGY *topo, const LWPOINT *pt, uint64_t *numelems, int fields)
static int _lwt_FindAdjacentEdges(LWT_TOPOLOGY *topo, LWT_ELEMID node, edgeend *data, edgeend *other, int myedge_id)
static void _lwt_release_edges(LWT_ISO_EDGE *edges, int num_edges)
static const POINT2D * getPoint2d_cp(const POINTARRAY *pa, uint32_t n)
Returns a POINT2D pointer into the POINTARRAY serialized_ptlist, suitable for reading from.
const LWT_BE_IFACE * be_iface