123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- /*=============================================================================
- Copyright (c) 2001-2014 Joel de Guzman
- Copyright (c) 2013-2014 Agustin Berge
- Distributed under the Boost Software License, Version 1.0. (See accompanying
- file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
- ==============================================================================*/
- #if !defined(BOOST_SPIRIT_X3_AUXILIARY_ANY_PARSER_APR_09_2014_1145PM)
- #define BOOST_SPIRIT_X3_AUXILIARY_ANY_PARSER_APR_09_2014_1145PM
- #include <boost/spirit/home/x3/core/parser.hpp>
- #include <boost/spirit/home/x3/support/context.hpp>
- #include <boost/spirit/home/x3/support/subcontext.hpp>
- #include <boost/spirit/home/x3/support/unused.hpp>
- #include <boost/spirit/home/x3/support/traits/container_traits.hpp>
- #include <boost/spirit/home/x3/support/traits/has_attribute.hpp>
- #include <boost/spirit/home/x3/support/traits/move_to.hpp>
- #include <boost/spirit/home/x3/support/traits/is_parser.hpp>
- #include <memory>
- #include <string>
- namespace boost { namespace spirit { namespace x3
- {
- template <
- typename Iterator
- , typename Attribute = unused_type
- , typename Context = subcontext<>>
- struct any_parser : parser<any_parser<Iterator, Attribute, Context>>
- {
- typedef Attribute attribute_type;
- static bool const has_attribute =
- !is_same<unused_type, attribute_type>::value;
- static bool const handles_container =
- traits::is_container<Attribute>::value;
- public:
- any_parser() = default;
- template <typename Expr,
- typename Enable = typename enable_if<traits::is_parser<Expr>>::type>
- any_parser(Expr const& expr)
- : _content(new holder<Expr>(expr)) {}
- any_parser(any_parser const& other)
- : _content(other._content ? other._content->clone() : nullptr) {}
- any_parser(any_parser&& other) = default;
- any_parser& operator=(any_parser const& other)
- {
- _content.reset(other._content ? other._content->clone() : nullptr);
- return *this;
- }
- any_parser& operator=(any_parser&& other) = default;
- template <typename Iterator_, typename Context_>
- bool parse(Iterator_& first, Iterator_ const& last
- , Context_ const& context, unused_type, Attribute& attr) const
- {
- BOOST_STATIC_ASSERT_MSG(
- (is_same<Iterator, Iterator_>::value)
- , "Incompatible iterator used"
- );
- BOOST_ASSERT_MSG(
- (_content != nullptr)
- , "Invalid use of uninitialized any_parser"
- );
- return _content->parse(first, last, context, attr);
- }
- template <typename Iterator_, typename Context_, typename Attribute_>
- bool parse(Iterator_& first, Iterator_ const& last
- , Context_ const& context, unused_type, Attribute_& attr_) const
- {
- Attribute attr;
- if (parse(first, last, context, unused, attr))
- {
- traits::move_to(attr, attr_);
- return true;
- }
- return false;
- }
- std::string get_info() const
- {
- return _content ? _content->get_info() : "";
- }
- private:
- struct placeholder
- {
- virtual placeholder* clone() const = 0;
- virtual bool parse(Iterator& first, Iterator const& last
- , Context const& context, Attribute& attr) const = 0;
- virtual std::string get_info() const = 0;
- virtual ~placeholder() {}
- };
- template <typename Expr>
- struct holder : placeholder
- {
- typedef typename extension::as_parser<Expr>::value_type parser_type;
- explicit holder(Expr const& p)
- : _parser(as_parser(p)) {}
- holder* clone() const override
- {
- return new holder(*this);
- }
- bool parse(Iterator& first, Iterator const& last
- , Context const& context, Attribute& attr) const override
- {
- return _parser.parse(first, last, context, unused, attr);
- }
- std::string get_info() const override
- {
- return x3::what(_parser);
- }
- parser_type _parser;
- };
- private:
- std::unique_ptr<placeholder> _content;
- };
- template <typename Iterator, typename Attribute, typename Context>
- struct get_info<any_parser<Iterator, Attribute, Context>>
- {
- typedef std::string result_type;
- std::string operator()(
- any_parser<Iterator, Attribute, Context> const& p) const
- {
- return p.get_info();
- }
- };
- }}}
- #endif
|