PostGIS  2.4.9dev-r@@SVN_REVISION@@

◆ ptarray_segmentize2d()

POINTARRAY* ptarray_segmentize2d ( const POINTARRAY ipa,
double  dist 
)

Returns a modified POINTARRAY so that no segment is longer than the given distance (computed using 2d).

Every input point is kept. Z and M values for added points (if needed) are set to 0.

Definition at line 415 of file ptarray.c.

References distance2d_pt_pt(), POINTARRAY::flags, FLAGS_GET_M, FLAGS_GET_Z, getPoint4d_p(), LW_FALSE, LW_ON_INTERRUPT, LW_TRUE, POINT4D::m, POINTARRAY::npoints, ptarray_append_point(), ptarray_construct_empty(), ptarray_free(), POINT4D::x, POINT4D::y, and POINT4D::z.

Referenced by lwline_segmentize2d(), and lwpoly_segmentize2d().

416 {
417  double segdist;
418  POINT4D p1, p2;
419  POINT4D pbuf;
420  POINTARRAY *opa;
421  int ipoff=0; /* input point offset */
422  int hasz = FLAGS_GET_Z(ipa->flags);
423  int hasm = FLAGS_GET_M(ipa->flags);
424 
425  pbuf.x = pbuf.y = pbuf.z = pbuf.m = 0;
426 
427  /* Initial storage */
428  opa = ptarray_construct_empty(hasz, hasm, ipa->npoints);
429 
430  /* Add first point */
431  getPoint4d_p(ipa, ipoff, &p1);
432  ptarray_append_point(opa, &p1, LW_FALSE);
433 
434  ipoff++;
435 
436  while (ipoff<ipa->npoints)
437  {
438  /*
439  * We use these pointers to avoid
440  * "strict-aliasing rules break" warning raised
441  * by gcc (3.3 and up).
442  *
443  * It looks that casting a variable address (also
444  * referred to as "type-punned pointer")
445  * breaks those "strict" rules.
446  *
447  */
448  POINT4D *p1ptr=&p1, *p2ptr=&p2;
449 
450  getPoint4d_p(ipa, ipoff, &p2);
451 
452  segdist = distance2d_pt_pt((POINT2D *)p1ptr, (POINT2D *)p2ptr);
453 
454  if (segdist > dist) /* add an intermediate point */
455  {
456  pbuf.x = p1.x + (p2.x-p1.x)/segdist * dist;
457  pbuf.y = p1.y + (p2.y-p1.y)/segdist * dist;
458  if( hasz )
459  pbuf.z = p1.z + (p2.z-p1.z)/segdist * dist;
460  if( hasm )
461  pbuf.m = p1.m + (p2.m-p1.m)/segdist * dist;
462  ptarray_append_point(opa, &pbuf, LW_FALSE);
463  p1 = pbuf;
464  }
465  else /* copy second point */
466  {
467  ptarray_append_point(opa, &p2, (ipa->npoints==2)?LW_TRUE:LW_FALSE);
468  p1 = p2;
469  ipoff++;
470  }
471 
472  LW_ON_INTERRUPT(ptarray_free(opa); return NULL);
473  }
474 
475  return opa;
476 }
double x
Definition: liblwgeom.h:352
double m
Definition: liblwgeom.h:352
int npoints
Definition: liblwgeom.h:371
#define LW_ON_INTERRUPT(x)
int ptarray_append_point(POINTARRAY *pa, const POINT4D *pt, int repeated_points)
Append a point to the end of an existing POINTARRAY If allow_duplicate is LW_FALSE, then a duplicate point will not be added.
Definition: ptarray.c:156
double distance2d_pt_pt(const POINT2D *p1, const POINT2D *p2)
The old function nessecary for ptarray_segmentize2d in ptarray.c.
Definition: measures.c:2317
#define LW_FALSE
Definition: liblwgeom.h:77
uint8_t flags
Definition: liblwgeom.h:369
#define LW_TRUE
Return types for functions with status returns.
Definition: liblwgeom.h:76
#define FLAGS_GET_Z(flags)
Macros for manipulating the &#39;flags&#39; byte.
Definition: liblwgeom.h:140
double z
Definition: liblwgeom.h:352
POINTARRAY * ptarray_construct_empty(char hasz, char hasm, uint32_t maxpoints)
Create a new POINTARRAY with no points.
Definition: ptarray.c:70
#define FLAGS_GET_M(flags)
Definition: liblwgeom.h:141
void ptarray_free(POINTARRAY *pa)
Definition: ptarray.c:330
double y
Definition: liblwgeom.h:352
int getPoint4d_p(const POINTARRAY *pa, int n, POINT4D *point)
Definition: lwgeom_api.c:122
Here is the call graph for this function:
Here is the caller graph for this function: