isotropy.hpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. /*
  2. Copyright 2008 Intel Corporation
  3. Use, modification and distribution are subject to the Boost Software License,
  4. Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  5. http://www.boost.org/LICENSE_1_0.txt).
  6. */
  7. #ifndef BOOST_POLYGON_ISOTROPY_HPP
  8. #define BOOST_POLYGON_ISOTROPY_HPP
  9. //external
  10. #include <cmath>
  11. #include <cstddef>
  12. #include <cstdlib>
  13. #include <vector>
  14. #include <deque>
  15. #include <map>
  16. #include <set>
  17. #include <list>
  18. //#include <iostream>
  19. #include <algorithm>
  20. #include <limits>
  21. #include <iterator>
  22. #include <string>
  23. #ifndef BOOST_POLYGON_NO_DEPS
  24. #include <boost/config.hpp>
  25. #ifdef BOOST_MSVC
  26. #define BOOST_POLYGON_MSVC
  27. #endif
  28. #ifdef BOOST_INTEL
  29. #define BOOST_POLYGON_ICC
  30. #endif
  31. #ifdef BOOST_HAS_LONG_LONG
  32. #define BOOST_POLYGON_USE_LONG_LONG
  33. typedef boost::long_long_type polygon_long_long_type;
  34. typedef boost::ulong_long_type polygon_ulong_long_type;
  35. //typedef long long polygon_long_long_type;
  36. //typedef unsigned long long polygon_ulong_long_type;
  37. #endif
  38. #include <boost/mpl/size_t.hpp>
  39. #include <boost/mpl/protect.hpp>
  40. #include <boost/utility/enable_if.hpp>
  41. #include <boost/mpl/bool.hpp>
  42. #include <boost/mpl/and.hpp>
  43. #include <boost/mpl/or.hpp>
  44. #else
  45. #ifdef _WIN32
  46. #define BOOST_POLYGON_MSVC
  47. #endif
  48. #ifdef __ICC
  49. #define BOOST_POLYGON_ICC
  50. #endif
  51. #define BOOST_POLYGON_USE_LONG_LONG
  52. typedef long long polygon_long_long_type;
  53. typedef unsigned long long polygon_ulong_long_type;
  54. namespace boost {
  55. template <bool B, class T = void>
  56. struct enable_if_c {
  57. typedef T type;
  58. };
  59. template <class T>
  60. struct enable_if_c<false, T> {};
  61. template <class Cond, class T = void>
  62. struct enable_if : public enable_if_c<Cond::value, T> {};
  63. template <bool B, class T>
  64. struct lazy_enable_if_c {
  65. typedef typename T::type type;
  66. };
  67. template <class T>
  68. struct lazy_enable_if_c<false, T> {};
  69. template <class Cond, class T>
  70. struct lazy_enable_if : public lazy_enable_if_c<Cond::value, T> {};
  71. template <bool B, class T = void>
  72. struct disable_if_c {
  73. typedef T type;
  74. };
  75. template <class T>
  76. struct disable_if_c<true, T> {};
  77. template <class Cond, class T = void>
  78. struct disable_if : public disable_if_c<Cond::value, T> {};
  79. template <bool B, class T>
  80. struct lazy_disable_if_c {
  81. typedef typename T::type type;
  82. };
  83. template <class T>
  84. struct lazy_disable_if_c<true, T> {};
  85. template <class Cond, class T>
  86. struct lazy_disable_if : public lazy_disable_if_c<Cond::value, T> {};
  87. }
  88. #endif
  89. namespace boost { namespace polygon{
  90. enum GEOMETRY_CONCEPT_ID {
  91. COORDINATE_CONCEPT,
  92. INTERVAL_CONCEPT,
  93. POINT_CONCEPT,
  94. POINT_3D_CONCEPT,
  95. RECTANGLE_CONCEPT,
  96. POLYGON_90_CONCEPT,
  97. POLYGON_90_WITH_HOLES_CONCEPT,
  98. POLYGON_45_CONCEPT,
  99. POLYGON_45_WITH_HOLES_CONCEPT,
  100. POLYGON_CONCEPT,
  101. POLYGON_WITH_HOLES_CONCEPT,
  102. POLYGON_90_SET_CONCEPT,
  103. POLYGON_45_SET_CONCEPT,
  104. POLYGON_SET_CONCEPT
  105. };
  106. struct undefined_concept {};
  107. template <typename T>
  108. struct geometry_concept { typedef undefined_concept type; };
  109. template <typename GCT, typename T>
  110. struct view_of {};
  111. template <typename T1, typename T2>
  112. view_of<T1, T2> view_as(const T2& obj) { return view_of<T1, T2>(obj); }
  113. template <typename T, bool /*enable*/ = true>
  114. struct coordinate_traits {};
  115. //used to override long double with an infinite precision datatype
  116. template <typename T>
  117. struct high_precision_type {
  118. typedef long double type;
  119. };
  120. template <typename T>
  121. T convert_high_precision_type(const typename high_precision_type<T>::type& v) {
  122. return T(v);
  123. }
  124. //used to override std::sort with an alternative (parallel) algorithm
  125. template <typename iter_type>
  126. void polygon_sort(iter_type _b_, iter_type _e_);
  127. template <typename iter_type, typename pred_type>
  128. void polygon_sort(iter_type _b_, iter_type _e_, const pred_type& _pred_);
  129. template <>
  130. struct coordinate_traits<int> {
  131. typedef int coordinate_type;
  132. typedef long double area_type;
  133. #ifdef BOOST_POLYGON_USE_LONG_LONG
  134. typedef polygon_long_long_type manhattan_area_type;
  135. typedef polygon_ulong_long_type unsigned_area_type;
  136. typedef polygon_long_long_type coordinate_difference;
  137. #else
  138. typedef long manhattan_area_type;
  139. typedef unsigned long unsigned_area_type;
  140. typedef long coordinate_difference;
  141. #endif
  142. typedef long double coordinate_distance;
  143. };
  144. template<>
  145. struct coordinate_traits<long, sizeof(long) == sizeof(int)> {
  146. typedef coordinate_traits<int> cT_;
  147. typedef cT_::coordinate_type coordinate_type;
  148. typedef cT_::area_type area_type;
  149. typedef cT_::manhattan_area_type manhattan_area_type;
  150. typedef cT_::unsigned_area_type unsigned_area_type;
  151. typedef cT_::coordinate_difference coordinate_difference;
  152. typedef cT_::coordinate_distance coordinate_distance;
  153. };
  154. #ifdef BOOST_POLYGON_USE_LONG_LONG
  155. template <>
  156. struct coordinate_traits<polygon_long_long_type> {
  157. typedef polygon_long_long_type coordinate_type;
  158. typedef long double area_type;
  159. typedef polygon_long_long_type manhattan_area_type;
  160. typedef polygon_ulong_long_type unsigned_area_type;
  161. typedef polygon_long_long_type coordinate_difference;
  162. typedef long double coordinate_distance;
  163. };
  164. template<>
  165. struct coordinate_traits<long, sizeof(long) == sizeof(polygon_long_long_type)> {
  166. typedef coordinate_traits<polygon_long_long_type> cT_;
  167. typedef cT_::coordinate_type coordinate_type;
  168. typedef cT_::area_type area_type;
  169. typedef cT_::manhattan_area_type manhattan_area_type;
  170. typedef cT_::unsigned_area_type unsigned_area_type;
  171. typedef cT_::coordinate_difference coordinate_difference;
  172. typedef cT_::coordinate_distance coordinate_distance;
  173. };
  174. #endif
  175. template <>
  176. struct coordinate_traits<float> {
  177. typedef float coordinate_type;
  178. typedef float area_type;
  179. typedef float manhattan_area_type;
  180. typedef float unsigned_area_type;
  181. typedef float coordinate_difference;
  182. typedef float coordinate_distance;
  183. };
  184. template <>
  185. struct coordinate_traits<double> {
  186. typedef double coordinate_type;
  187. typedef double area_type;
  188. typedef double manhattan_area_type;
  189. typedef double unsigned_area_type;
  190. typedef double coordinate_difference;
  191. typedef double coordinate_distance;
  192. };
  193. template <>
  194. struct coordinate_traits<long double> {
  195. typedef long double coordinate_type;
  196. typedef long double area_type;
  197. typedef long double manhattan_area_type;
  198. typedef long double unsigned_area_type;
  199. typedef long double coordinate_difference;
  200. typedef long double coordinate_distance;
  201. };
  202. template <typename T>
  203. struct scaling_policy {
  204. template <typename T2>
  205. static inline T round(T2 t2) {
  206. return (T)std::floor(t2+0.5);
  207. }
  208. static inline T round(T t2) {
  209. return t2;
  210. }
  211. };
  212. struct coordinate_concept {};
  213. template <>
  214. struct geometry_concept<int> { typedef coordinate_concept type; };
  215. #ifdef BOOST_POLYGON_USE_LONG_LONG
  216. template <>
  217. struct geometry_concept<polygon_long_long_type> { typedef coordinate_concept type; };
  218. #endif
  219. template <>
  220. struct geometry_concept<float> { typedef coordinate_concept type; };
  221. template <>
  222. struct geometry_concept<double> { typedef coordinate_concept type; };
  223. template <>
  224. struct geometry_concept<long double> { typedef coordinate_concept type; };
  225. #ifndef BOOST_POLYGON_NO_DEPS
  226. struct gtl_no : mpl::bool_<false> {};
  227. struct gtl_yes : mpl::bool_<true> {};
  228. template <typename T, typename T2>
  229. struct gtl_and : mpl::and_<T, T2> {};
  230. template <typename T, typename T2, typename T3>
  231. struct gtl_and_3 : mpl::and_<T, T2, T3> {};
  232. template <typename T, typename T2, typename T3, typename T4>
  233. struct gtl_and_4 : mpl::and_<T, T2, T3, T4> {};
  234. // template <typename T, typename T2>
  235. // struct gtl_or : mpl::or_<T, T2> {};
  236. // template <typename T, typename T2, typename T3>
  237. // struct gtl_or_3 : mpl::or_<T, T2, T3> {};
  238. // template <typename T, typename T2, typename T3, typename T4>
  239. // struct gtl_or_4 : mpl::or_<T, T2, T3, T4> {};
  240. #else
  241. struct gtl_no { static const bool value = false; };
  242. struct gtl_yes { typedef gtl_yes type;
  243. static const bool value = true; };
  244. template <bool T, bool T2>
  245. struct gtl_and_c { typedef gtl_no type; };
  246. template <>
  247. struct gtl_and_c<true, true> { typedef gtl_yes type; };
  248. template <typename T, typename T2>
  249. struct gtl_and : gtl_and_c<T::value, T2::value> {};
  250. template <typename T, typename T2, typename T3>
  251. struct gtl_and_3 { typedef typename gtl_and<
  252. T, typename gtl_and<T2, T3>::type>::type type; };
  253. template <typename T, typename T2, typename T3, typename T4>
  254. struct gtl_and_4 { typedef typename gtl_and_3<
  255. T, T2, typename gtl_and<T3, T4>::type>::type type; };
  256. #endif
  257. template <typename T, typename T2>
  258. struct gtl_or { typedef gtl_yes type; };
  259. template <typename T>
  260. struct gtl_or<T, T> { typedef T type; };
  261. template <typename T, typename T2, typename T3>
  262. struct gtl_or_3 { typedef typename gtl_or<
  263. T, typename gtl_or<T2, T3>::type>::type type; };
  264. template <typename T, typename T2, typename T3, typename T4>
  265. struct gtl_or_4 { typedef typename gtl_or<
  266. T, typename gtl_or_3<T2, T3, T4>::type>::type type; };
  267. template <typename T>
  268. struct gtl_not { typedef gtl_no type; };
  269. template <>
  270. struct gtl_not<gtl_no> { typedef gtl_yes type; };
  271. template <typename T>
  272. struct gtl_if {
  273. #ifdef BOOST_POLYGON_MSVC
  274. typedef gtl_no type;
  275. #endif
  276. };
  277. template <>
  278. struct gtl_if<gtl_yes> { typedef gtl_yes type; };
  279. template <typename T, typename T2>
  280. struct gtl_same_type { typedef gtl_no type; };
  281. template <typename T>
  282. struct gtl_same_type<T, T> { typedef gtl_yes type; };
  283. template <typename T, typename T2>
  284. struct gtl_different_type { typedef typename gtl_not<typename gtl_same_type<T, T2>::type>::type type; };
  285. struct manhattan_domain {};
  286. struct forty_five_domain {};
  287. struct general_domain {};
  288. template <typename T>
  289. struct geometry_domain { typedef general_domain type; };
  290. template <typename domain_type, typename coordinate_type>
  291. struct area_type_by_domain { typedef typename coordinate_traits<coordinate_type>::area_type type; };
  292. template <typename coordinate_type>
  293. struct area_type_by_domain<manhattan_domain, coordinate_type> {
  294. typedef typename coordinate_traits<coordinate_type>::manhattan_area_type type; };
  295. struct y_c_edist : gtl_yes {};
  296. template <typename coordinate_type_1, typename coordinate_type_2>
  297. typename enable_if<
  298. typename gtl_and_3<y_c_edist, typename gtl_same_type<typename geometry_concept<coordinate_type_1>::type, coordinate_concept>::type,
  299. typename gtl_same_type<typename geometry_concept<coordinate_type_1>::type, coordinate_concept>::type>::type,
  300. typename coordinate_traits<coordinate_type_1>::coordinate_difference>::type
  301. euclidean_distance(const coordinate_type_1& lvalue, const coordinate_type_2& rvalue) {
  302. typedef typename coordinate_traits<coordinate_type_1>::coordinate_difference Unit;
  303. return (lvalue < rvalue) ? (Unit)rvalue - (Unit)lvalue : (Unit)lvalue - (Unit)rvalue;
  304. }
  305. // predicated_swap swaps a and b if pred is true
  306. // predicated_swap is guarenteed to behave the same as
  307. // if(pred){
  308. // T tmp = a;
  309. // a = b;
  310. // b = tmp;
  311. // }
  312. // but will not generate a branch instruction.
  313. // predicated_swap always creates a temp copy of a, but does not
  314. // create more than one temp copy of an input.
  315. // predicated_swap can be used to optimize away branch instructions in C++
  316. template <class T>
  317. inline bool predicated_swap(const bool& pred,
  318. T& a,
  319. T& b) {
  320. const T tmp = a;
  321. const T* input[2] = {&b, &tmp};
  322. a = *input[!pred];
  323. b = *input[pred];
  324. return pred;
  325. }
  326. enum direction_1d_enum { LOW = 0, HIGH = 1,
  327. LEFT = 0, RIGHT = 1,
  328. CLOCKWISE = 0, COUNTERCLOCKWISE = 1,
  329. REVERSE = 0, FORWARD = 1,
  330. NEGATIVE = 0, POSITIVE = 1 };
  331. enum orientation_2d_enum { HORIZONTAL = 0, VERTICAL = 1 };
  332. enum direction_2d_enum { WEST = 0, EAST = 1, SOUTH = 2, NORTH = 3 };
  333. enum orientation_3d_enum { PROXIMAL = 2 };
  334. enum direction_3d_enum { DOWN = 4, UP = 5 };
  335. enum winding_direction {
  336. clockwise_winding = 0,
  337. counterclockwise_winding = 1,
  338. unknown_winding = 2
  339. };
  340. class direction_2d;
  341. class direction_3d;
  342. class orientation_2d;
  343. class direction_1d {
  344. private:
  345. unsigned int val_;
  346. explicit direction_1d(int d);
  347. public:
  348. inline direction_1d() : val_(LOW) {}
  349. inline direction_1d(const direction_1d& that) : val_(that.val_) {}
  350. inline direction_1d(const direction_1d_enum val) : val_(val) {}
  351. explicit inline direction_1d(const direction_2d& that);
  352. explicit inline direction_1d(const direction_3d& that);
  353. inline direction_1d& operator = (const direction_1d& d) {
  354. val_ = d.val_; return * this; }
  355. inline bool operator==(direction_1d d) const { return (val_ == d.val_); }
  356. inline bool operator!=(direction_1d d) const { return !((*this) == d); }
  357. inline unsigned int to_int(void) const { return val_; }
  358. inline direction_1d& backward() { val_ ^= 1; return *this; }
  359. inline int get_sign() const { return val_ * 2 - 1; }
  360. };
  361. class direction_2d;
  362. class orientation_2d {
  363. private:
  364. unsigned int val_;
  365. explicit inline orientation_2d(int o);
  366. public:
  367. inline orientation_2d() : val_(HORIZONTAL) {}
  368. inline orientation_2d(const orientation_2d& ori) : val_(ori.val_) {}
  369. inline orientation_2d(const orientation_2d_enum val) : val_(val) {}
  370. explicit inline orientation_2d(const direction_2d& that);
  371. inline orientation_2d& operator=(const orientation_2d& ori) {
  372. val_ = ori.val_; return * this; }
  373. inline bool operator==(orientation_2d that) const { return (val_ == that.val_); }
  374. inline bool operator!=(orientation_2d that) const { return (val_ != that.val_); }
  375. inline unsigned int to_int() const { return (val_); }
  376. inline void turn_90() { val_ = val_^ 1; }
  377. inline orientation_2d get_perpendicular() const {
  378. orientation_2d retval = *this;
  379. retval.turn_90();
  380. return retval;
  381. }
  382. inline direction_2d get_direction(direction_1d dir) const;
  383. };
  384. class direction_2d {
  385. private:
  386. int val_;
  387. public:
  388. inline direction_2d() : val_(WEST) {}
  389. inline direction_2d(const direction_2d& that) : val_(that.val_) {}
  390. inline direction_2d(const direction_2d_enum val) : val_(val) {}
  391. inline direction_2d& operator=(const direction_2d& d) {
  392. val_ = d.val_;
  393. return * this;
  394. }
  395. inline ~direction_2d() { }
  396. inline bool operator==(direction_2d d) const { return (val_ == d.val_); }
  397. inline bool operator!=(direction_2d d) const { return !((*this) == d); }
  398. inline bool operator< (direction_2d d) const { return (val_ < d.val_); }
  399. inline bool operator<=(direction_2d d) const { return (val_ <= d.val_); }
  400. inline bool operator> (direction_2d d) const { return (val_ > d.val_); }
  401. inline bool operator>=(direction_2d d) const { return (val_ >= d.val_); }
  402. // Casting to int
  403. inline unsigned int to_int(void) const { return val_; }
  404. inline direction_2d backward() const {
  405. // flip the LSB, toggles 0 - 1 and 2 - 3
  406. return direction_2d(direction_2d_enum(val_ ^ 1));
  407. }
  408. // Returns a direction 90 degree left (LOW) or right(HIGH) to this one
  409. inline direction_2d turn(direction_1d t) const {
  410. return direction_2d(direction_2d_enum(val_ ^ 3 ^ (val_ >> 1) ^ t.to_int()));
  411. }
  412. // Returns a direction 90 degree left to this one
  413. inline direction_2d left() const {return turn(HIGH);}
  414. // Returns a direction 90 degree right to this one
  415. inline direction_2d right() const {return turn(LOW);}
  416. // N, E are positive, S, W are negative
  417. inline bool is_positive() const {return (val_ & 1);}
  418. inline bool is_negative() const {return !is_positive();}
  419. inline int get_sign() const {return ((is_positive()) << 1) -1;}
  420. };
  421. direction_1d::direction_1d(const direction_2d& that) : val_(that.to_int() & 1) {}
  422. orientation_2d::orientation_2d(const direction_2d& that) : val_(that.to_int() >> 1) {}
  423. direction_2d orientation_2d::get_direction(direction_1d dir) const {
  424. return direction_2d(direction_2d_enum((val_ << 1) + dir.to_int()));
  425. }
  426. class orientation_3d {
  427. private:
  428. unsigned int val_;
  429. explicit inline orientation_3d(int o);
  430. public:
  431. inline orientation_3d() : val_((int)HORIZONTAL) {}
  432. inline orientation_3d(const orientation_3d& ori) : val_(ori.val_) {}
  433. inline orientation_3d(orientation_2d ori) : val_(ori.to_int()) {}
  434. inline orientation_3d(const orientation_3d_enum val) : val_(val) {}
  435. explicit inline orientation_3d(const direction_2d& that);
  436. explicit inline orientation_3d(const direction_3d& that);
  437. inline ~orientation_3d() { }
  438. inline orientation_3d& operator=(const orientation_3d& ori) {
  439. val_ = ori.val_; return * this; }
  440. inline bool operator==(orientation_3d that) const { return (val_ == that.val_); }
  441. inline bool operator!=(orientation_3d that) const { return (val_ != that.val_); }
  442. inline unsigned int to_int() const { return (val_); }
  443. inline direction_3d get_direction(direction_1d dir) const;
  444. };
  445. class direction_3d {
  446. private:
  447. int val_;
  448. public:
  449. inline direction_3d() : val_(WEST) {}
  450. inline direction_3d(direction_2d that) : val_(that.to_int()) {}
  451. inline direction_3d(const direction_3d& that) : val_(that.val_) {}
  452. inline direction_3d(const direction_2d_enum val) : val_(val) {}
  453. inline direction_3d(const direction_3d_enum val) : val_(val) {}
  454. inline direction_3d& operator=(direction_3d d) {
  455. val_ = d.val_;
  456. return * this;
  457. }
  458. inline ~direction_3d() { }
  459. inline bool operator==(direction_3d d) const { return (val_ == d.val_); }
  460. inline bool operator!=(direction_3d d) const { return !((*this) == d); }
  461. inline bool operator< (direction_3d d) const { return (val_ < d.val_); }
  462. inline bool operator<=(direction_3d d) const { return (val_ <= d.val_); }
  463. inline bool operator> (direction_3d d) const { return (val_ > d.val_); }
  464. inline bool operator>=(direction_3d d) const { return (val_ >= d.val_); }
  465. // Casting to int
  466. inline unsigned int to_int(void) const { return val_; }
  467. inline direction_3d backward() const {
  468. // flip the LSB, toggles 0 - 1 and 2 - 3 and 4 - 5
  469. return direction_2d(direction_2d_enum(val_ ^ 1));
  470. }
  471. // N, E, U are positive, S, W, D are negative
  472. inline bool is_positive() const {return (val_ & 1);}
  473. inline bool is_negative() const {return !is_positive();}
  474. inline int get_sign() const {return ((is_positive()) << 1) -1;}
  475. };
  476. direction_1d::direction_1d(const direction_3d& that) : val_(that.to_int() & 1) {}
  477. orientation_3d::orientation_3d(const direction_3d& that) : val_(that.to_int() >> 1) {}
  478. orientation_3d::orientation_3d(const direction_2d& that) : val_(that.to_int() >> 1) {}
  479. direction_3d orientation_3d::get_direction(direction_1d dir) const {
  480. return direction_3d(direction_3d_enum((val_ << 1) + dir.to_int()));
  481. }
  482. }
  483. }
  484. #endif