Geogram  Version 1.9.1
A programming library of geometric algorithms
generic_RVD_vertex.h
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2000-2022 Inria
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are met:
7  *
8  * * Redistributions of source code must retain the above copyright notice,
9  * this list of conditions and the following disclaimer.
10  * * Redistributions in binary form must reproduce the above copyright notice,
11  * this list of conditions and the following disclaimer in the documentation
12  * and/or other materials provided with the distribution.
13  * * Neither the name of the ALICE Project-Team nor the names of its
14  * contributors may be used to endorse or promote products derived from this
15  * software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27  * POSSIBILITY OF SUCH DAMAGE.
28  *
29  * Contact: Bruno Levy
30  *
31  * https://www.inria.fr/fr/bruno-levy
32  *
33  * Inria,
34  * Domaine de Voluceau,
35  * 78150 Le Chesnay - Rocquencourt
36  * FRANCE
37  *
38  */
39 
40 #ifndef GEOGRAM_VORONOI_GENERIC_RVD_VERTEX
41 #define GEOGRAM_VORONOI_GENERIC_RVD_VERTEX
42 
43 #include <geogram/basic/common.h>
44 #include <geogram/mesh/mesh.h>
46 #include <geogram/basic/assert.h>
47 #include <geogram/basic/process.h>
49 
60 namespace GEOGen {
61 
62  using GEO::Delaunay;
63  using GEO::index_t;
64  using GEO::signed_index_t;
65  using GEO::coord_index_t;
66  using GEO::Sign;
68  using GEO::Mesh;
69 
78  template <class T, index_t DIM>
79  class small_set {
80 
83 
84  public:
86  typedef T* iterator;
87 
89  typedef const T* const_iterator;
90 
92  typedef T& reference;
93 
95  typedef T value_type;
96 
101  size_(0) {
102  }
103 
107  index_t size() const {
108  return size_;
109  }
110 
115  index_t capacity() const {
116  return (index_t) DIM;
117  }
118 
123  return data_;
124  }
125 
130  return data_ + size_;
131  }
132 
138  return data_ + DIM;
139  }
140 
145  return data_;
146  }
147 
151  const_iterator end() const {
152  return data_ + size_;
153  }
154 
160  return data_ + DIM;
161  }
162 
169  iterator insert(const T& x) {
170  return insert(x, find_i(x));
171  }
172 
181  iterator insert(const T& x, iterator where) {
182  if(where == end()) {
183  *where = x;
184  grow();
185  return where;
186  }
187  if(*where == x) {
188  return where;
189  }
190  grow();
191  if(where == end() - 1) {
192  *where = x;
193  return where;
194  }
195  for(iterator i = end() - 1; i != where; i--) {
196  geo_debug_assert(i != begin());
197  *i = *(i - 1);
198  }
199  *where = x;
200 #ifdef GEO_DEBUG
201  for(iterator i = begin(); i != end() - 1; ++i) {
202  geo_debug_assert(*i < *(i + 1));
203  }
204 #endif
205  return where;
206  }
207 
211  void clear() {
212  size_ = 0;
213  }
214 
220  iterator find(const T& x) {
221  iterator result = find_i(x);
222  if(*result != x) {
223  result = end();
224  }
225  return result;
226  }
227 
233  const_iterator find(const T& x) const {
234  const_iterator result = find_i(x);
235  if(*result != x) {
236  result = end();
237  }
238  return result;
239  }
240 
246  void push_back(const T& x) {
247 #ifdef GEO_DEBUG
248  for(iterator i = begin(); i != end(); ++i) {
249  geo_debug_assert(*i < x);
250  }
251 #endif
252  *end() = x;
253  grow();
254  }
255 
259  void print(std::ostream& out) const {
260  out << "[ ";
261  for(const_iterator it = begin(); it != end(); ++it) {
262  out << *it << " ";
263  }
264  out << "]";
265  }
266 
273  geo_debug_assert(i >= 0);
274  geo_debug_assert(begin() + i < end());
275  return begin()[i];
276  }
277 
283  const T& operator[] (signed_index_t i) const {
284  geo_debug_assert(i >= 0);
285  geo_debug_assert(begin() + i < end());
286  return begin()[i];
287  }
288 
289  protected:
294  void grow() {
296  size_++;
297  }
298 
299  // Note: maybe we should start from end() instead of begin()
300  // since negative indices are inserted first.
301 
309  iterator find_i(const T& x) {
310  iterator result = begin();
311  while(result != end() && *result < x) {
312  result++;
313  }
314  return result;
315  }
316 
323  const_iterator find_i(const T& x) const {
324  const_iterator result = begin();
325  while(result != end() && *result < x) {
326  result++;
327  }
328  return result;
329  }
330 
331  protected:
332  T data_[DIM];
333  index_t size_;
334  };
335 
339  template <class T, index_t DIM>
340  inline std::ostream& operator<< (
341  std::ostream& out,
342  const small_set<T, DIM>& S) {
343  S.print(out);
344  return out;
345  }
346 
353  template <class T, index_t DIM1, index_t DIM2, index_t DIM3>
354  inline void sets_intersect(
355  const small_set<T, DIM1>& S1,
356  const small_set<T, DIM2>& S2,
358  ) {
359  I.clear();
360  auto i1 = S1.begin();
361  auto i2 = S2.begin();
362  while(i1 < S1.end() && i2 < S2.end()) {
363  if(*i1 < *i2) {
364  ++i1;
365  }
366  else if(*i2 < *i1) {
367  ++i2;
368  }
369  else {
370  I.push_back(*i1);
371  ++i1;
372  ++i2;
373  }
374  }
375  }
376 
408  class SymbolicVertex : public small_set<GEO::signed_index_t, 3> {
409 
411  typedef SymbolicVertex thisclass;
412 
415 
416  public:
421  v1_(0),
422  v2_(0) {
423  }
424 
430  }
431 
437  }
438 
444  index_t result = 0;
445  for(auto it = baseclass::begin();
446  it != baseclass::end() && *it < 0; ++it) {
447  result++;
448  }
449  return result;
450  }
451 
456  index_t result = 0;
457  for(auto it = baseclass::end() - 1;
458  it != baseclass::begin() - 1 && *it > 0; --it) {
459  result++;
460  }
461  return result;
462  }
463 
470  geo_debug_assert(x >= 0);
471  return (index_t) (x);
472  }
473 
483  return to_unsigned_int((baseclass::end()[-1 - i]) - 1);
484  }
485 
494  return to_unsigned_int(-(baseclass::begin()[i]) - 1);
495  }
496 
502  bool has_bisector(index_t i) const {
503  return baseclass::find(signed_index_t(i) + 1) != baseclass::end();
504  }
505 
511  bool has_boundary_facet(index_t i) const {
512  return baseclass::find(-signed_index_t(i) - 1) != baseclass::end();
513  }
514 
522  geo_debug_assert(v1_ != 0);
523  return v1_ - 1;
524  }
525 
533  void get_boundary_edge(index_t& v1, index_t& v2) const {
535  geo_debug_assert(v1_ != 0);
536  geo_debug_assert(v2_ != 0);
537  v1 = v1_ - 1;
538  v2 = v2_ - 1;
539  }
540 
546  v1_ = v + 1;
547  v2_ = 0;
548  }
549 
556  v1_ = v1 + 1;
557  v2_ = v2 + 1;
558  }
559 
566  geo_debug_assert(rhs.nb_bisectors() == 1);
567  geo_debug_assert(rhs.v1_ > 0);
568  geo_debug_assert(rhs.v2_ > 0);
569  v1_ = rhs.v1_;
570  v2_ = rhs.v2_;
571  }
572 
583  const thisclass& v1,
584  const thisclass& v2,
585  index_t E
586  ) {
587 
588  // Compute the symbolic representation as the intersection
589  // of three planes.
590  sets_intersect(v1, v2, *this);
591  // this computes the set of planes that contain
592  // the edge [v1,v2]
593 
594  add_bisector(E); // the intersection is on E.
595 
596  // Compute the symbolic representation as intersection between
597  // bisector and boundary edge
598  // (it's redundant and less elegant than the representation
599  // as planes interactions,
600  // but we need this to handle degenerate configurations properly,
601  // and to use exact predicates with original boundary vertices
602  // coordinates).
603 
604  if(nb_boundary_facets() == 2) {
605  // If *this is on the intersection of two boundary facets,
606  // then *this is on
607  // a boundary edge, and we need to retrieve the indices of the
608  // two extremities of this boundary edge.
609 
610  index_t nb1 = v1.nb_boundary_facets();
611  index_t nb2 = v2.nb_boundary_facets();
612  if(nb1 == 3 && nb2 == 3) {
613  // If v1 and v2 are boundary vertices,
614  // then I is on the boundary
615  // edge that connects v1 and v2
617  v1.get_boundary_vertex(),
619  );
620  } else if(nb1 == 2) {
622  // If v1 is on a boundary edge,
623  // then I is on the same boundary edge as v1
625  } else if(nb2 == 2) {
627  // If v2 is on a boundary edge,
628  // then I is on the same boundary edge as v2
630  }
631  }
632 
633  // Sanity check: problem detected here, we
634  // notify the caller that will use a workaround
635  // (see clip_by_plane())
636  if(baseclass::size() != 3) {
637  return false;
638  }
639  return true;
640  }
641 
642  private:
643  index_t v1_;
644  index_t v2_;
645  };
646 
647 
669  public:
675  size_(0),
676  capacity_(0),
677  dimension_(dim) {
678  }
679 
685  double* new_item() {
686  if(size_ == capacity_) {
687  grow();
688  }
689  size_++;
690  return item(size_ - 1);
691  }
692 
696  void clear() {
697  size_ = 0;
698  }
699 
705  for(index_t c = 0; c < chunks_.size(); c++) {
706  GEO::Memory::aligned_free(chunks_[c]);
707  }
708  }
709 
715  return dimension_;
716  }
717 
718  protected:
722  enum {
723  CHUNK_SHIFT = 8,
724  CHUNK_SIZE = 1 << CHUNK_SHIFT,
725  CHUNK_MASK = CHUNK_SIZE - 1
726  };
727 
731  void grow() {
732  chunks_.push_back(
733  reinterpret_cast<double*>(
735  index_t(CHUNK_SIZE) * dimension_ * sizeof(double)
736  )
737  )
738  );
739  capacity_ += CHUNK_SIZE;
740  }
741 
747  double* item(index_t i) {
748  geo_debug_assert(i < size_);
749  return &(chunks_[i >> CHUNK_SHIFT][(i & CHUNK_MASK) * dimension_]);
750  }
751 
752  private:
753  index_t size_;
754  index_t capacity_;
755  coord_index_t dimension_;
756  std::vector<double*> chunks_;
757  };
758 
762  enum {
763  ORIGINAL = 1,
764  INTERSECT = 2
765  };
766 
772 
776  typedef index_t EdgeFlag;
777 
787  class Vertex {
788 
790  typedef Vertex thisclass;
791 
792  public:
801  const double* p, double w, signed_index_t f,
802  const SymbolicVertex& sym
803  ) :
804  point_(p),
805  weight_(w),
806  f_(f),
807  seed_(-1),
808  sym_(sym),
809  flags_(ORIGINAL) {
810  }
811 
818  Vertex(const double* p, double w, signed_index_t f) :
819  point_(p),
820  weight_(w),
821  f_(f),
822  seed_(-1),
823  flags_(ORIGINAL) {
824  }
825 
829  Vertex() :
830  point_(nullptr),
831  weight_(1.0),
832  f_(-1),
833  seed_(-1),
834  flags_(0) {
835  }
836 
841  const double* point() const {
842  return point_;
843  }
844 
849  void set_point(const double* p) {
850  point_ = p;
851  }
852 
858  double weight() const {
859  return weight_;
860  }
861 
867  void set_weight(double w) {
868  weight_ = w;
869  }
870 
876  return seed_;
877  }
878 
884  seed_ = s;
885  }
886 
892  const SymbolicVertex& sym() const {
893  return sym_;
894  }
895 
900  return sym_;
901  }
902 
908  return f_;
909  }
910 
916  f_ = f;
917  }
918 
926  operator const double* () const {
927  return point_;
928  }
929 
933  void clear() {
934  flags_ = 0;
935  f_ = -1;
936  }
937 
941  void set_flag(EdgeFlag f) {
942  flags_ |= f;
943  }
944 
949  flags_ &= ~f;
950  }
951 
955  bool check_flag(EdgeFlag f) const {
956  return (flags_ & f) != 0;
957  }
958 
962  void copy_edge_from(const Vertex& rhs) {
964  flags_ = rhs.flags_;
965  }
966 
976  template <index_t DIM>
978  PointAllocator& target_intersections,
979  const Vertex& vq1, const Vertex& vq2,
980  const double* p1, const double* p2
981  ) {
982  const double* q1 = vq1.point();
983  const double* q2 = vq2.point();
984  double* Ipoint = target_intersections.new_item();
985  set_point(Ipoint);
986  double d = 0.0, l1 = 0.0, l2 = 0.0;
987  for(coord_index_t c = 0; c < DIM; ++c) {
988  double n = p1[c] - p2[c];
989  d -= n * (p2[c] + p1[c]);
990  l1 += q2[c] * n;
991  l2 += q1[c] * n;
992  }
993  d = 0.5 * d;
994  l1 = ::fabs(l1 + d);
995  l2 = ::fabs(l2 + d);
996  double l12 = l1 + l2;
997  if(l12 > 1e-30) {
998  l1 /= l12;
999  l2 /= l12;
1000  } else {
1001  l1 = 0.5;
1002  l2 = 0.5;
1003  }
1004  for(coord_index_t c = 0; c < DIM; ++c) {
1005  Ipoint[c] = l1 * q1[c] + l2 * q2[c];
1006  }
1007  set_weight(l1 * vq1.weight() + l2 * vq2.weight());
1008  }
1009 
1020  template <index_t DIM>
1022  const double* p1, const double* p2
1023  ) const {
1024  double r = 0.0;
1025  for(index_t c = 0; c < DIM; ++c) {
1026  r += GEO::geo_sqr(p2[c] - point()[c]);
1027  r -= GEO::geo_sqr(p1[c] - point()[c]);
1028  }
1029  return GEO::geo_sgn(r);
1030  }
1031 
1032  private:
1033  const double* point_;
1034  double weight_;
1035 
1040  signed_index_t f_;
1041 
1047  signed_index_t seed_;
1048 
1050  SymbolicVertex sym_;
1051 
1056  EdgeFlags flags_;
1057  };
1058 }
1059 
1060 /*
1061  namespace GEO {
1062  template <> struct can_be_used_as_attribute<GEOGen::SymbolicVertex> {
1063  static constexpr auto value = std::integral_constant<bool,true>();
1064  };
1065  }
1066 */
1067 
1068 
1069 #endif
Assertion checking mechanism.
#define geo_debug_assert(x)
Verifies that a condition is met.
Definition: assert.h:196
Generic mechanism for attributes.
An allocator for points that are created from intersections in GenericVoronoiDiagram.
void clear()
Clears this PointAllocator.
double * new_item()
Allocates a new point.
~PointAllocator()
PointAllocator destructor.
double * item(index_t i)
Gets a pointer to one of the allocated points from its index.
void grow()
Allocates a new chunk of memory.
coord_index_t dimension() const
Gets the dimension of the points stored in this PointAllocator.
PointAllocator(coord_index_t dim)
Creates a new empty PointAllocator.
A set of three integers that encodes the equation of a vertex in GenericVoronoiDiagram.
bool has_bisector(index_t i) const
Tests whether a bisector is present in the symbolic representation this vertex.
void copy_boundary_edge_from(const thisclass &rhs)
Copies a boundary edge from the symbolic representation of another vertex.
void set_boundary_edge(index_t v1, index_t v2)
Sets the boundary edge on which this vertex is located.
index_t boundary_facet(signed_index_t i) const
Gets a boundary facet.
void add_boundary_facet(index_t i)
Adds a boundary facet to the symbolic representation.
void get_boundary_edge(index_t &v1, index_t &v2) const
Gets the global indices of the boundary vertices that define the boundary edge on which this vertex i...
index_t nb_boundary_facets() const
Gets the number of boundary facets in the symbolic representation.
index_t get_boundary_vertex() const
Gets the global index of the boundary vertex that corresponds to this vertex.
index_t nb_bisectors() const
Gets the number of bisectors in the symbolic representation.
void set_boundary_vertex(index_t v)
Sets the boundary vertex on which this vertex is located.
bool has_boundary_facet(index_t i) const
Tests whether a boundary facet is present in the symbolic representation of this vertex.
SymbolicVertex()
Creates an uninitialized SymbolicVertex.
static index_t to_unsigned_int(signed_index_t x)
Casts a signed_index_t into an (unsigned) index_t.
index_t bisector(signed_index_t i) const
Gets a bisector.
void add_bisector(index_t i)
Adds a bisector to the symbolic representation.
bool intersect_symbolic(const thisclass &v1, const thisclass &v2, index_t E)
Computes the symbolic representation of the intersection between a segment and a bisector.
Internal representation of vertices in GenericVoronoiDiagram.
void clear()
Clears this Vertex.
Vertex(const double *p, double w, signed_index_t f, const SymbolicVertex &sym)
Creates a new Vertex.
void set_adjacent_seed(signed_index_t s)
Sets the adjacent seed.
void set_flag(EdgeFlag f)
Sets an EdgeFlag in this Vertex.
Vertex()
Creates an uninitialized Vertex.
void intersect_geom(PointAllocator &target_intersections, const Vertex &vq1, const Vertex &vq2, const double *p1, const double *p2)
Computes the intersection between a segment and a bisector.
Sign side_fast(const double *p1, const double *p2) const
Computes the side of this vertex relative to a bisector.
const SymbolicVertex & sym() const
Gets the symbolic representation.
void set_adjacent_facet(signed_index_t f)
Sets the adjacent facet.
void set_point(const double *p)
Sets the geometric location at this vertex.
double weight() const
Gets Vertex weight.
void copy_edge_from(const Vertex &rhs)
Copies adjacent facet and edge flags from another Vertex.
void unset_flag(EdgeFlag f)
Resets an EdgeFlag in this Vertex.
void set_weight(double w)
Sets the vertex weight.
SymbolicVertex & sym()
Gets the symbolic representation.
signed_index_t adjacent_seed() const
Gets the adjacent seed.
Vertex(const double *p, double w, signed_index_t f)
Creates a new Vertex.
const double * point() const
Gets the geometric location at this Vertex.
signed_index_t adjacent_facet() const
Gets the adjacent facet.
bool check_flag(EdgeFlag f) const
Tests an EdgeFlag in this Vertex.
Small_set is similar to std::set, but with fixed maximum size (and no dynamic memory allocation).
const_iterator begin() const
Gets a const iterator to the first element..
iterator find(const T &x)
Finds an element by value.
void grow()
Increases the size of this small_set.
iterator end()
Gets an iterator one position past the last element.
T value_type
Type of the elements.
index_t capacity() const
Gets the maximum number of elements that can be stored in this small_set.
const T * const_iterator
A random access iterator to const elements.
small_set()
Constructs an empty small_set.
iterator insert(const T &x, iterator where)
Inserts a new element at a specified location..
const_iterator find(const T &x) const
Finds an element by value.
T & operator[](signed_index_t i)
Direct access to an element.
const_iterator find_i(const T &x) const
Finds where an element should be located from its value.
T * iterator
A random access iterator to elements
const_iterator end_of_storage() const
Gets a const iterator one position past the last element that can be stored.
iterator begin()
Gets an iterator to the first element.
void clear()
Clears this small_set.
T & reference
Reference to element.
iterator find_i(const T &x)
Finds where an element is or where it should be inserted from its value.
void print(std::ostream &out) const
Displays the stored elements.
iterator insert(const T &x)
Insert a new element.
void push_back(const T &x)
Appends an element to the end of the list.
index_t size() const
Gets the number of element in this small_set.
const_iterator end() const
Gets a const iterator one position past the last element.
iterator end_of_storage()
Gets an iterator one position past the last element that can be stored.
Abstract interface for Delaunay triangulation in Nd.
Definition: delaunay.h:71
Represents a mesh.
Definition: mesh.h:2701
Implementation of Delaunay using nearest neighbors.
std::ostream & operator<<(std::ostream &os, const GEO::expansion_nt &a)
Displays the approximated value of an expansion_nt to a stream.
index_t EdgeFlags
A set of EdgeFlags.
void sets_intersect(const small_set< T, DIM1 > &S1, const small_set< T, DIM2 > &S2, small_set< T, DIM3 > &I)
Computes the intersection between two small_sets.
index_t EdgeFlag
An individual edge flag.
Common include file, providing basic definitions. Should be included before anything else by all head...
The class that represents a mesh.
void * aligned_malloc(size_t size, size_t alignment=GEO_MEMORY_ALIGNMENT)
Allocates aligned memory.
Definition: memory.h:281
void aligned_free(void *p)
Deallocates aligned memory.
Definition: memory.h:309
T geo_sqr(T x)
Gets the square value of a value.
Definition: numeric.h:301
geo_signed_index_t signed_index_t
The type for storing and manipulating indices differences.
Definition: numeric.h:343
Sign geo_sgn(const T &x)
Gets the sign of a value.
Definition: numeric.h:90
geo_index_t index_t
The type for storing and manipulating indices.
Definition: numeric.h:329
Sign
Integer constants that represent the sign of a value.
Definition: numeric.h:68
geo_coord_index_t coord_index_t
The type for storing coordinate indices, and iterating on the coordinates of a point.
Definition: numeric.h:363
Function and classes for process manipulation.