source: NonGTP/Boost/boost/spirit/utility/impl/chset/range_run.ipp @ 857

Revision 857, 6.9 KB checked in by igarcia, 19 years ago (diff)
Line 
1/*=============================================================================
2    Copyright (c) 2001-2003 Joel de Guzman
3    http://spirit.sourceforge.net/
4
5    Use, modification and distribution is subject to the Boost Software
6    License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
7    http://www.boost.org/LICENSE_1_0.txt)
8=============================================================================*/
9#ifndef BOOST_SPIRIT_RANGE_RUN_IPP
10#define BOOST_SPIRIT_RANGE_RUN_IPP
11
12///////////////////////////////////////////////////////////////////////////////
13#include <algorithm> // for std::lower_bound
14#include <boost/spirit/core/assert.hpp> // for BOOST_SPIRIT_ASSERT
15#include <boost/spirit/utility/impl/chset/range_run.hpp>
16#include <boost/spirit/debug.hpp>
17#include <boost/limits.hpp>
18
19///////////////////////////////////////////////////////////////////////////////
20namespace boost { namespace spirit {
21
22    namespace utility { namespace impl {
23
24        ///////////////////////////////////////////////////////////////////////
25        //
26        //  range class implementation
27        //
28        ///////////////////////////////////////////////////////////////////////
29        template <typename CharT>
30        inline range<CharT>::range(CharT first_, CharT last_)
31        : first(first_), last(last_) {}
32
33        //////////////////////////////////
34        template <typename CharT>
35        inline bool
36        range<CharT>::is_valid() const
37        { return first <= last; }
38
39        //////////////////////////////////
40        template <typename CharT>
41        inline bool
42        range<CharT>::includes(range const& r) const
43        { return (first <= r.first) && (last >= r.last); }
44
45        //////////////////////////////////
46        template <typename CharT>
47        inline bool
48        range<CharT>::includes(CharT v) const
49        { return (first <= v) && (last >= v); }
50
51        //////////////////////////////////
52        template <typename CharT>
53        inline bool
54        range<CharT>::overlaps(range const& r) const
55        {
56            CharT decr_first =
57                first == (std::numeric_limits<CharT>::min)() ? first : first-1;
58            CharT incr_last =
59                last == (std::numeric_limits<CharT>::max)() ? last : last+1;
60
61            return (decr_first <= r.last) && (incr_last >= r.first);
62        }
63
64        //////////////////////////////////
65        template <typename CharT>
66        inline void
67        range<CharT>::merge(range const& r)
68        {
69            first = (std::min)(first, r.first);
70            last = (std::max)(last, r.last);
71        }
72
73        ///////////////////////////////////////////////////////////////////////
74        //
75        //  range_run class implementation
76        //
77        ///////////////////////////////////////////////////////////////////////
78        template <typename CharT>
79        inline bool
80        range_run<CharT>::test(CharT v) const
81        {
82            if (!run.empty())
83            {
84                const_iterator iter =
85                    std::lower_bound(
86                        run.begin(), run.end(), v,
87                        range_char_compare<CharT>()
88                    );
89
90                if (iter != run.end() && iter->includes(v))
91                    return true;
92                if (iter != run.begin())
93                    return (--iter)->includes(v);
94            }
95            return false;
96        }
97
98        //////////////////////////////////
99        template <typename CharT>
100        inline void
101        range_run<CharT>::swap(range_run& rr)
102        { run.swap(rr.run); }
103
104        //////////////////////////////////
105        template <typename CharT>
106        void
107        range_run<CharT>::merge(iterator iter, range<CharT> const& r)
108        {
109            iter->merge(r);
110            iterator i = iter + 1;
111
112            while (i != run.end() && iter->overlaps(*i))
113                iter->merge(*i++);
114
115            run.erase(iter+1, i);
116        }
117
118        //////////////////////////////////
119        template <typename CharT>
120        void
121        range_run<CharT>::set(range<CharT> const& r)
122        {
123            BOOST_SPIRIT_ASSERT(r.is_valid());
124            if (!run.empty())
125            {
126                iterator iter =
127                    std::lower_bound(
128                        run.begin(), run.end(), r,
129                        range_compare<CharT>()
130                    );
131
132                if (iter != run.end() && iter->includes(r) ||
133                    ((iter != run.begin()) && (iter - 1)->includes(r)))
134                    return;
135
136                if (iter != run.begin() && (iter - 1)->overlaps(r))
137                    merge(--iter, r);
138
139                else if (iter != run.end() && iter->overlaps(r))
140                    merge(iter, r);
141
142                else
143                    run.insert(iter, r);
144            }
145            else
146            {
147                run.push_back(r);
148            }
149        }
150
151        //////////////////////////////////
152        template <typename CharT>
153        void
154        range_run<CharT>::clear(range<CharT> const& r)
155        {
156            BOOST_SPIRIT_ASSERT(r.is_valid());
157            if (!run.empty())
158            {
159                iterator iter =
160                    std::lower_bound(
161                        run.begin(), run.end(), r,
162                        range_compare<CharT>()
163                    );
164
165                iterator left_iter;
166
167                if ((iter != run.begin()) &&
168                        (left_iter = (iter - 1))->includes(r.first))
169                    if (left_iter->last > r.last)
170                    {
171                        CharT save_last = left_iter->last;
172                        left_iter->last = r.first-1;
173                        run.insert(iter, range<CharT>(r.last+1, save_last));
174                        return;
175                    }
176                    else
177                    {
178                        left_iter->last = r.first-1;
179                    }
180
181                iterator i = iter;
182                while (i != run.end() && r.includes(*i))
183                    i++;
184                if (i != run.end() && i->includes(r.last))
185                    i->first = r.last+1;
186                run.erase(iter, i);
187            }
188        }
189
190        //////////////////////////////////
191        template <typename CharT>
192        inline void
193        range_run<CharT>::clear()
194        { run.clear(); }
195
196        //////////////////////////////////
197        template <typename CharT>
198        inline typename range_run<CharT>::const_iterator
199        range_run<CharT>::begin() const
200        { return run.begin(); }
201
202        //////////////////////////////////
203        template <typename CharT>
204        inline typename range_run<CharT>::const_iterator
205        range_run<CharT>::end() const
206        { return run.end(); }
207
208    }} // namespace utility::impl
209
210}} // namespace boost::spirit
211
212#endif
Note: See TracBrowser for help on using the repository browser.