/*=============================================================================
Copyright (c) 2002-2003 Joel de Guzman
Copyright (c) 2002 Juan Carlos Arevalo-Baeza
Copyright (c) 2002-2003 Martin Wille
http://spirit.sourceforge.net/
Use, modification and distribution is subject to 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_IF_HPP
#define BOOST_SPIRIT_IF_HPP
#include
#include
#include
namespace boost { namespace spirit {
namespace impl {
//////////////////////////////////
// if-else-parser, holds two alternative parsers and a conditional functor
// that selects between them.
template
struct if_else_parser
: public condition_evaluator
, public binary
<
typename as_parser::type,
typename as_parser::type,
parser< if_else_parser >
>
{
typedef if_else_parser self_t;
typedef as_parser as_parser_true_t;
typedef as_parser as_parser_false_t;
typedef typename as_parser_true_t::type parser_true_t;
typedef typename as_parser_false_t::type parser_false_t;
typedef as_parser cond_as_parser_t;
typedef typename cond_as_parser_t::type condition_t;
typedef binary base_t;
typedef condition_evaluator eval_t;
if_else_parser
(
ParsableTrueT const& p_true,
ParsableFalseT const& p_false,
CondT const& cond_
)
: eval_t(cond_as_parser_t::convert(cond_))
, base_t
(
as_parser_true_t::convert(p_true),
as_parser_false_t::convert(p_false)
)
{ }
template
struct result
{
typedef typename match_result::type type;
};
template
typename parser_result::type
parse(ScannerT const& scan) const
{
typedef typename parser_result
::type then_result_t;
typedef typename parser_result
::type else_result_t;
typename ScannerT::iterator_t const save(scan.first);
std::ptrdiff_t length = this->evaluate(scan);
if (length >= 0)
{
then_result_t then_result(this->left().parse(scan));
if (then_result)
{
length += then_result.length();
return scan.create_match(std::size_t(length), nil_t(), save, scan.first);
}
}
else
{
else_result_t else_result(this->right().parse(scan));
if (else_result)
{
length = else_result.length();
return scan.create_match(std::size_t(length), nil_t(), save, scan.first);
}
}
return scan.no_match();
}
};
//////////////////////////////////
// if-else-parser generator, takes the false-parser in brackets
// and returns the if-else-parser.
template
struct if_else_parser_gen
{
if_else_parser_gen(ParsableTrueT const& p_true_, CondT const& cond_)
: p_true(p_true_)
, cond(cond_) {}
template
if_else_parser
<
ParsableTrueT,
ParsableFalseT,
CondT
>
operator[](ParsableFalseT const& p_false) const
{
return if_else_parser
(
p_true,
p_false,
cond
);
}
ParsableTrueT const &p_true;
CondT const &cond;
};
//////////////////////////////////
// if-parser, conditionally runs a parser is a functor condition is true.
// If the condition is fales, it fails the parse.
// It can optionally become an if-else-parser through the member else_p.
template
struct if_parser
: public condition_evaluator
, public unary
<
typename as_parser::type,
parser >
{
typedef if_parser self_t;
typedef as_parser as_parser_t;
typedef typename as_parser_t::type parser_t;
typedef as_parser cond_as_parser_t;
typedef typename cond_as_parser_t::type condition_t;
typedef condition_evaluator eval_t;
typedef unary base_t;
if_parser(ParsableT const& p, CondT const& cond_)
: eval_t(cond_as_parser_t::convert(cond_))
, base_t(as_parser_t::convert(p))
, else_p(p, cond_)
{}
template
struct result
{
typedef typename match_result::type type;
};
template
typename parser_result::type
parse(ScannerT const& scan) const
{
typedef typename parser_result::type t_result_t;
typename ScannerT::iterator_t const save(scan.first);
std::ptrdiff_t length = this->evaluate(scan);
if (length >= 0)
{
t_result_t then_result(this->subject().parse(scan));
if (then_result)
{
length += then_result.length();
return scan.create_match(std::size_t(length), nil_t(), save, scan.first);
}
return scan.no_match();
}
return scan.empty_match();
}
if_else_parser_gen else_p;
};
//////////////////////////////////
// if-parser generator, takes the true-parser in brackets and returns the
// if-parser.
template
struct if_parser_gen
{
if_parser_gen(CondT const& cond_) : cond(cond_) {}
template
if_parser
<
ParsableT,
CondT
>
operator[](ParsableT const& subject) const
{
return if_parser(subject, cond);
}
CondT const &cond;
};
} // namespace impl
//////////////////////////////////
// if_p function, returns "if" parser generator
template
impl::if_parser_gen
if_p(CondT const& cond)
{
return impl::if_parser_gen(cond);
}
}} // namespace boost::spirit
#endif // BOOST_SPIRIT_IF_HPP