Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
5 : * This Source Code Form is subject to the terms of the Mozilla Public
6 : * License, v. 2.0. If a copy of the MPL was not distributed with this
7 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 : *
9 : * This file incorporates work covered by the following license notice:
10 : *
11 : * Licensed to the Apache Software Foundation (ASF) under one or more
12 : * contributor license agreements. See the NOTICE file distributed
13 : * with this work for additional information regarding copyright
14 : * ownership. The ASF licenses this file to you under the Apache
15 : * License, Version 2.0 (the "License"); you may not use this file
16 : * except in compliance with the License. You may obtain a copy of
17 : * the License at http://www.apache.org/licenses/LICENSE-2.0 .
18 : */
19 :
20 : #ifndef INCLUDED_BASEGFX_RANGE_BASICRANGE_HXX
21 : #define INCLUDED_BASEGFX_RANGE_BASICRANGE_HXX
22 :
23 : #include <sal/types.h>
24 : #include <float.h>
25 : #include <basegfx/numeric/ftools.hxx>
26 :
27 :
28 : namespace basegfx
29 : {
30 : template< typename T, typename Traits > class BasicRange
31 : {
32 : protected:
33 : T mnMinimum;
34 : T mnMaximum;
35 :
36 : public:
37 : typedef T ValueType;
38 : typedef Traits TraitsType;
39 :
40 0 : BasicRange() :
41 0 : mnMinimum(Traits::maxVal()),
42 0 : mnMaximum(Traits::minVal())
43 : {
44 0 : }
45 :
46 0 : explicit BasicRange( T nValue ) :
47 : mnMinimum(nValue),
48 0 : mnMaximum(nValue)
49 : {
50 0 : }
51 :
52 0 : void reset()
53 : {
54 0 : mnMinimum = Traits::maxVal();
55 0 : mnMaximum = Traits::minVal();
56 0 : }
57 :
58 0 : bool isEmpty() const
59 : {
60 0 : return Traits::maxVal() == mnMinimum;
61 : }
62 :
63 0 : T getMinimum() const { return mnMinimum; }
64 0 : T getMaximum() const { return mnMaximum; }
65 :
66 0 : double getCenter() const
67 : {
68 0 : if(isEmpty())
69 : {
70 0 : return 0.0;
71 : }
72 : else
73 : {
74 0 : return ((mnMaximum + mnMinimum) / 2.0);
75 : }
76 : }
77 :
78 0 : bool isInside(T nValue) const
79 : {
80 0 : if(isEmpty())
81 : {
82 0 : return false;
83 : }
84 : else
85 : {
86 0 : return (nValue >= mnMinimum) && (nValue <= mnMaximum);
87 : }
88 : }
89 :
90 0 : bool isInside(const BasicRange& rRange) const
91 : {
92 0 : if(isEmpty())
93 : {
94 0 : return false;
95 : }
96 : else
97 : {
98 0 : if(rRange.isEmpty())
99 : {
100 0 : return false;
101 : }
102 : else
103 : {
104 0 : return (rRange.mnMinimum >= mnMinimum) && (rRange.mnMaximum <= mnMaximum);
105 : }
106 : }
107 : }
108 :
109 0 : bool overlaps(const BasicRange& rRange) const
110 : {
111 0 : if(isEmpty())
112 : {
113 0 : return false;
114 : }
115 : else
116 : {
117 0 : if(rRange.isEmpty())
118 : {
119 0 : return false;
120 : }
121 : else
122 : {
123 0 : return !((rRange.mnMaximum < mnMinimum) || (rRange.mnMinimum > mnMaximum));
124 : }
125 : }
126 : }
127 :
128 0 : bool overlapsMore(const BasicRange& rRange) const
129 : {
130 0 : if(isEmpty() || rRange.isEmpty())
131 0 : return false;
132 : // returns true if the overlap is more than just a touching at the limits
133 0 : return ((rRange.mnMaximum > mnMinimum) && (rRange.mnMinimum < mnMaximum));
134 : }
135 :
136 0 : bool operator==( const BasicRange& rRange ) const
137 : {
138 0 : return (mnMinimum == rRange.mnMinimum && mnMaximum == rRange.mnMaximum);
139 : }
140 :
141 0 : bool operator!=( const BasicRange& rRange ) const
142 : {
143 0 : return (mnMinimum != rRange.mnMinimum || mnMaximum != rRange.mnMaximum);
144 : }
145 :
146 0 : bool equal(const BasicRange& rRange) const
147 : {
148 : return (
149 0 : fTools::equal(mnMinimum, rRange.mnMinimum) &&
150 0 : fTools::equal(mnMaximum, rRange.mnMaximum));
151 : }
152 :
153 0 : void expand(T nValue)
154 : {
155 0 : if(isEmpty())
156 : {
157 0 : mnMinimum = mnMaximum = nValue;
158 : }
159 : else
160 : {
161 0 : if(nValue < mnMinimum)
162 : {
163 0 : mnMinimum = nValue;
164 : }
165 :
166 0 : if(nValue > mnMaximum)
167 : {
168 0 : mnMaximum = nValue;
169 : }
170 : }
171 0 : }
172 :
173 0 : void expand(const BasicRange& rRange)
174 : {
175 0 : if(isEmpty())
176 : {
177 0 : mnMinimum = rRange.mnMinimum;
178 0 : mnMaximum = rRange.mnMaximum;
179 : }
180 : else
181 : {
182 0 : if(!rRange.isEmpty())
183 : {
184 0 : if(rRange.mnMinimum < mnMinimum)
185 : {
186 0 : mnMinimum = rRange.mnMinimum;
187 : }
188 :
189 0 : if(rRange.mnMaximum > mnMaximum)
190 : {
191 0 : mnMaximum = rRange.mnMaximum;
192 : }
193 : }
194 : }
195 0 : }
196 :
197 0 : void intersect(const BasicRange& rRange)
198 : {
199 : // here, overlaps also tests all isEmpty() conditions already.
200 0 : if( !overlaps( rRange ) )
201 : {
202 0 : reset();
203 : }
204 : else
205 : {
206 0 : if(rRange.mnMinimum > mnMinimum)
207 : {
208 0 : mnMinimum = rRange.mnMinimum;
209 : }
210 :
211 0 : if(rRange.mnMaximum < mnMaximum)
212 : {
213 0 : mnMaximum = rRange.mnMaximum;
214 : }
215 : }
216 0 : }
217 :
218 0 : void grow(T nValue)
219 : {
220 0 : if(!isEmpty())
221 : {
222 0 : bool bLessThanZero(nValue < 0);
223 :
224 0 : if(nValue > 0 || bLessThanZero)
225 : {
226 0 : mnMinimum -= nValue;
227 0 : mnMaximum += nValue;
228 :
229 0 : if(bLessThanZero)
230 : {
231 : // test if range did collapse
232 0 : if(mnMinimum > mnMaximum)
233 : {
234 : // if yes, collapse to center
235 0 : mnMinimum = mnMaximum = (mnMinimum + mnMaximum) / 2;
236 : }
237 : }
238 : }
239 : }
240 0 : }
241 :
242 0 : typename Traits::DifferenceType getRange() const
243 : {
244 0 : if(isEmpty())
245 : {
246 0 : return Traits::neutral();
247 : }
248 : else
249 : {
250 0 : return (mnMaximum - mnMinimum);
251 : }
252 : }
253 : };
254 :
255 : // some pre-fabricated traits
256 : struct DoubleTraits
257 : {
258 0 : static double minVal() { return DBL_MIN; };
259 0 : static double maxVal() { return DBL_MAX; };
260 0 : static double neutral() { return 0.0; };
261 :
262 : typedef double DifferenceType;
263 : };
264 :
265 : struct Int32Traits
266 : {
267 0 : static sal_Int32 minVal() { return SAL_MIN_INT32; };
268 0 : static sal_Int32 maxVal() { return SAL_MAX_INT32; };
269 0 : static sal_Int32 neutral() { return 0L; };
270 :
271 : typedef sal_Int64 DifferenceType;
272 : };
273 :
274 : } // end of namespace basegfx
275 :
276 : #endif // INCLUDED_BASEGFX_RANGE_BASICRANGE_HXX
277 :
278 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|