PostGIS  3.2.2dev-r@@SVN_REVISION@@

◆ lw_dist2d_seg_seg()

int lw_dist2d_seg_seg ( const POINT2D A,
const POINT2D B,
const POINT2D C,
const POINT2D D,
DISTPTS dl 
)

Finds the shortest distance between two segments.

This function is changed so it is not doing any comparison of distance but just sending every possible combination further to lw_dist2d_pt_seg

Definition at line 1829 of file measures.c.

1830 {
1831  double s_top, s_bot, s;
1832  double r_top, r_bot, r;
1833 
1834  /*A and B are the same point */
1835  if ((A->x == B->x) && (A->y == B->y))
1836  {
1837  return lw_dist2d_pt_seg(A, C, D, dl);
1838  }
1839 
1840  /*U and V are the same point */
1841  if ((C->x == D->x) && (C->y == D->y))
1842  {
1843  dl->twisted = ((dl->twisted) * (-1));
1844  return lw_dist2d_pt_seg(D, A, B, dl);
1845  }
1846 
1847  /* AB and CD are line segments */
1848  /* from comp.graphics.algo
1849 
1850  Solving the above for r and s yields
1851  (Ay-Cy)(Dx-Cx)-(Ax-Cx)(Dy-Cy)
1852  r = ----------------------------- (eqn 1)
1853  (Bx-Ax)(Dy-Cy)-(By-Ay)(Dx-Cx)
1854 
1855  (Ay-Cy)(Bx-Ax)-(Ax-Cx)(By-Ay)
1856  s = ----------------------------- (eqn 2)
1857  (Bx-Ax)(Dy-Cy)-(By-Ay)(Dx-Cx)
1858  Let P be the position vector of the intersection point, then
1859  P=A+r(B-A) or
1860  Px=Ax+r(Bx-Ax)
1861  Py=Ay+r(By-Ay)
1862  By examining the values of r & s, you can also determine some other limiting conditions:
1863  If 0<=r<=1 & 0<=s<=1, intersection exists
1864  r<0 or r>1 or s<0 or s>1 line segments do not intersect
1865  If the denominator in eqn 1 is zero, AB & CD are parallel
1866  If the numerator in eqn 1 is also zero, AB & CD are collinear.
1867 
1868  */
1869  r_top = (A->y - C->y) * (D->x - C->x) - (A->x - C->x) * (D->y - C->y);
1870  r_bot = (B->x - A->x) * (D->y - C->y) - (B->y - A->y) * (D->x - C->x);
1871 
1872  s_top = (A->y - C->y) * (B->x - A->x) - (A->x - C->x) * (B->y - A->y);
1873  s_bot = (B->x - A->x) * (D->y - C->y) - (B->y - A->y) * (D->x - C->x);
1874 
1875  if ((r_bot == 0) || (s_bot == 0))
1876  {
1877  if ((lw_dist2d_pt_seg(A, C, D, dl)) && (lw_dist2d_pt_seg(B, C, D, dl)))
1878  {
1879  /* change the order of inputted geometries and that we notice by changing sign on dl->twisted*/
1880  dl->twisted *= -1;
1881  return ((lw_dist2d_pt_seg(C, A, B, dl)) && (lw_dist2d_pt_seg(D, A, B, dl)));
1882  }
1883  else
1884  return LW_FALSE; /* if any of the calls to lw_dist2d_pt_seg goes wrong we return false*/
1885  }
1886 
1887  s = s_top / s_bot;
1888  r = r_top / r_bot;
1889 
1890  if (((r < 0) || (r > 1) || (s < 0) || (s > 1)) || (dl->mode == DIST_MAX))
1891  {
1892  if ((lw_dist2d_pt_seg(A, C, D, dl)) && (lw_dist2d_pt_seg(B, C, D, dl)))
1893  {
1894  /* change the order of inputted geometries and that we notice by changing sign on dl->twisted*/
1895  dl->twisted *= -1;
1896  return ((lw_dist2d_pt_seg(C, A, B, dl)) && (lw_dist2d_pt_seg(D, A, B, dl)));
1897  }
1898  else
1899  return LW_FALSE; /* if any of the calls to lw_dist2d_pt_seg goes wrong we return false*/
1900  }
1901  else
1902  {
1903  /* If there is intersection we identify the intersection point and return it but only if we are looking
1904  * for mindistance */
1905  if (dl->mode == DIST_MIN)
1906  {
1907  POINT2D theP;
1908 
1909  if (((A->x == C->x) && (A->y == C->y)) || ((A->x == D->x) && (A->y == D->y)))
1910  {
1911  theP.x = A->x;
1912  theP.y = A->y;
1913  }
1914  else if (((B->x == C->x) && (B->y == C->y)) || ((B->x == D->x) && (B->y == D->y)))
1915  {
1916  theP.x = B->x;
1917  theP.y = B->y;
1918  }
1919  else
1920  {
1921  theP.x = A->x + r * (B->x - A->x);
1922  theP.y = A->y + r * (B->y - A->y);
1923  }
1924  lw_dist2d_distpts_set(dl, 0, &theP, &theP);
1925  }
1926  return LW_TRUE;
1927  }
1928 }
char * s
Definition: cu_in_wkt.c:23
char * r
Definition: cu_in_wkt.c:24
#define LW_FALSE
Definition: liblwgeom.h:108
#define LW_TRUE
Return types for functions with status returns.
Definition: liblwgeom.h:107
static void lw_dist2d_distpts_set(DISTPTS *dl, double distance, const POINT2D *p1, const POINT2D *p2)
Definition: measures.c:78
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:2216
#define DIST_MIN
Definition: measures.h:44
#define DIST_MAX
Definition: measures.h:43
int twisted
Definition: measures.h:55
int mode
Definition: measures.h:54
double y
Definition: liblwgeom.h:404
double x
Definition: liblwgeom.h:404

References DIST_MAX, DIST_MIN, lw_dist2d_distpts_set(), lw_dist2d_pt_seg(), LW_FALSE, LW_TRUE, DISTPTS::mode, r, s, DISTPTS::twisted, POINT2D::x, and POINT2D::y.

Referenced by lw_dist2d_arc_arc(), lw_dist2d_ptarray_ptarray(), lw_dist2d_seg_arc(), and rect_leaf_node_distance().

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