| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189 | /*=============================================================================    Copyright (c) 2002-2003 Joel de Guzman    Copyright (c) 2002-2003 Martin Wille    http://spirit.sourceforge.net/  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)=============================================================================*/#ifndef BOOST_SPIRIT_WHILE_HPP#define BOOST_SPIRIT_WHILE_HPP#include <boost/spirit/home/classic/namespace.hpp>#include <boost/spirit/home/classic/core/parser.hpp>#include <boost/spirit/home/classic/core/composite/composite.hpp>#include <boost/spirit/home/classic/dynamic/impl/conditions.ipp>////////////////////////////////////////////////////////////////////////////////namespace boost { namespace spirit {BOOST_SPIRIT_CLASSIC_NAMESPACE_BEGIN    namespace impl {    //////////////////////////////////    // while parser    // object are created by while_parser_gen and do_parser_gen    template <typename ParsableT, typename CondT, bool is_do_parser>    struct while_parser        : public condition_evaluator< typename as_parser<CondT>::type >        , public unary // the parent stores a copy of the body parser        <            typename as_parser<ParsableT>::type,            parser<while_parser<ParsableT, CondT, is_do_parser> >        >    {        typedef while_parser<ParsableT, CondT, is_do_parser> self_t;        typedef as_parser<ParsableT>            as_parser_t;        typedef typename as_parser_t::type      parser_t;        typedef as_parser<CondT>                cond_as_parser_t;        typedef typename cond_as_parser_t::type condition_t;        typedef unary<parser_t, parser<self_t> > base_t;        typedef condition_evaluator<condition_t> eval_t;        //////////////////////////////        // constructor, saves condition and body parser        while_parser(ParsableT const &body, CondT const &cond)            : eval_t(cond_as_parser_t::convert(cond))            , base_t(as_parser_t::convert(body))        {}        //////////////////////////////        // result type computer.        template <typename ScannerT>        struct result        {            typedef typename match_result                <ScannerT, nil_t>::type type;        };        //////////////////////////////        // parse member function        template <typename ScannerT>        typename parser_result<self_t, ScannerT>::type        parse(ScannerT const& scan) const        {            typedef typename parser_result<parser_t, ScannerT>::type sresult_t;            typedef typename ScannerT::iterator_t                    iterator_t;            iterator_t save(scan.first);            std::size_t length = 0;            std::ptrdiff_t eval_length = 0;            bool dont_check_condition = is_do_parser;            while (dont_check_condition || (eval_length=this->evaluate(scan))>=0)            {                dont_check_condition = false;                length += eval_length;                sresult_t tmp(this->subject().parse(scan));                if (tmp)                {                    length+=tmp.length();                }                else                {                    return scan.no_match();                }            }            return scan.create_match(length, nil_t(), save, scan.first);        }    };    //////////////////////////////////    // while-parser generator, takes the body-parser in brackets    // and returns the actual while-parser.    template <typename CondT>    struct while_parser_gen    {        //////////////////////////////        // constructor, saves the condition for use by operator[]        while_parser_gen(CondT const& cond_) : cond(cond_) {}        //////////////////////////////        // operator[] returns the actual while-parser object        template <typename ParsableT>        while_parser<ParsableT, CondT, false>        operator[](ParsableT const &subject) const        {            return while_parser<ParsableT, CondT, false>(subject, cond);        }    private:        //////////////////////////////        // the condition is stored by reference here.        // this should not cause any harm since object of type        // while_parser_gen<> are only used as temporaries        // the while-parser object constructed by the operator[]        // stores a copy of the condition.        CondT const &cond;    };    //////////////////////////////////    // do-while-parser generator, takes the condition as    // parameter to while_p member function and returns the    // actual do-while-parser.    template <typename ParsableT>    struct do_while_parser_gen    {        //////////////////////////////        // constructor. saves the body parser for use by while_p.        explicit do_while_parser_gen(ParsableT const &body_parser)            : body(body_parser)        {}        //////////////////////////////        // while_p returns the actual while-parser object        template <typename CondT>        while_parser<ParsableT, CondT, true>        while_p(CondT cond) const        {            return while_parser<ParsableT, CondT, true>(body, cond);        }    private:        //////////////////////////////        // the body is stored by reference here        // this should not cause any harm since object of type        // do_while_parser_gen<> are only used as temporaries        // the while-parser object constructed by the while_p        // member function stores a copy of the body parser.        ParsableT const &body;    };    struct do_parser_gen    {        inline do_parser_gen() {}        template <typename ParsableT>        impl::do_while_parser_gen<ParsableT>        operator[](ParsableT const& body) const        {            return impl::do_while_parser_gen<ParsableT>(body);        }    };} // namespace impl//////////////////////////////////// while_p function, while-parser generator// Usage: spirit::while_p(Condition)[Body]template <typename CondT>impl::while_parser_gen<CondT>while_p(CondT const& cond){    return impl::while_parser_gen<CondT>(cond);}//////////////////////////////////// do_p functor, do-while-parser generator// Usage: spirit::do_p[Body].while_p(Condition)impl::do_parser_gen const do_p = impl::do_parser_gen();BOOST_SPIRIT_CLASSIC_NAMESPACE_END}} // namespace BOOST_SPIRIT_CLASSIC_NS#endif // BOOST_SPIRIT_WHILE_HPP
 |