skip_over.hpp 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. /*=============================================================================
  2. Copyright (c) 2001-2014 Joel de Guzman
  3. Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. ==============================================================================*/
  6. #if !defined(BOOST_SPIRIT_X3_SKIP_APRIL_16_2006_0625PM)
  7. #define BOOST_SPIRIT_X3_SKIP_APRIL_16_2006_0625PM
  8. #include <boost/spirit/home/x3/support/unused.hpp>
  9. #include <boost/spirit/home/x3/support/context.hpp>
  10. #include <boost/spirit/home/x3/support/traits/attribute_category.hpp>
  11. #include <boost/mpl/bool.hpp>
  12. #include <boost/mpl/not.hpp>
  13. #include <boost/type_traits/remove_cv.hpp>
  14. #include <boost/type_traits/remove_reference.hpp>
  15. #include <boost/utility/declval.hpp>
  16. namespace boost { namespace spirit { namespace x3
  17. {
  18. ///////////////////////////////////////////////////////////////////////////
  19. // Move the /first/ iterator to the first non-matching position
  20. // given a skip-parser. The function is a no-op if unused_type or
  21. // unused_skipper is passed as the skip-parser.
  22. ///////////////////////////////////////////////////////////////////////////
  23. template <typename Skipper>
  24. struct unused_skipper : unused_type
  25. {
  26. unused_skipper(Skipper const& skipper)
  27. : skipper(skipper) {}
  28. Skipper const& skipper;
  29. };
  30. namespace detail
  31. {
  32. template <typename Skipper>
  33. struct is_unused_skipper
  34. : mpl::false_ {};
  35. template <typename Skipper>
  36. struct is_unused_skipper<unused_skipper<Skipper>>
  37. : mpl::true_ {};
  38. template <>
  39. struct is_unused_skipper<unused_type>
  40. : mpl::true_ {};
  41. template <typename Skipper>
  42. inline Skipper const&
  43. get_unused_skipper(Skipper const& skipper)
  44. {
  45. return skipper;
  46. }
  47. template <typename Skipper>
  48. inline Skipper const&
  49. get_unused_skipper(unused_skipper<Skipper> const& unused_skipper)
  50. {
  51. return unused_skipper.skipper;
  52. }
  53. template <typename Iterator, typename Skipper>
  54. inline void skip_over(
  55. Iterator& first, Iterator const& last, Skipper const& skipper)
  56. {
  57. while (skipper.parse(first, last, unused, unused, unused))
  58. /***/;
  59. }
  60. template <typename Iterator>
  61. inline void skip_over(Iterator&, Iterator const&, unused_type)
  62. {
  63. }
  64. template <typename Iterator, typename Skipper>
  65. inline void skip_over(
  66. Iterator&, Iterator const&, unused_skipper<Skipper> const&)
  67. {
  68. }
  69. }
  70. // this tag is used to find the skipper from the context
  71. struct skipper_tag;
  72. template <typename Context>
  73. struct has_skipper
  74. : mpl::not_<detail::is_unused_skipper<
  75. typename remove_cv<typename remove_reference<
  76. decltype(x3::get<skipper_tag>(boost::declval<Context>()))
  77. >::type>::type
  78. >> {};
  79. template <typename Iterator, typename Context>
  80. inline void skip_over(
  81. Iterator& first, Iterator const& last, Context const& context)
  82. {
  83. detail::skip_over(first, last, x3::get<skipper_tag>(context));
  84. }
  85. }}}
  86. #endif