sl@0
|
1 |
// Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#include <e32def.h>
|
sl@0
|
17 |
#include "regionextend.h"
|
sl@0
|
18 |
|
sl@0
|
19 |
//Only call this if certain neither rectangle is empty.
|
sl@0
|
20 |
//Region rectangles are always non-empty
|
sl@0
|
21 |
inline /*static*/
|
sl@0
|
22 |
TRegionExtend::TOverlapFlags TestDifferenceNotEmpty(const TRect& aThis,const TRect& aThat)
|
sl@0
|
23 |
{
|
sl@0
|
24 |
struct SubAdd
|
sl@0
|
25 |
{
|
sl@0
|
26 |
inline static TRegionExtend::TOverlapFlags Conv(TInt delta)
|
sl@0
|
27 |
{ //returns sub, exact or add for delta of each edge pair
|
sl@0
|
28 |
//neg --> +1 //pos --> +2 //zero ==> 0
|
sl@0
|
29 |
return TRegionExtend::TOverlapFlags(
|
sl@0
|
30 |
((delta>>31)&1) +(((-delta)>>30)&2) );
|
sl@0
|
31 |
}
|
sl@0
|
32 |
};
|
sl@0
|
33 |
//could use SubAdd for this if...
|
sl@0
|
34 |
if ( aThis.iTl.iX>=aThat.iBr.iX
|
sl@0
|
35 |
|| aThis.iTl.iY>=aThat.iBr.iY
|
sl@0
|
36 |
|| aThat.iTl.iX>=aThis.iBr.iX
|
sl@0
|
37 |
|| aThat.iTl.iY>=aThis.iBr.iY
|
sl@0
|
38 |
)
|
sl@0
|
39 |
return TRegionExtend::TOverlapFlags(TRegionExtend::EDiffers|TRegionExtend::ENoIntersect);
|
sl@0
|
40 |
|
sl@0
|
41 |
TInt subAdd=( SubAdd::Conv(aThis.iTl.iX-aThat.iTl.iX)
|
sl@0
|
42 |
| SubAdd::Conv(aThis.iTl.iY-aThat.iTl.iY)
|
sl@0
|
43 |
| SubAdd::Conv(aThat.iBr.iX-aThis.iBr.iX)
|
sl@0
|
44 |
| SubAdd::Conv(aThat.iBr.iY-aThis.iBr.iY)
|
sl@0
|
45 |
);
|
sl@0
|
46 |
return
|
sl@0
|
47 |
TRegionExtend::TOverlapFlags(subAdd);
|
sl@0
|
48 |
}
|
sl@0
|
49 |
|
sl@0
|
50 |
/** Calc total area of a list of non-intersecting rectangles (ie a region)
|
sl@0
|
51 |
* @param aThisRects the array of rectangles
|
sl@0
|
52 |
* @param aThisCount the number in the array
|
sl@0
|
53 |
*/
|
sl@0
|
54 |
inline TUint RectListArea(const TRect* aThisRects,TInt aThisCount)
|
sl@0
|
55 |
{
|
sl@0
|
56 |
TInt thisArea=0;
|
sl@0
|
57 |
for (TInt stepThis=aThisCount-1;stepThis>=0;stepThis--)
|
sl@0
|
58 |
{
|
sl@0
|
59 |
TSize size=aThisRects[stepThis].Size();
|
sl@0
|
60 |
__ASSERT_DEBUG((size.iWidth|size.iHeight)<32768,User::Panic(_L("TRegionExtend"),4003));
|
sl@0
|
61 |
thisArea+=size.iWidth*size.iHeight;
|
sl@0
|
62 |
}
|
sl@0
|
63 |
return thisArea;
|
sl@0
|
64 |
}
|
sl@0
|
65 |
/** Returns the result code flag based on the calculated intersection of two areas
|
sl@0
|
66 |
* The intersection is always less than or equal to this and that.
|
sl@0
|
67 |
* @param aThisArea Start area
|
sl@0
|
68 |
* @param aIntersectArea Intersection
|
sl@0
|
69 |
* @param aThatArea Final area
|
sl@0
|
70 |
* @return
|
sl@0
|
71 |
* - EDiffers if both this and that differ from intersect
|
sl@0
|
72 |
* - EExact if both this and that are same as intersect
|
sl@0
|
73 |
* - ESub if only that is same as intersect, so this is bigger
|
sl@0
|
74 |
* - EAdd if only this is same as intersect, so that is bigger
|
sl@0
|
75 |
*
|
sl@0
|
76 |
**/
|
sl@0
|
77 |
inline TRegionExtend::TOverlapFlags AreaDiffResults(TUint aThisArea,TUint aIntersectArea,TUint aThatArea)
|
sl@0
|
78 |
{
|
sl@0
|
79 |
if (aThisArea>aIntersectArea)
|
sl@0
|
80 |
if (aThatArea>aIntersectArea)
|
sl@0
|
81 |
return TRegionExtend::EDiffers;
|
sl@0
|
82 |
else
|
sl@0
|
83 |
return TRegionExtend::ESub;
|
sl@0
|
84 |
else
|
sl@0
|
85 |
if (aThatArea>aIntersectArea)
|
sl@0
|
86 |
return TRegionExtend::EAdd;
|
sl@0
|
87 |
else
|
sl@0
|
88 |
return TRegionExtend::EExact;
|
sl@0
|
89 |
}
|
sl@0
|
90 |
/** Calculates the intersection area of one rectangle with an array of non intersecting rectangles
|
sl@0
|
91 |
* It is presumed that the caller will loop through all the cells of a target region,
|
sl@0
|
92 |
* repeating this call for a source region, so we can add an extra optimisation
|
sl@0
|
93 |
* to avoid revisiting any source rectangles that have been consumed completely in later passes.
|
sl@0
|
94 |
* The simplest test is that the source cell is wholy inside the target rect.
|
sl@0
|
95 |
* The simplest record is a bit field, but that only works up to 32 elements, then will not optimise further elements.
|
sl@0
|
96 |
* @param aThisCount num elements in rect array
|
sl@0
|
97 |
* @param aThisRects array of rectangles
|
sl@0
|
98 |
* @param aThatRect intersecting rectangle
|
sl@0
|
99 |
* @param aOptimiseThisBits record of elements of rect aray that have been fully consumed
|
sl@0
|
100 |
* @return total intersection area
|
sl@0
|
101 |
**/
|
sl@0
|
102 |
inline TUint TestDifferenceRegionInnerLoop(TInt aThisCount,const TRect* aThisRects,TRect& aThatRect,TUint& aOptimiseThisBits)
|
sl@0
|
103 |
{
|
sl@0
|
104 |
TUint intersectArea=0;
|
sl@0
|
105 |
for (TInt stepThis=aThisCount-1,bit=1;stepThis>=0;stepThis--,bit<<=1)
|
sl@0
|
106 |
{
|
sl@0
|
107 |
if (!(aOptimiseThisBits&bit))
|
sl@0
|
108 |
{
|
sl@0
|
109 |
const TRect& thisRect=aThisRects[stepThis];
|
sl@0
|
110 |
TRegionExtend::TOverlapFlags flags=TestDifferenceNotEmpty(thisRect,aThatRect);
|
sl@0
|
111 |
if (!(flags&TRegionExtend::ENoIntersect))
|
sl@0
|
112 |
{
|
sl@0
|
113 |
if (!(flags&TRegionExtend::EAdd))
|
sl@0
|
114 |
{ //Skip rest of inner loop if a containing rect is found
|
sl@0
|
115 |
TSize size=aThatRect.Size();
|
sl@0
|
116 |
intersectArea+=size.iWidth*size.iHeight;
|
sl@0
|
117 |
if (!(flags&TRegionExtend::ESub)) //equal rects...
|
sl@0
|
118 |
{ //skip this cell for rest of outer loop if a contains rect is found
|
sl@0
|
119 |
aOptimiseThisBits|=bit;
|
sl@0
|
120 |
}
|
sl@0
|
121 |
break; //this cell contains the target rect so don't bother checking any more
|
sl@0
|
122 |
}
|
sl@0
|
123 |
else
|
sl@0
|
124 |
if (!(flags&TRegionExtend::ESub))
|
sl@0
|
125 |
{ //skip this cell for rest of outer loop if a contains rect is found
|
sl@0
|
126 |
aOptimiseThisBits|=bit;
|
sl@0
|
127 |
TSize size=thisRect.Size();
|
sl@0
|
128 |
intersectArea+=size.iWidth*size.iHeight;
|
sl@0
|
129 |
}
|
sl@0
|
130 |
else
|
sl@0
|
131 |
{
|
sl@0
|
132 |
TRect intersect=thisRect;
|
sl@0
|
133 |
intersect.Intersection(aThatRect);
|
sl@0
|
134 |
TSize size=intersect.Size();
|
sl@0
|
135 |
intersectArea+=size.iWidth*size.iHeight;
|
sl@0
|
136 |
}
|
sl@0
|
137 |
}
|
sl@0
|
138 |
}
|
sl@0
|
139 |
}
|
sl@0
|
140 |
return intersectArea;
|
sl@0
|
141 |
}
|
sl@0
|
142 |
/** Avoids the use of a temp region by performing area calc on the fly.
|
sl@0
|
143 |
* If both regions are empty then EOverlapNoIntersect only is returned.
|
sl@0
|
144 |
* @param aThat target region
|
sl@0
|
145 |
* @return flags from TOverlapFlags enumeration
|
sl@0
|
146 |
* - EExact =0 if rgions are exactly identical
|
sl@0
|
147 |
* - ESub Flagged if some rectangles are removed converting current region to that target
|
sl@0
|
148 |
* - EAdd Flagged if some rectangles are added converting current region to that target
|
sl@0
|
149 |
* - ENoIntersect if there is no common region between the current region and that target
|
sl@0
|
150 |
* - EErrorRegion One of the regions is signalling CheckError()
|
sl@0
|
151 |
**/
|
sl@0
|
152 |
TRegionExtend::TOverlapFlags TRegionExtend::TestDifference(const TRegion& aThat)const
|
sl@0
|
153 |
{
|
sl@0
|
154 |
TInt intersectArea=0;
|
sl@0
|
155 |
TInt thisArea=0;
|
sl@0
|
156 |
TInt thatArea=0;
|
sl@0
|
157 |
const TRect* thisRects=RectangleList();
|
sl@0
|
158 |
const TRect* thatRects=aThat.RectangleList();
|
sl@0
|
159 |
TInt thatCount=aThat.Count();
|
sl@0
|
160 |
TInt thisCount=Count();
|
sl@0
|
161 |
|
sl@0
|
162 |
if (CheckError()||aThat.CheckError())
|
sl@0
|
163 |
return EErrorRegion;
|
sl@0
|
164 |
if (thisCount==0)
|
sl@0
|
165 |
if (thatCount==0)
|
sl@0
|
166 |
return ENoIntersect;
|
sl@0
|
167 |
else
|
sl@0
|
168 |
return TOverlapFlags(ENoIntersect|EAdd);
|
sl@0
|
169 |
//both regions are populated. How big is the intersection?
|
sl@0
|
170 |
//The following optimisation bit is that
|
sl@0
|
171 |
//if any rect is fully contained by a rect in the opposite region
|
sl@0
|
172 |
//then further compares against that rect are skipped. For this, inner loop is skipped immediately
|
sl@0
|
173 |
//Can track the first 32 rects of aThat. The remainder won't benefit from the optimisation
|
sl@0
|
174 |
TUint optimiseThisBits=0;
|
sl@0
|
175 |
for (TInt stepThat=thatCount-1;stepThat>=0;stepThat--)
|
sl@0
|
176 |
{
|
sl@0
|
177 |
TRect thatRect=thatRects[stepThat];
|
sl@0
|
178 |
intersectArea+=TestDifferenceRegionInnerLoop(thisCount,thisRects,thatRect,optimiseThisBits);
|
sl@0
|
179 |
}
|
sl@0
|
180 |
if (intersectArea==0)
|
sl@0
|
181 |
if (thatCount==0)
|
sl@0
|
182 |
return TOverlapFlags(ENoIntersect|ESub);
|
sl@0
|
183 |
else
|
sl@0
|
184 |
return TOverlapFlags(ENoIntersect|EAdd|ESub);
|
sl@0
|
185 |
thatArea=RectListArea(thatRects,thatCount);
|
sl@0
|
186 |
thisArea=RectListArea(thisRects,thisCount);
|
sl@0
|
187 |
return AreaDiffResults( thisArea, intersectArea, thatArea );
|
sl@0
|
188 |
}
|
sl@0
|
189 |
|
sl@0
|
190 |
/** Avoids the use of a temp region by performing area calc on the fly.
|
sl@0
|
191 |
* This version further optimises the process by avoiding the client having to re-origin either region.
|
sl@0
|
192 |
* If both regions are empty then EOverlapNoIntersect only is returned.
|
sl@0
|
193 |
* @param aThat target region
|
sl@0
|
194 |
* @return flags from TOverlapFlags enumeration
|
sl@0
|
195 |
* - EExact =0 if rgions are exactly identical
|
sl@0
|
196 |
* - ESub Flagged if some rectangles are removed converting current region to that target
|
sl@0
|
197 |
* - EAdd Flagged if some rectangles are added converting current region to that target
|
sl@0
|
198 |
* - ENoIntersect if there is no common region between the current region and that target
|
sl@0
|
199 |
* - EErrorRegion One of the regions is signalling CheckError()
|
sl@0
|
200 |
**/
|
sl@0
|
201 |
TRegionExtend::TOverlapFlags TRegionExtend::TestDifference(const TRegion& aThat,TPoint aOffsetToThat)const
|
sl@0
|
202 |
{
|
sl@0
|
203 |
TInt intersectArea=0;
|
sl@0
|
204 |
TInt thisArea=0;
|
sl@0
|
205 |
TInt thatArea=0;
|
sl@0
|
206 |
const TRect* thisRects=RectangleList();
|
sl@0
|
207 |
const TRect* thatRects=aThat.RectangleList();
|
sl@0
|
208 |
TInt thatCount=aThat.Count();
|
sl@0
|
209 |
TInt thisCount=Count();
|
sl@0
|
210 |
|
sl@0
|
211 |
if (CheckError()||aThat.CheckError())
|
sl@0
|
212 |
return EErrorRegion;
|
sl@0
|
213 |
if (thisCount==0)
|
sl@0
|
214 |
if (thatCount==0)
|
sl@0
|
215 |
return ENoIntersect;
|
sl@0
|
216 |
else
|
sl@0
|
217 |
return TOverlapFlags(ENoIntersect|EAdd);
|
sl@0
|
218 |
//both regions are populated. How big is the intersection?
|
sl@0
|
219 |
//The following optimisation bit is that
|
sl@0
|
220 |
//if any rect is fully contained by a rect in the opposite region
|
sl@0
|
221 |
//then further compares against that rect are skipped. For this, inner loop is skipped immediately
|
sl@0
|
222 |
//Can track the first 32 rects of aThat. The remainder won't benefit from the optimisation
|
sl@0
|
223 |
TUint optimiseThisBits=0;
|
sl@0
|
224 |
for (TInt stepThat=thatCount-1;stepThat>=0;stepThat--)
|
sl@0
|
225 |
{
|
sl@0
|
226 |
TRect thatRect=thatRects[stepThat];
|
sl@0
|
227 |
thatRect.Move(-aOffsetToThat.iX,-aOffsetToThat.iY); //this line is the only difference, but the next lump has a lot of parameters...
|
sl@0
|
228 |
intersectArea+=TestDifferenceRegionInnerLoop(thisCount,thisRects,thatRect,optimiseThisBits);
|
sl@0
|
229 |
}
|
sl@0
|
230 |
if (intersectArea==0)
|
sl@0
|
231 |
if (thatCount==0)
|
sl@0
|
232 |
return TOverlapFlags(ENoIntersect|ESub);
|
sl@0
|
233 |
else
|
sl@0
|
234 |
return TOverlapFlags(ENoIntersect|EAdd|ESub);
|
sl@0
|
235 |
|
sl@0
|
236 |
thatArea=RectListArea(thatRects,thatCount);
|
sl@0
|
237 |
thisArea=RectListArea(thisRects,thisCount);
|
sl@0
|
238 |
return AreaDiffResults( thisArea, intersectArea, thatArea );
|
sl@0
|
239 |
}
|
sl@0
|
240 |
|
sl@0
|
241 |
/** This returns the same comparrison flags between two rectangles.
|
sl@0
|
242 |
* Note that a zero return means exact intersection...
|
sl@0
|
243 |
* Intended as an internal method, but there doesn't seem to be a useful public alternative.
|
sl@0
|
244 |
* @param aThis source rectangle
|
sl@0
|
245 |
* @param aThat target rectangle
|
sl@0
|
246 |
* @return flags from TOverlapFlags enumeration
|
sl@0
|
247 |
* - EExact =0 if rgions are exactly identical
|
sl@0
|
248 |
* - ESub Flagged if some rectangles are removed converting this source rectangle to that target
|
sl@0
|
249 |
* - EAdd Flagged if some rectangles are added converting this source rectangle to that target
|
sl@0
|
250 |
* - ENoIntersect if there is no common region between this source rectangle and that target
|
sl@0
|
251 |
**/
|
sl@0
|
252 |
TRegionExtend::TOverlapFlags TRegionExtend::TestDifference(const TRect& aThis,const TRect& aThat)
|
sl@0
|
253 |
{
|
sl@0
|
254 |
if (aThis.IsEmpty())
|
sl@0
|
255 |
if (aThat.IsEmpty())
|
sl@0
|
256 |
return ENoIntersect;
|
sl@0
|
257 |
else
|
sl@0
|
258 |
return TOverlapFlags(EAdd|ENoIntersect);
|
sl@0
|
259 |
else
|
sl@0
|
260 |
if (aThat.IsEmpty())
|
sl@0
|
261 |
return TOverlapFlags(ESub|ENoIntersect);
|
sl@0
|
262 |
return TestDifferenceNotEmpty(aThis,aThat);
|
sl@0
|
263 |
}
|
sl@0
|
264 |
|
sl@0
|
265 |
|
sl@0
|
266 |
/** Returns total area of the region
|
sl@0
|
267 |
* @return total area
|
sl@0
|
268 |
**/
|
sl@0
|
269 |
TUint TRegionExtend::Area()const
|
sl@0
|
270 |
{
|
sl@0
|
271 |
return RectListArea(RectangleList(),Count());
|
sl@0
|
272 |
}
|
sl@0
|
273 |
|
sl@0
|
274 |
/** Avoids the use of a temp region by performing area calc on the fly.
|
sl@0
|
275 |
* If both are empty then EOverlapNoIntersect only is returned.
|
sl@0
|
276 |
* @param aThat target rectangle
|
sl@0
|
277 |
* @return flags from TOverlapFlags enumeration
|
sl@0
|
278 |
* - EExact =0 if region exactly fills rectangle
|
sl@0
|
279 |
* - ESub Flagged if some rectangles are removed converting current region to that target
|
sl@0
|
280 |
* - EAdd Flagged if some rectangles are added converting current region to that target
|
sl@0
|
281 |
* - ENoIntersect if there is no common region between the current region and that target
|
sl@0
|
282 |
* - EErrorRegion One of the region is signalling CheckError()
|
sl@0
|
283 |
**/
|
sl@0
|
284 |
TRegionExtend::TOverlapFlags TRegionExtend::TestDifference(const TRect& aThat)const
|
sl@0
|
285 |
{
|
sl@0
|
286 |
TInt intersectArea=0;
|
sl@0
|
287 |
const TRect* thisRects=RectangleList();
|
sl@0
|
288 |
TInt thisCount=Count();
|
sl@0
|
289 |
|
sl@0
|
290 |
if (aThat.IsEmpty())
|
sl@0
|
291 |
if (thisCount==0)
|
sl@0
|
292 |
return ENoIntersect;
|
sl@0
|
293 |
else
|
sl@0
|
294 |
return TOverlapFlags(ENoIntersect|ESub);
|
sl@0
|
295 |
if (CheckError())
|
sl@0
|
296 |
return EErrorRegion;
|
sl@0
|
297 |
TInt output=ENoIntersect;
|
sl@0
|
298 |
for (TInt stepThis=thisCount-1,bit=1;stepThis>=0;stepThis--,bit+=bit)
|
sl@0
|
299 |
{
|
sl@0
|
300 |
TOverlapFlags flags=TestDifferenceNotEmpty(thisRects[stepThis],aThat);
|
sl@0
|
301 |
if (!(flags&ENoIntersect))
|
sl@0
|
302 |
{
|
sl@0
|
303 |
if (!(flags&EAdd)) //the target rect does not add anything to this region element
|
sl@0
|
304 |
{ //Skip rest of inner loop if a containing rect is found
|
sl@0
|
305 |
if ((flags&ESub)||thisCount>1)
|
sl@0
|
306 |
return ESub; //the region element is bigger or there are more region elements
|
sl@0
|
307 |
else
|
sl@0
|
308 |
return EExact;
|
sl@0
|
309 |
}
|
sl@0
|
310 |
else
|
sl@0
|
311 |
{
|
sl@0
|
312 |
TRect intersect=thisRects[stepThis];
|
sl@0
|
313 |
intersect.Intersection(aThat);
|
sl@0
|
314 |
TSize size=intersect.Size();
|
sl@0
|
315 |
__ASSERT_DEBUG((size.iWidth|size.iHeight)<32768,User::Panic(_L("TRegionExtend"),1003));
|
sl@0
|
316 |
intersectArea+=size.iWidth*size.iHeight;
|
sl@0
|
317 |
|
sl@0
|
318 |
}
|
sl@0
|
319 |
output&=~ENoIntersect;
|
sl@0
|
320 |
}
|
sl@0
|
321 |
output|=(flags&ESub);
|
sl@0
|
322 |
}
|
sl@0
|
323 |
if (intersectArea==0)
|
sl@0
|
324 |
{
|
sl@0
|
325 |
return TOverlapFlags(output|EAdd);
|
sl@0
|
326 |
}
|
sl@0
|
327 |
TSize size=aThat.Size();
|
sl@0
|
328 |
__ASSERT_DEBUG((size.iWidth|size.iHeight)<32768,User::Panic(_L("TRegionExtend"),2003));
|
sl@0
|
329 |
TInt thatArea=size.iWidth*size.iHeight;
|
sl@0
|
330 |
if (thatArea>intersectArea)
|
sl@0
|
331 |
return TOverlapFlags(output|EAdd);
|
sl@0
|
332 |
else
|
sl@0
|
333 |
return TOverlapFlags(output);
|
sl@0
|
334 |
}
|
sl@0
|
335 |
|
sl@0
|
336 |
|