charset_transforms.hpp

来自「support vector clustering for vc++」· HPP 代码 · 共 141 行

HPP
141
字号
///////////////////////////////////////////////////////////////////////////////
// charset_transforms.hpp
//
//  Copyright 2004 Eric Niebler. 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_XPRESSIVE_DETAIL_STATIC_PRODUCTIONS_CHARSET_TRANSFORMS_HPP_EAN_10_04_2005
#define BOOST_XPRESSIVE_DETAIL_STATIC_PRODUCTIONS_CHARSET_TRANSFORMS_HPP_EAN_10_04_2005

#include <boost/mpl/assert.hpp>
#include <boost/mpl/assert.hpp>
#include <boost/mpl/sizeof.hpp>
#include <boost/mpl/size_t.hpp>
#include <boost/xpressive/detail/detail_fwd.hpp>
#include <boost/xpressive/proto/proto.hpp>
#include <boost/xpressive/detail/utility/never_true.hpp>
#include <boost/xpressive/detail/utility/chset/chset.hpp>

namespace boost { namespace xpressive { namespace detail
{

    ///////////////////////////////////////////////////////////////////////////////
    // charset_state
    //
    template<typename CharSet, typename Traits>
    struct charset_state
    {
        typedef Traits traits_type;
        typedef typename CharSet::char_type char_type;
        typedef typename CharSet::icase_type icase_type;
      
        explicit charset_state(CharSet &charset, traits_type const &traits)
          : charset_(charset)
          , traits_(traits)
        {
        }

        template<bool Not>
        void set(literal_matcher<traits_type, icase_type::value, Not> const &ch) const
        {
            // BUGBUG fixme!
            BOOST_MPL_ASSERT_NOT((mpl::bool_<Not>));
            set_char(this->charset_.charset_, ch.ch_, this->traits_, icase_type());
        }

        void set(range_matcher<traits_type, icase_type::value> const &rg) const
        {
            // BUGBUG fixme!
            BOOST_ASSERT(!rg.not_);
            set_range(this->charset_.charset_, rg.ch_min_, rg.ch_max_, this->traits_, icase_type());
        }

        template<int Size>
        void set(set_matcher<traits_type, Size> const &set_) const
        {
            // BUGBUG fixme!
            BOOST_ASSERT(!set_.not_);
            for(int i=0; i<Size; ++i)
            {
                set_char(this->charset_.charset_, set_.set_[i], this->traits_, icase_type::value);
            }
        }

        void set(posix_charset_matcher<traits_type> const &posix) const
        {
            set_class(this->charset_.charset_, posix.mask_, posix.not_, this->traits_);
        }

        template<typename Unknown>
        void set(Unknown const &) const
        {
            // If this assert fires, it means that you have put something in a set[] that doesn't
            // belong there. For instance, set["hello"]. Legal members of sets are characters, 
            // character ranges, list-initialized sets such as (set='a','b','c') and posix-style
            // character sets such as digit and ~alpha.
            BOOST_MPL_ASSERT((never_true<Unknown>));
        }

    private:
        charset_state &operator =(charset_state const &);

        CharSet &charset_;
        traits_type const &traits_;
    };

    template<typename CharSet, typename Traits>
    charset_state<CharSet, Traits> make_charset_state(CharSet &charset, Traits const &traits)
    {
        return charset_state<CharSet, Traits>(charset, traits);
    }

    ///////////////////////////////////////////////////////////////////////////////
    //
    struct charset_transform
    {
        template<typename, typename, typename Visitor>
        struct apply
        {
            typedef typename Visitor::char_type char_type;

            // if sizeof(char_type)==1, merge everything into a basic_chset
            // BUGBUG this is not optimal.
            typedef typename mpl::if_
            <
                mpl::equal_to<mpl::sizeof_<char_type>, mpl::size_t<1> >
              , basic_chset<char_type>
              , compound_charset<typename Visitor::traits_type>
            >::type charset_type;

            typedef charset_matcher
            <
                typename Visitor::traits_type
              , Visitor::icase_type::value
              , charset_type
            > matcher_type;

            typedef proto::unary_op<matcher_type, proto::noop_tag> type;
        };

        template<typename Op, typename State, typename Visitor>
        static typename apply<Op, State, Visitor>::type
        call(Op const &op, State const &, Visitor &visitor, bool complement = false)
        {
            typedef typename apply<Op, State, Visitor>::matcher_type matcher_type;
            matcher_type matcher;
            // Walks the tree and fills in the charset
            proto::compile(proto::right(op), make_charset_state(matcher, visitor.traits()), visitor, set_tag());
            if(complement)
            {
                matcher.charset_.inverse();
            }
            return proto::noop(matcher);
        }
    };

}}}

#endif

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?