predicates.hpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425
  1. // Boost.Geometry Index
  2. //
  3. // Spatial query predicates
  4. //
  5. // Copyright (c) 2011-2015 Adam Wulkiewicz, Lodz, Poland.
  6. //
  7. // Use, modification and distribution is subject to the Boost Software License,
  8. // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  9. // http://www.boost.org/LICENSE_1_0.txt)
  10. #ifndef BOOST_GEOMETRY_INDEX_PREDICATES_HPP
  11. #define BOOST_GEOMETRY_INDEX_PREDICATES_HPP
  12. #include <utility>
  13. #include <boost/tuple/tuple.hpp>
  14. #include <boost/mpl/assert.hpp>
  15. #include <boost/geometry/index/detail/predicates.hpp>
  16. #include <boost/geometry/index/detail/tuples.hpp>
  17. /*!
  18. \defgroup predicates Predicates (boost::geometry::index::)
  19. */
  20. namespace boost { namespace geometry { namespace index {
  21. /*!
  22. \brief Generate \c contains() predicate.
  23. Generate a predicate defining Value and Geometry relationship.
  24. Value will be returned by the query if <tt>bg::within(Geometry, Indexable)</tt>
  25. returns true.
  26. \par Example
  27. \verbatim
  28. bgi::query(spatial_index, bgi::contains(box), std::back_inserter(result));
  29. \endverbatim
  30. \ingroup predicates
  31. \tparam Geometry The Geometry type.
  32. \param g The Geometry object.
  33. */
  34. template <typename Geometry> inline
  35. detail::predicates::spatial_predicate<Geometry, detail::predicates::contains_tag, false>
  36. contains(Geometry const& g)
  37. {
  38. return detail::predicates::spatial_predicate
  39. <
  40. Geometry,
  41. detail::predicates::contains_tag,
  42. false
  43. >(g);
  44. }
  45. /*!
  46. \brief Generate \c covered_by() predicate.
  47. Generate a predicate defining Value and Geometry relationship.
  48. Value will be returned by the query if <tt>bg::covered_by(Indexable, Geometry)</tt>
  49. returns true.
  50. \par Example
  51. \verbatim
  52. bgi::query(spatial_index, bgi::covered_by(box), std::back_inserter(result));
  53. \endverbatim
  54. \ingroup predicates
  55. \tparam Geometry The Geometry type.
  56. \param g The Geometry object.
  57. */
  58. template <typename Geometry> inline
  59. detail::predicates::spatial_predicate<Geometry, detail::predicates::covered_by_tag, false>
  60. covered_by(Geometry const& g)
  61. {
  62. return detail::predicates::spatial_predicate
  63. <
  64. Geometry,
  65. detail::predicates::covered_by_tag,
  66. false
  67. >(g);
  68. }
  69. /*!
  70. \brief Generate \c covers() predicate.
  71. Generate a predicate defining Value and Geometry relationship.
  72. Value will be returned by the query if <tt>bg::covered_by(Geometry, Indexable)</tt>
  73. returns true.
  74. \par Example
  75. \verbatim
  76. bgi::query(spatial_index, bgi::covers(box), std::back_inserter(result));
  77. \endverbatim
  78. \ingroup predicates
  79. \tparam Geometry The Geometry type.
  80. \param g The Geometry object.
  81. */
  82. template <typename Geometry> inline
  83. detail::predicates::spatial_predicate<Geometry, detail::predicates::covers_tag, false>
  84. covers(Geometry const& g)
  85. {
  86. return detail::predicates::spatial_predicate
  87. <
  88. Geometry,
  89. detail::predicates::covers_tag,
  90. false
  91. >(g);
  92. }
  93. /*!
  94. \brief Generate \c disjoint() predicate.
  95. Generate a predicate defining Value and Geometry relationship.
  96. Value will be returned by the query if <tt>bg::disjoint(Indexable, Geometry)</tt>
  97. returns true.
  98. \par Example
  99. \verbatim
  100. bgi::query(spatial_index, bgi::disjoint(box), std::back_inserter(result));
  101. \endverbatim
  102. \ingroup predicates
  103. \tparam Geometry The Geometry type.
  104. \param g The Geometry object.
  105. */
  106. template <typename Geometry> inline
  107. detail::predicates::spatial_predicate<Geometry, detail::predicates::disjoint_tag, false>
  108. disjoint(Geometry const& g)
  109. {
  110. return detail::predicates::spatial_predicate
  111. <
  112. Geometry,
  113. detail::predicates::disjoint_tag,
  114. false
  115. >(g);
  116. }
  117. /*!
  118. \brief Generate \c intersects() predicate.
  119. Generate a predicate defining Value and Geometry relationship.
  120. Value will be returned by the query if <tt>bg::intersects(Indexable, Geometry)</tt>
  121. returns true.
  122. \par Example
  123. \verbatim
  124. bgi::query(spatial_index, bgi::intersects(box), std::back_inserter(result));
  125. bgi::query(spatial_index, bgi::intersects(ring), std::back_inserter(result));
  126. bgi::query(spatial_index, bgi::intersects(polygon), std::back_inserter(result));
  127. \endverbatim
  128. \ingroup predicates
  129. \tparam Geometry The Geometry type.
  130. \param g The Geometry object.
  131. */
  132. template <typename Geometry> inline
  133. detail::predicates::spatial_predicate<Geometry, detail::predicates::intersects_tag, false>
  134. intersects(Geometry const& g)
  135. {
  136. return detail::predicates::spatial_predicate
  137. <
  138. Geometry,
  139. detail::predicates::intersects_tag,
  140. false
  141. >(g);
  142. }
  143. /*!
  144. \brief Generate \c overlaps() predicate.
  145. Generate a predicate defining Value and Geometry relationship.
  146. Value will be returned by the query if <tt>bg::overlaps(Indexable, Geometry)</tt>
  147. returns true.
  148. \par Example
  149. \verbatim
  150. bgi::query(spatial_index, bgi::overlaps(box), std::back_inserter(result));
  151. \endverbatim
  152. \ingroup predicates
  153. \tparam Geometry The Geometry type.
  154. \param g The Geometry object.
  155. */
  156. template <typename Geometry> inline
  157. detail::predicates::spatial_predicate<Geometry, detail::predicates::overlaps_tag, false>
  158. overlaps(Geometry const& g)
  159. {
  160. return detail::predicates::spatial_predicate
  161. <
  162. Geometry,
  163. detail::predicates::overlaps_tag,
  164. false
  165. >(g);
  166. }
  167. #ifdef BOOST_GEOMETRY_INDEX_DETAIL_EXPERIMENTAL
  168. /*!
  169. \brief Generate \c touches() predicate.
  170. Generate a predicate defining Value and Geometry relationship.
  171. Value will be returned by the query if <tt>bg::touches(Indexable, Geometry)</tt>
  172. returns true.
  173. \ingroup predicates
  174. \tparam Geometry The Geometry type.
  175. \param g The Geometry object.
  176. */
  177. template <typename Geometry> inline
  178. detail::predicates::spatial_predicate<Geometry, detail::predicates::touches_tag, false>
  179. touches(Geometry const& g)
  180. {
  181. return detail::predicates::spatial_predicate
  182. <
  183. Geometry,
  184. detail::predicates::touches_tag,
  185. false
  186. >(g);
  187. }
  188. #endif // BOOST_GEOMETRY_INDEX_DETAIL_EXPERIMENTAL
  189. /*!
  190. \brief Generate \c within() predicate.
  191. Generate a predicate defining Value and Geometry relationship.
  192. Value will be returned by the query if <tt>bg::within(Indexable, Geometry)</tt>
  193. returns true.
  194. \par Example
  195. \verbatim
  196. bgi::query(spatial_index, bgi::within(box), std::back_inserter(result));
  197. \endverbatim
  198. \ingroup predicates
  199. \tparam Geometry The Geometry type.
  200. \param g The Geometry object.
  201. */
  202. template <typename Geometry> inline
  203. detail::predicates::spatial_predicate<Geometry, detail::predicates::within_tag, false>
  204. within(Geometry const& g)
  205. {
  206. return detail::predicates::spatial_predicate
  207. <
  208. Geometry,
  209. detail::predicates::within_tag,
  210. false
  211. >(g);
  212. }
  213. /*!
  214. \brief Generate satisfies() predicate.
  215. A wrapper around user-defined UnaryPredicate checking if Value should be returned by spatial query.
  216. \par Example
  217. \verbatim
  218. bool is_red(Value const& v) { return v.is_red(); }
  219. struct is_red_o {
  220. template <typename Value> bool operator()(Value const& v) { return v.is_red(); }
  221. }
  222. // ...
  223. rt.query(index::intersects(box) && index::satisfies(is_red),
  224. std::back_inserter(result));
  225. rt.query(index::intersects(box) && index::satisfies(is_red_o()),
  226. std::back_inserter(result));
  227. #ifndef BOOST_NO_CXX11_LAMBDAS
  228. rt.query(index::intersects(box) && index::satisfies([](Value const& v) { return v.is_red(); }),
  229. std::back_inserter(result));
  230. #endif
  231. \endverbatim
  232. \ingroup predicates
  233. \tparam UnaryPredicate A type of unary predicate function or function object.
  234. \param pred The unary predicate function or function object.
  235. */
  236. template <typename UnaryPredicate> inline
  237. detail::predicates::satisfies<UnaryPredicate, false>
  238. satisfies(UnaryPredicate const& pred)
  239. {
  240. return detail::predicates::satisfies<UnaryPredicate, false>(pred);
  241. }
  242. /*!
  243. \brief Generate nearest() predicate.
  244. When nearest predicate is passed to the query, k-nearest neighbour search will be performed.
  245. \c nearest() predicate takes a \c Geometry from which distances to \c Values are calculated
  246. and the maximum number of \c Values that should be returned. Internally
  247. boost::geometry::comparable_distance() is used to perform the calculation.
  248. \par Example
  249. \verbatim
  250. bgi::query(spatial_index, bgi::nearest(pt, 5), std::back_inserter(result));
  251. bgi::query(spatial_index, bgi::nearest(pt, 5) && bgi::intersects(box), std::back_inserter(result));
  252. bgi::query(spatial_index, bgi::nearest(box, 5), std::back_inserter(result));
  253. \endverbatim
  254. \warning
  255. Only one \c nearest() predicate may be used in a query.
  256. \ingroup predicates
  257. \param geometry The geometry from which distance is calculated.
  258. \param k The maximum number of values to return.
  259. */
  260. template <typename Geometry> inline
  261. detail::predicates::nearest<Geometry>
  262. nearest(Geometry const& geometry, unsigned k)
  263. {
  264. return detail::predicates::nearest<Geometry>(geometry, k);
  265. }
  266. #ifdef BOOST_GEOMETRY_INDEX_DETAIL_EXPERIMENTAL
  267. /*!
  268. \brief Generate path() predicate.
  269. When path predicate is passed to the query, the returned values are k values along the path closest to
  270. its begin. \c path() predicate takes a \c Segment or a \c Linestring defining the path and the maximum
  271. number of \c Values that should be returned.
  272. \par Example
  273. \verbatim
  274. bgi::query(spatial_index, bgi::path(segment, 5), std::back_inserter(result));
  275. bgi::query(spatial_index, bgi::path(linestring, 5) && bgi::intersects(box), std::back_inserter(result));
  276. \endverbatim
  277. \warning
  278. Only one distance predicate (\c nearest() or \c path()) may be used in a query.
  279. \ingroup predicates
  280. \param linestring The path along which distance is calculated.
  281. \param k The maximum number of values to return.
  282. */
  283. template <typename SegmentOrLinestring> inline
  284. detail::predicates::path<SegmentOrLinestring>
  285. path(SegmentOrLinestring const& linestring, unsigned k)
  286. {
  287. return detail::predicates::path<SegmentOrLinestring>(linestring, k);
  288. }
  289. #endif // BOOST_GEOMETRY_INDEX_DETAIL_EXPERIMENTAL
  290. namespace detail { namespace predicates {
  291. // operator! generators
  292. template <typename Fun, bool Negated> inline
  293. satisfies<Fun, !Negated>
  294. operator!(satisfies<Fun, Negated> const& p)
  295. {
  296. return satisfies<Fun, !Negated>(p);
  297. }
  298. template <typename Geometry, typename Tag, bool Negated> inline
  299. spatial_predicate<Geometry, Tag, !Negated>
  300. operator!(spatial_predicate<Geometry, Tag, Negated> const& p)
  301. {
  302. return spatial_predicate<Geometry, Tag, !Negated>(p.geometry);
  303. }
  304. // operator&& generators
  305. template <typename Pred1, typename Pred2> inline
  306. boost::tuples::cons<
  307. Pred1,
  308. boost::tuples::cons<Pred2, boost::tuples::null_type>
  309. >
  310. operator&&(Pred1 const& p1, Pred2 const& p2)
  311. {
  312. /*typedef typename boost::mpl::if_c<is_predicate<Pred1>::value, Pred1, Pred1 const&>::type stored1;
  313. typedef typename boost::mpl::if_c<is_predicate<Pred2>::value, Pred2, Pred2 const&>::type stored2;*/
  314. namespace bt = boost::tuples;
  315. return
  316. bt::cons< Pred1, bt::cons<Pred2, bt::null_type> >
  317. ( p1, bt::cons<Pred2, bt::null_type>(p2, bt::null_type()) );
  318. }
  319. template <typename Head, typename Tail, typename Pred> inline
  320. typename tuples::push_back<
  321. boost::tuples::cons<Head, Tail>, Pred
  322. >::type
  323. operator&&(boost::tuples::cons<Head, Tail> const& t, Pred const& p)
  324. {
  325. //typedef typename boost::mpl::if_c<is_predicate<Pred>::value, Pred, Pred const&>::type stored;
  326. namespace bt = boost::tuples;
  327. return
  328. tuples::push_back<
  329. bt::cons<Head, Tail>, Pred
  330. >::apply(t, p);
  331. }
  332. }} // namespace detail::predicates
  333. }}} // namespace boost::geometry::index
  334. #endif // BOOST_GEOMETRY_INDEX_PREDICATES_HPP