// Copyright (c) 2001-2011 Hartmut Kaiser // // 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(SPIRIT_KARMA_BUFFER_AUG_03_2009_0949AM) #define SPIRIT_KARMA_BUFFER_AUG_03_2009_0949AM #if defined(_MSC_VER) #pragma once #endif #include #include #include #include #include #include #include #include #include #include namespace boost { namespace spirit { /////////////////////////////////////////////////////////////////////////// // Enablers /////////////////////////////////////////////////////////////////////////// template <> struct use_directive // enables buffer : mpl::true_ {}; }} namespace boost { namespace spirit { namespace karma { #ifndef BOOST_SPIRIT_NO_PREDEFINED_TERMINALS using spirit::buffer; #endif using spirit::buffer_type; /////////////////////////////////////////////////////////////////////////// // buffer_directive buffers all generated output of the embedded generator // and flushes it only if the whole embedded generator succeeds /////////////////////////////////////////////////////////////////////////// template struct buffer_directive : unary_generator > { typedef Subject subject_type; typedef mpl::int_< subject_type::properties::value | generator_properties::countingbuffer > properties; buffer_directive(Subject const& subject) : subject(subject) {} template struct attribute : traits::attribute_of {}; template bool generate(OutputIterator& sink, Context& ctx, Delimiter const& d , Attribute const& attr) const { // wrap the given output iterator to avoid output as long as the // embedded generator (subject) fails detail::enable_buffering buffering(sink); bool r = false; { detail::disable_counting nocounting(sink); r = subject.generate(sink, ctx, d, attr); } if (r) buffering.buffer_copy(); return r; } template info what(Context& context) const { return info("buffer", subject.what(context)); } Subject subject; }; /////////////////////////////////////////////////////////////////////////// // Generator generators: make_xxx function (objects) /////////////////////////////////////////////////////////////////////////// template struct make_directive { typedef buffer_directive result_type; result_type operator()(unused_type, Subject const& subject , unused_type) const { return result_type(subject); } }; // make sure buffer[buffer[...]] does not result in double buffering template struct make_directive, Modifiers> { typedef buffer_directive result_type; result_type operator()(unused_type , buffer_directive const& subject, unused_type) const { return subject; } }; }}} namespace boost { namespace spirit { namespace traits { /////////////////////////////////////////////////////////////////////////// template struct has_semantic_action > : unary_has_semantic_action {}; /////////////////////////////////////////////////////////////////////////// template struct handles_container, Attribute , Context, Iterator> : unary_handles_container {}; }}} #endif