PostGIS  3.3.9dev-r@@SVN_REVISION@@

◆ lw_dist2d_pt_arc()

int lw_dist2d_pt_arc ( const POINT2D P,
const POINT2D A1,
const POINT2D A2,
const POINT2D A3,
DISTPTS dl 
)

Definition at line 1523 of file measures.c.

1524 {
1525  double radius_A, d;
1526  POINT2D C; /* center of circle defined by arc A */
1527  POINT2D X; /* point circle(A) where line from C to P crosses */
1528 
1529  if (dl->mode < 0)
1530  lwerror("lw_dist2d_pt_arc does not support maxdistance mode");
1531 
1532  /* What if the arc is a point? */
1533  if (lw_arc_is_pt(A1, A2, A3))
1534  return lw_dist2d_pt_pt(P, A1, dl);
1535 
1536  /* Calculate centers and radii of circles. */
1537  radius_A = lw_arc_center(A1, A2, A3, &C);
1538 
1539  /* This "arc" is actually a line (A2 is colinear with A1,A3) */
1540  if (radius_A < 0.0)
1541  return lw_dist2d_pt_seg(P, A1, A3, dl);
1542 
1543  /* Distance from point to center */
1544  d = distance2d_pt_pt(&C, P);
1545 
1546  /* P is the center of the circle */
1547  if (FP_EQUALS(d, 0.0))
1548  {
1549  dl->distance = radius_A;
1550  dl->p1 = *A1;
1551  dl->p2 = *P;
1552  return LW_TRUE;
1553  }
1554 
1555  /* X is the point on the circle where the line from P to C crosses */
1556  X.x = C.x + (P->x - C.x) * radius_A / d;
1557  X.y = C.y + (P->y - C.y) * radius_A / d;
1558 
1559  /* Is crossing point inside the arc? Or arc is actually circle? */
1560  if (p2d_same(A1, A3) || lw_pt_in_arc(&X, A1, A2, A3))
1561  {
1562  lw_dist2d_pt_pt(P, &X, dl);
1563  }
1564  else
1565  {
1566  /* Distance is the minimum of the distances to the arc end points */
1567  lw_dist2d_pt_pt(A1, P, dl);
1568  lw_dist2d_pt_pt(A3, P, dl);
1569  }
1570  return LW_TRUE;
1571 }
#define LW_TRUE
Return types for functions with status returns.
Definition: liblwgeom.h:108
double lw_arc_center(const POINT2D *p1, const POINT2D *p2, const POINT2D *p3, POINT2D *result)
Determines the center of the circle defined by the three given points.
Definition: lwalgorithm.c:229
#define FP_EQUALS(A, B)
int lw_arc_is_pt(const POINT2D *A1, const POINT2D *A2, const POINT2D *A3)
Returns true if arc A is actually a point (all vertices are the same) .
Definition: lwalgorithm.c:106
int lw_pt_in_arc(const POINT2D *P, const POINT2D *A1, const POINT2D *A2, const POINT2D *A3)
Returns true if P is on the same side of the plane partition defined by A1/A3 as A2 is.
Definition: lwalgorithm.c:86
int p2d_same(const POINT2D *p1, const POINT2D *p2)
Definition: lwalgorithm.c:50
void lwerror(const char *fmt,...)
Write a notice out to the error handler.
Definition: lwutil.c:190
double distance2d_pt_pt(const POINT2D *p1, const POINT2D *p2)
Definition: measures.c:2445
int lw_dist2d_pt_seg(const POINT2D *p, const POINT2D *A, const POINT2D *B, DISTPTS *dl)
lw_dist2d_comp from p to line A->B This one is now sending every occasion to lw_dist2d_pt_pt Before i...
Definition: measures.c:2316
int lw_dist2d_pt_pt(const POINT2D *thep1, const POINT2D *thep2, DISTPTS *dl)
Compares incoming points and stores the points closest to each other or most far away from each other...
Definition: measures.c:2413
POINT2D p1
Definition: measures.h:52
POINT2D p2
Definition: measures.h:53
int mode
Definition: measures.h:54
double distance
Definition: measures.h:51
double y
Definition: liblwgeom.h:405
double x
Definition: liblwgeom.h:405

References DISTPTS::distance, distance2d_pt_pt(), FP_EQUALS, lw_arc_center(), lw_arc_is_pt(), lw_dist2d_pt_pt(), lw_dist2d_pt_seg(), lw_pt_in_arc(), LW_TRUE, lwerror(), DISTPTS::mode, DISTPTS::p1, DISTPTS::p2, p2d_same(), POINT2D::x, and POINT2D::y.

Referenced by lw_dist2d_arc_arc(), lw_dist2d_pt_ptarrayarc(), lw_dist2d_seg_arc(), rect_leaf_node_distance(), rect_leaf_node_intersects(), and test_lw_dist2d_pt_arc().

Here is the call graph for this function:
Here is the caller graph for this function: