1 /*
2  * Copyright 2006 The Android Open Source Project
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #ifndef SkRect_DEFINED
9 #define SkRect_DEFINED
10 
11 #include "SkPoint.h"
12 #include "SkSize.h"
13 
14 struct SkRect;
15 
16 /** \struct SkIRect
17 
18     SkIRect holds four 32 bit integer coordinates for a rectangle
19 */
20 struct SK_API SkIRect {
21     int32_t fLeft, fTop, fRight, fBottom;
22 
MakeEmptySkIRect23     static SkIRect SK_WARN_UNUSED_RESULT MakeEmpty() {
24         SkIRect r;
25         r.setEmpty();
26         return r;
27     }
28 
MakeLargestSkIRect29     static SkIRect SK_WARN_UNUSED_RESULT MakeLargest() {
30         SkIRect r;
31         r.setLargest();
32         return r;
33     }
34 
MakeWHSkIRect35     static SkIRect SK_WARN_UNUSED_RESULT MakeWH(int32_t w, int32_t h) {
36         SkIRect r;
37         r.set(0, 0, w, h);
38         return r;
39     }
40 
MakeSizeSkIRect41     static SkIRect SK_WARN_UNUSED_RESULT MakeSize(const SkISize& size) {
42         SkIRect r;
43         r.set(0, 0, size.width(), size.height());
44         return r;
45     }
46 
MakeLTRBSkIRect47     static SkIRect SK_WARN_UNUSED_RESULT MakeLTRB(int32_t l, int32_t t, int32_t r, int32_t b) {
48         SkIRect rect;
49         rect.set(l, t, r, b);
50         return rect;
51     }
52 
MakeXYWHSkIRect53     static SkIRect SK_WARN_UNUSED_RESULT MakeXYWH(int32_t x, int32_t y, int32_t w, int32_t h) {
54         SkIRect r;
55         r.set(x, y, x + w, y + h);
56         return r;
57     }
58 
leftSkIRect59     int left() const { return fLeft; }
topSkIRect60     int top() const { return fTop; }
rightSkIRect61     int right() const { return fRight; }
bottomSkIRect62     int bottom() const { return fBottom; }
63 
64     /** return the left edge of the rect */
xSkIRect65     int x() const { return fLeft; }
66     /** return the top edge of the rect */
ySkIRect67     int y() const { return fTop; }
68     /**
69      *  Returns the rectangle's width. This does not check for a valid rect
70      *  (i.e. left <= right) so the result may be negative.
71      */
widthSkIRect72     int width() const { return fRight - fLeft; }
73 
74     /**
75      *  Returns the rectangle's height. This does not check for a valid rect
76      *  (i.e. top <= bottom) so the result may be negative.
77      */
heightSkIRect78     int height() const { return fBottom - fTop; }
79 
sizeSkIRect80     SkISize size() const { return SkISize::Make(this->width(), this->height()); }
81 
82     /**
83      *  Since the center of an integer rect may fall on a factional value, this
84      *  method is defined to return (right + left) >> 1.
85      *
86      *  This is a specific "truncation" of the average, which is different than
87      *  (right + left) / 2 when the sum is negative.
88      */
centerXSkIRect89     int centerX() const { return (fRight + fLeft) >> 1; }
90 
91     /**
92      *  Since the center of an integer rect may fall on a factional value, this
93      *  method is defined to return (bottom + top) >> 1
94      *
95      *  This is a specific "truncation" of the average, which is different than
96      *  (bottom + top) / 2 when the sum is negative.
97      */
centerYSkIRect98     int centerY() const { return (fBottom + fTop) >> 1; }
99 
100     /**
101      *  Return true if the rectangle's width or height are <= 0
102      */
isEmptySkIRect103     bool isEmpty() const { return fLeft >= fRight || fTop >= fBottom; }
104 
isLargestSkIRect105     bool isLargest() const { return SK_MinS32 == fLeft &&
106                                     SK_MinS32 == fTop &&
107                                     SK_MaxS32 == fRight &&
108                                     SK_MaxS32 == fBottom; }
109 
110     friend bool operator==(const SkIRect& a, const SkIRect& b) {
111         return !memcmp(&a, &b, sizeof(a));
112     }
113 
114     friend bool operator!=(const SkIRect& a, const SkIRect& b) {
115         return !(a == b);
116     }
117 
is16BitSkIRect118     bool is16Bit() const {
119         return  SkIsS16(fLeft) && SkIsS16(fTop) &&
120                 SkIsS16(fRight) && SkIsS16(fBottom);
121     }
122 
123     /** Set the rectangle to (0,0,0,0)
124     */
setEmptySkIRect125     void setEmpty() { memset(this, 0, sizeof(*this)); }
126 
setSkIRect127     void set(int32_t left, int32_t top, int32_t right, int32_t bottom) {
128         fLeft   = left;
129         fTop    = top;
130         fRight  = right;
131         fBottom = bottom;
132     }
133     // alias for set(l, t, r, b)
setLTRBSkIRect134     void setLTRB(int32_t left, int32_t top, int32_t right, int32_t bottom) {
135         this->set(left, top, right, bottom);
136     }
137 
setXYWHSkIRect138     void setXYWH(int32_t x, int32_t y, int32_t width, int32_t height) {
139         fLeft = x;
140         fTop = y;
141         fRight = x + width;
142         fBottom = y + height;
143     }
144 
145     /**
146      *  Make the largest representable rectangle
147      */
setLargestSkIRect148     void setLargest() {
149         fLeft = fTop = SK_MinS32;
150         fRight = fBottom = SK_MaxS32;
151     }
152 
153     /**
154      *  Make the largest representable rectangle, but inverted (e.g. fLeft will
155      *  be max 32bit and right will be min 32bit).
156      */
setLargestInvertedSkIRect157     void setLargestInverted() {
158         fLeft = fTop = SK_MaxS32;
159         fRight = fBottom = SK_MinS32;
160     }
161 
162     /**
163      *  Return a new IRect, built as an offset of this rect.
164      */
makeOffsetSkIRect165     SkIRect makeOffset(int32_t dx, int32_t dy) const {
166         return MakeLTRB(fLeft + dx, fTop + dy, fRight + dx, fBottom + dy);
167     }
168 
169     /**
170      *  Return a new IRect, built as an inset of this rect.
171      */
makeInsetSkIRect172     SkIRect makeInset(int32_t dx, int32_t dy) const {
173         return MakeLTRB(fLeft + dx, fTop + dy, fRight - dx, fBottom - dy);
174     }
175 
176     /**
177      *  Return a new Rect, built as an outset of this rect.
178      */
makeOutsetSkIRect179     SkIRect makeOutset(int32_t dx, int32_t dy) const {
180         return MakeLTRB(fLeft - dx, fTop - dy, fRight + dx, fBottom + dy);
181     }
182 
183     /** Offset set the rectangle by adding dx to its left and right,
184         and adding dy to its top and bottom.
185     */
offsetSkIRect186     void offset(int32_t dx, int32_t dy) {
187         fLeft   += dx;
188         fTop    += dy;
189         fRight  += dx;
190         fBottom += dy;
191     }
192 
offsetSkIRect193     void offset(const SkIPoint& delta) {
194         this->offset(delta.fX, delta.fY);
195     }
196 
197     /**
198      *  Offset this rect such its new x() and y() will equal newX and newY.
199      */
offsetToSkIRect200     void offsetTo(int32_t newX, int32_t newY) {
201         fRight += newX - fLeft;
202         fBottom += newY - fTop;
203         fLeft = newX;
204         fTop = newY;
205     }
206 
207     /** Inset the rectangle by (dx,dy). If dx is positive, then the sides are moved inwards,
208         making the rectangle narrower. If dx is negative, then the sides are moved outwards,
209         making the rectangle wider. The same holds true for dy and the top and bottom.
210     */
insetSkIRect211     void inset(int32_t dx, int32_t dy) {
212         fLeft   += dx;
213         fTop    += dy;
214         fRight  -= dx;
215         fBottom -= dy;
216     }
217 
218    /** Outset the rectangle by (dx,dy). If dx is positive, then the sides are
219        moved outwards, making the rectangle wider. If dx is negative, then the
220        sides are moved inwards, making the rectangle narrower. The same holds
221        true for dy and the top and bottom.
222     */
outsetSkIRect223     void outset(int32_t dx, int32_t dy)  { this->inset(-dx, -dy); }
224 
quickRejectSkIRect225     bool quickReject(int l, int t, int r, int b) const {
226         return l >= fRight || fLeft >= r || t >= fBottom || fTop >= b;
227     }
228 
229     /** Returns true if (x,y) is inside the rectangle and the rectangle is not
230         empty. The left and top are considered to be inside, while the right
231         and bottom are not. Thus for the rectangle (0, 0, 5, 10), the
232         points (0,0) and (0,9) are inside, while (-1,0) and (5,9) are not.
233     */
containsSkIRect234     bool contains(int32_t x, int32_t y) const {
235         return  (unsigned)(x - fLeft) < (unsigned)(fRight - fLeft) &&
236                 (unsigned)(y - fTop) < (unsigned)(fBottom - fTop);
237     }
238 
239     /** Returns true if the 4 specified sides of a rectangle are inside or equal to this rectangle.
240         If either rectangle is empty, contains() returns false.
241     */
containsSkIRect242     bool contains(int32_t left, int32_t top, int32_t right, int32_t bottom) const {
243         return  left < right && top < bottom && !this->isEmpty() && // check for empties
244                 fLeft <= left && fTop <= top &&
245                 fRight >= right && fBottom >= bottom;
246     }
247 
248     /** Returns true if the specified rectangle r is inside or equal to this rectangle.
249     */
containsSkIRect250     bool contains(const SkIRect& r) const {
251         return  !r.isEmpty() && !this->isEmpty() &&     // check for empties
252                 fLeft <= r.fLeft && fTop <= r.fTop &&
253                 fRight >= r.fRight && fBottom >= r.fBottom;
254     }
255 
256     /** Returns true if the specified rectangle r is inside or equal to this rectangle.
257     */
258     bool contains(const SkRect& r) const;
259 
260     /** Return true if this rectangle contains the specified rectangle.
261         For speed, this method does not check if either this or the specified
262         rectangles are empty, and if either is, its return value is undefined.
263         In the debugging build however, we assert that both this and the
264         specified rectangles are non-empty.
265     */
containsNoEmptyCheckSkIRect266     bool containsNoEmptyCheck(int32_t left, int32_t top,
267                               int32_t right, int32_t bottom) const {
268         SkASSERT(fLeft < fRight && fTop < fBottom);
269         SkASSERT(left < right && top < bottom);
270 
271         return fLeft <= left && fTop <= top &&
272                fRight >= right && fBottom >= bottom;
273     }
274 
containsNoEmptyCheckSkIRect275     bool containsNoEmptyCheck(const SkIRect& r) const {
276         return containsNoEmptyCheck(r.fLeft, r.fTop, r.fRight, r.fBottom);
277     }
278 
279     /** If r intersects this rectangle, return true and set this rectangle to that
280         intersection, otherwise return false and do not change this rectangle.
281         If either rectangle is empty, do nothing and return false.
282     */
intersectSkIRect283     bool intersect(const SkIRect& r) {
284         return this->intersect(r.fLeft, r.fTop, r.fRight, r.fBottom);
285     }
286 
287     /** If rectangles a and b intersect, return true and set this rectangle to
288         that intersection, otherwise return false and do not change this
289         rectangle. If either rectangle is empty, do nothing and return false.
290     */
intersectSkIRect291     bool SK_WARN_UNUSED_RESULT intersect(const SkIRect& a, const SkIRect& b) {
292 
293         if (!a.isEmpty() && !b.isEmpty() &&
294                 a.fLeft < b.fRight && b.fLeft < a.fRight &&
295                 a.fTop < b.fBottom && b.fTop < a.fBottom) {
296             fLeft   = SkMax32(a.fLeft,   b.fLeft);
297             fTop    = SkMax32(a.fTop,    b.fTop);
298             fRight  = SkMin32(a.fRight,  b.fRight);
299             fBottom = SkMin32(a.fBottom, b.fBottom);
300             return true;
301         }
302         return false;
303     }
304 
305     /** If rectangles a and b intersect, return true and set this rectangle to
306         that intersection, otherwise return false and do not change this
307         rectangle. For speed, no check to see if a or b are empty is performed.
308         If either is, then the return result is undefined. In the debug build,
309         we assert that both rectangles are non-empty.
310     */
intersectNoEmptyCheckSkIRect311     bool SK_WARN_UNUSED_RESULT intersectNoEmptyCheck(const SkIRect& a, const SkIRect& b) {
312         SkASSERT(!a.isEmpty() && !b.isEmpty());
313 
314         if (a.fLeft < b.fRight && b.fLeft < a.fRight &&
315                 a.fTop < b.fBottom && b.fTop < a.fBottom) {
316             fLeft   = SkMax32(a.fLeft,   b.fLeft);
317             fTop    = SkMax32(a.fTop,    b.fTop);
318             fRight  = SkMin32(a.fRight,  b.fRight);
319             fBottom = SkMin32(a.fBottom, b.fBottom);
320             return true;
321         }
322         return false;
323     }
324 
325     /** If the rectangle specified by left,top,right,bottom intersects this rectangle,
326         return true and set this rectangle to that intersection,
327         otherwise return false and do not change this rectangle.
328         If either rectangle is empty, do nothing and return false.
329     */
intersectSkIRect330     bool intersect(int32_t left, int32_t top,
331                                          int32_t right, int32_t bottom) {
332         if (left < right && top < bottom && !this->isEmpty() &&
333                 fLeft < right && left < fRight && fTop < bottom && top < fBottom) {
334             if (fLeft < left) fLeft = left;
335             if (fTop < top) fTop = top;
336             if (fRight > right) fRight = right;
337             if (fBottom > bottom) fBottom = bottom;
338             return true;
339         }
340         return false;
341     }
342 
343     /** Returns true if a and b are not empty, and they intersect
344      */
IntersectsSkIRect345     static bool Intersects(const SkIRect& a, const SkIRect& b) {
346         return  !a.isEmpty() && !b.isEmpty() &&              // check for empties
347                 a.fLeft < b.fRight && b.fLeft < a.fRight &&
348                 a.fTop < b.fBottom && b.fTop < a.fBottom;
349     }
350 
351     /**
352      *  Returns true if a and b intersect. debug-asserts that neither are empty.
353      */
IntersectsNoEmptyCheckSkIRect354     static bool IntersectsNoEmptyCheck(const SkIRect& a, const SkIRect& b) {
355         SkASSERT(!a.isEmpty());
356         SkASSERT(!b.isEmpty());
357         return  a.fLeft < b.fRight && b.fLeft < a.fRight &&
358                 a.fTop < b.fBottom && b.fTop < a.fBottom;
359     }
360 
361     /** Update this rectangle to enclose itself and the specified rectangle.
362         If this rectangle is empty, just set it to the specified rectangle. If the specified
363         rectangle is empty, do nothing.
364     */
365     void join(int32_t left, int32_t top, int32_t right, int32_t bottom);
366 
367     /** Update this rectangle to enclose itself and the specified rectangle.
368         If this rectangle is empty, just set it to the specified rectangle. If the specified
369         rectangle is empty, do nothing.
370     */
joinSkIRect371     void join(const SkIRect& r) {
372         this->join(r.fLeft, r.fTop, r.fRight, r.fBottom);
373     }
374 
375     /** Swap top/bottom or left/right if there are flipped.
376         This can be called if the edges are computed separately,
377         and may have crossed over each other.
378         When this returns, left <= right && top <= bottom
379     */
380     void sort();
381 
EmptyIRectSkIRect382     static const SkIRect& SK_WARN_UNUSED_RESULT EmptyIRect() {
383         static const SkIRect gEmpty = { 0, 0, 0, 0 };
384         return gEmpty;
385     }
386 };
387 
388 /** \struct SkRect
389 */
390 struct SK_API SkRect {
391     SkScalar    fLeft, fTop, fRight, fBottom;
392 
MakeEmptySkRect393     static constexpr SkRect SK_WARN_UNUSED_RESULT MakeEmpty() {
394         return SkRect{0, 0, 0, 0};
395     }
396 
MakeLargestSkRect397     static SkRect SK_WARN_UNUSED_RESULT MakeLargest() {
398         SkRect r;
399         r.setLargest();
400         return r;
401     }
402 
MakeWHSkRect403     static SkRect SK_WARN_UNUSED_RESULT MakeWH(SkScalar w, SkScalar h) {
404         SkRect r;
405         r.set(0, 0, w, h);
406         return r;
407     }
408 
MakeIWHSkRect409     static SkRect SK_WARN_UNUSED_RESULT MakeIWH(int w, int h) {
410         SkRect r;
411         r.set(0, 0, SkIntToScalar(w), SkIntToScalar(h));
412         return r;
413     }
414 
MakeSizeSkRect415     static SkRect SK_WARN_UNUSED_RESULT MakeSize(const SkSize& size) {
416         SkRect r;
417         r.set(0, 0, size.width(), size.height());
418         return r;
419     }
420 
MakeLTRBSkRect421     static constexpr SkRect SK_WARN_UNUSED_RESULT MakeLTRB(SkScalar l, SkScalar t, SkScalar r,
422                                                            SkScalar b) {
423         return SkRect {l, t, r, b};
424     }
425 
MakeXYWHSkRect426     static SkRect SK_WARN_UNUSED_RESULT MakeXYWH(SkScalar x, SkScalar y, SkScalar w, SkScalar h) {
427         SkRect r;
428         r.set(x, y, x + w, y + h);
429         return r;
430     }
431 
432     SK_ATTR_DEPRECATED("use Make()")
MakeFromIRectSkRect433     static SkRect SK_WARN_UNUSED_RESULT MakeFromIRect(const SkIRect& irect) {
434         SkRect r;
435         r.set(SkIntToScalar(irect.fLeft),
436               SkIntToScalar(irect.fTop),
437               SkIntToScalar(irect.fRight),
438               SkIntToScalar(irect.fBottom));
439         return r;
440     }
441 
MakeSkRect442     static SkRect Make(const SkISize& size) {
443         return MakeIWH(size.width(), size.height());
444     }
445 
MakeSkRect446     static SkRect SK_WARN_UNUSED_RESULT Make(const SkIRect& irect) {
447         SkRect r;
448         r.set(SkIntToScalar(irect.fLeft),
449               SkIntToScalar(irect.fTop),
450               SkIntToScalar(irect.fRight),
451               SkIntToScalar(irect.fBottom));
452         return r;
453     }
454 
455     /**
456      *  Return true if the rectangle's width or height are <= 0
457      */
isEmptySkRect458     bool isEmpty() const { return fLeft >= fRight || fTop >= fBottom; }
459 
isLargestSkRect460     bool isLargest() const { return SK_ScalarMin == fLeft &&
461                                     SK_ScalarMin == fTop &&
462                                     SK_ScalarMax == fRight &&
463                                     SK_ScalarMax == fBottom; }
464 
465     /**
466      *  Returns true iff all values in the rect are finite. If any are
467      *  infinite or NaN then this returns false.
468      */
isFiniteSkRect469     bool isFinite() const {
470         float accum = 0;
471         accum *= fLeft;
472         accum *= fTop;
473         accum *= fRight;
474         accum *= fBottom;
475 
476         // accum is either NaN or it is finite (zero).
477         SkASSERT(0 == accum || SkScalarIsNaN(accum));
478 
479         // value==value will be true iff value is not NaN
480         // TODO: is it faster to say !accum or accum==accum?
481         return !SkScalarIsNaN(accum);
482     }
483 
xSkRect484     SkScalar    x() const { return fLeft; }
ySkRect485     SkScalar    y() const { return fTop; }
leftSkRect486     SkScalar    left() const { return fLeft; }
topSkRect487     SkScalar    top() const { return fTop; }
rightSkRect488     SkScalar    right() const { return fRight; }
bottomSkRect489     SkScalar    bottom() const { return fBottom; }
widthSkRect490     SkScalar    width() const { return fRight - fLeft; }
heightSkRect491     SkScalar    height() const { return fBottom - fTop; }
centerXSkRect492     SkScalar    centerX() const { return SkScalarHalf(fLeft + fRight); }
centerYSkRect493     SkScalar    centerY() const { return SkScalarHalf(fTop + fBottom); }
494 
495     friend bool operator==(const SkRect& a, const SkRect& b) {
496         return SkScalarsEqual((SkScalar*)&a, (SkScalar*)&b, 4);
497     }
498 
499     friend bool operator!=(const SkRect& a, const SkRect& b) {
500         return !SkScalarsEqual((SkScalar*)&a, (SkScalar*)&b, 4);
501     }
502 
503     /** return the 4 points that enclose the rectangle (top-left, top-right, bottom-right,
504         bottom-left). TODO: Consider adding param to control whether quad is CW or CCW.
505      */
506     void toQuad(SkPoint quad[4]) const;
507 
508     /** Set this rectangle to the empty rectangle (0,0,0,0)
509     */
setEmptySkRect510     void setEmpty() { *this = MakeEmpty(); }
511 
setSkRect512     void set(const SkIRect& src) {
513         fLeft   = SkIntToScalar(src.fLeft);
514         fTop    = SkIntToScalar(src.fTop);
515         fRight  = SkIntToScalar(src.fRight);
516         fBottom = SkIntToScalar(src.fBottom);
517     }
518 
setSkRect519     void set(SkScalar left, SkScalar top, SkScalar right, SkScalar bottom) {
520         fLeft   = left;
521         fTop    = top;
522         fRight  = right;
523         fBottom = bottom;
524     }
525     // alias for set(l, t, r, b)
setLTRBSkRect526     void setLTRB(SkScalar left, SkScalar top, SkScalar right, SkScalar bottom) {
527         this->set(left, top, right, bottom);
528     }
529 
530     /** Initialize the rect with the 4 specified integers. The routine handles
531         converting them to scalars (by calling SkIntToScalar)
532      */
isetSkRect533     void iset(int left, int top, int right, int bottom) {
534         fLeft   = SkIntToScalar(left);
535         fTop    = SkIntToScalar(top);
536         fRight  = SkIntToScalar(right);
537         fBottom = SkIntToScalar(bottom);
538     }
539 
540     /**
541      *  Set this rectangle to be left/top at 0,0, and have the specified width
542      *  and height (automatically converted to SkScalar).
543      */
isetWHSkRect544     void isetWH(int width, int height) {
545         fLeft = fTop = 0;
546         fRight = SkIntToScalar(width);
547         fBottom = SkIntToScalar(height);
548     }
549 
550     /** Set this rectangle to be the bounds of the array of points.
551         If the array is empty (count == 0), then set this rectangle
552         to the empty rectangle (0,0,0,0)
553     */
setSkRect554     void set(const SkPoint pts[], int count) {
555         // set() had been checking for non-finite values, so keep that behavior
556         // for now. Now that we have setBoundsCheck(), we may decide to make
557         // set() be simpler/faster, and not check for those.
558         (void)this->setBoundsCheck(pts, count);
559     }
560 
561     // alias for set(pts, count)
setBoundsSkRect562     void setBounds(const SkPoint pts[], int count) {
563         (void)this->setBoundsCheck(pts, count);
564     }
565 
566     /**
567      *  Compute the bounds of the array of points, and set this rect to that
568      *  bounds and return true... unless a non-finite value is encountered,
569      *  in which case this rect is set to empty and false is returned.
570      */
571     bool setBoundsCheck(const SkPoint pts[], int count);
572 
setSkRect573     void set(const SkPoint& p0, const SkPoint& p1) {
574         fLeft =   SkMinScalar(p0.fX, p1.fX);
575         fRight =  SkMaxScalar(p0.fX, p1.fX);
576         fTop =    SkMinScalar(p0.fY, p1.fY);
577         fBottom = SkMaxScalar(p0.fY, p1.fY);
578     }
579 
setXYWHSkRect580     void setXYWH(SkScalar x, SkScalar y, SkScalar width, SkScalar height) {
581         fLeft = x;
582         fTop = y;
583         fRight = x + width;
584         fBottom = y + height;
585     }
586 
setWHSkRect587     void setWH(SkScalar width, SkScalar height) {
588         fLeft = 0;
589         fTop = 0;
590         fRight = width;
591         fBottom = height;
592     }
593 
594     /**
595      *  Make the largest representable rectangle
596      */
setLargestSkRect597     void setLargest() {
598         fLeft = fTop = SK_ScalarMin;
599         fRight = fBottom = SK_ScalarMax;
600     }
601 
602     /**
603      *  Make the largest representable rectangle, but inverted (e.g. fLeft will
604      *  be max and right will be min).
605      */
setLargestInvertedSkRect606     void setLargestInverted() {
607         fLeft = fTop = SK_ScalarMax;
608         fRight = fBottom = SK_ScalarMin;
609     }
610 
611     /**
612      *  Return a new Rect, built as an offset of this rect.
613      */
makeOffsetSkRect614     SkRect makeOffset(SkScalar dx, SkScalar dy) const {
615         return MakeLTRB(fLeft + dx, fTop + dy, fRight + dx, fBottom + dy);
616     }
617 
618     /**
619      *  Return a new Rect, built as an inset of this rect.
620      */
makeInsetSkRect621     SkRect makeInset(SkScalar dx, SkScalar dy) const {
622         return MakeLTRB(fLeft + dx, fTop + dy, fRight - dx, fBottom - dy);
623     }
624 
625     /**
626      *  Return a new Rect, built as an outset of this rect.
627      */
makeOutsetSkRect628     SkRect makeOutset(SkScalar dx, SkScalar dy) const {
629         return MakeLTRB(fLeft - dx, fTop - dy, fRight + dx, fBottom + dy);
630     }
631 
632     /** Offset set the rectangle by adding dx to its left and right,
633         and adding dy to its top and bottom.
634     */
offsetSkRect635     void offset(SkScalar dx, SkScalar dy) {
636         fLeft   += dx;
637         fTop    += dy;
638         fRight  += dx;
639         fBottom += dy;
640     }
641 
offsetSkRect642     void offset(const SkPoint& delta) {
643         this->offset(delta.fX, delta.fY);
644     }
645 
646     /**
647      *  Offset this rect such its new x() and y() will equal newX and newY.
648      */
offsetToSkRect649     void offsetTo(SkScalar newX, SkScalar newY) {
650         fRight += newX - fLeft;
651         fBottom += newY - fTop;
652         fLeft = newX;
653         fTop = newY;
654     }
655 
656     /** Inset the rectangle by (dx,dy). If dx is positive, then the sides are
657         moved inwards, making the rectangle narrower. If dx is negative, then
658         the sides are moved outwards, making the rectangle wider. The same holds
659          true for dy and the top and bottom.
660     */
insetSkRect661     void inset(SkScalar dx, SkScalar dy)  {
662         fLeft   += dx;
663         fTop    += dy;
664         fRight  -= dx;
665         fBottom -= dy;
666     }
667 
668    /** Outset the rectangle by (dx,dy). If dx is positive, then the sides are
669        moved outwards, making the rectangle wider. If dx is negative, then the
670        sides are moved inwards, making the rectangle narrower. The same holds
671        true for dy and the top and bottom.
672     */
outsetSkRect673     void outset(SkScalar dx, SkScalar dy)  { this->inset(-dx, -dy); }
674 
675     /** If this rectangle intersects r, return true and set this rectangle to that
676         intersection, otherwise return false and do not change this rectangle.
677         If either rectangle is empty, do nothing and return false.
678     */
679     bool intersect(const SkRect& r);
680 
681     /** If this rectangle intersects the rectangle specified by left, top, right, bottom,
682         return true and set this rectangle to that intersection, otherwise return false
683         and do not change this rectangle.
684         If either rectangle is empty, do nothing and return false.
685     */
686     bool intersect(SkScalar left, SkScalar top, SkScalar right, SkScalar bottom);
687 
688     /**
689      *  If rectangles a and b intersect, return true and set this rectangle to
690      *  that intersection, otherwise return false and do not change this
691      *  rectangle. If either rectangle is empty, do nothing and return false.
692      */
693     bool SK_WARN_UNUSED_RESULT intersect(const SkRect& a, const SkRect& b);
694 
695 
696 private:
IntersectsSkRect697     static bool Intersects(SkScalar al, SkScalar at, SkScalar ar, SkScalar ab,
698                            SkScalar bl, SkScalar bt, SkScalar br, SkScalar bb) {
699         SkScalar L = SkMaxScalar(al, bl);
700         SkScalar R = SkMinScalar(ar, br);
701         SkScalar T = SkMaxScalar(at, bt);
702         SkScalar B = SkMinScalar(ab, bb);
703         return L < R && T < B;
704     }
705 
706 public:
707     /**
708      *  Return true if this rectangle is not empty, and the specified sides of
709      *  a rectangle are not empty, and they intersect.
710      */
intersectsSkRect711     bool intersects(SkScalar left, SkScalar top, SkScalar right, SkScalar bottom) const {
712         return Intersects(fLeft, fTop, fRight, fBottom, left, top, right, bottom);
713     }
714 
intersectsSkRect715     bool intersects(const SkRect& r) const {
716         return Intersects(fLeft, fTop, fRight, fBottom,
717                           r.fLeft, r.fTop, r.fRight, r.fBottom);
718     }
719 
720     /**
721      *  Return true if rectangles a and b are not empty and intersect.
722      */
IntersectsSkRect723     static bool Intersects(const SkRect& a, const SkRect& b) {
724         return Intersects(a.fLeft, a.fTop, a.fRight, a.fBottom,
725                           b.fLeft, b.fTop, b.fRight, b.fBottom);
726     }
727 
728     /**
729      *  Update this rectangle to enclose itself and the specified rectangle.
730      *  If this rectangle is empty, just set it to the specified rectangle.
731      *  If the specified rectangle is empty, do nothing.
732      */
733     void join(SkScalar left, SkScalar top, SkScalar right, SkScalar bottom);
734 
735     /** Update this rectangle to enclose itself and the specified rectangle.
736         If this rectangle is empty, just set it to the specified rectangle. If the specified
737         rectangle is empty, do nothing.
738     */
joinSkRect739     void join(const SkRect& r) {
740         this->join(r.fLeft, r.fTop, r.fRight, r.fBottom);
741     }
742 
joinNonEmptyArgSkRect743     void joinNonEmptyArg(const SkRect& r) {
744         SkASSERT(!r.isEmpty());
745         // if we are empty, just assign
746         if (fLeft >= fRight || fTop >= fBottom) {
747             *this = r;
748         } else {
749             this->joinPossiblyEmptyRect(r);
750         }
751     }
752 
753     /**
754      * Joins the rectangle with another without checking if either are empty (may produce unexpected
755      * results if either rect is inverted).
756      */
joinPossiblyEmptyRectSkRect757     void joinPossiblyEmptyRect(const SkRect& r) {
758         fLeft   = SkMinScalar(fLeft, r.left());
759         fTop    = SkMinScalar(fTop, r.top());
760         fRight  = SkMaxScalar(fRight, r.right());
761         fBottom = SkMaxScalar(fBottom, r.bottom());
762     }
763 
764     /**
765      *  Grow the rect to include the specified (x,y). After this call, the
766      *  following will be true: fLeft <= x <= fRight && fTop <= y <= fBottom.
767      *
768      *  This is close, but not quite the same contract as contains(), since
769      *  contains() treats the left and top different from the right and bottom.
770      *  contains(x,y) -> fLeft <= x < fRight && fTop <= y < fBottom. Also note
771      *  that contains(x,y) always returns false if the rect is empty.
772      */
growToIncludeSkRect773     void growToInclude(SkScalar x, SkScalar y) {
774         fLeft  = SkMinScalar(x, fLeft);
775         fRight = SkMaxScalar(x, fRight);
776         fTop    = SkMinScalar(y, fTop);
777         fBottom = SkMaxScalar(y, fBottom);
778     }
779 
780     /** Bulk version of growToInclude */
growToIncludeSkRect781     void growToInclude(const SkPoint pts[], int count) {
782         this->growToInclude(pts, sizeof(SkPoint), count);
783     }
784 
785     /** Bulk version of growToInclude with stride. */
growToIncludeSkRect786     void growToInclude(const SkPoint pts[], size_t stride, int count) {
787         SkASSERT(count >= 0);
788         SkASSERT(stride >= sizeof(SkPoint));
789         const SkPoint* end = (const SkPoint*)((intptr_t)pts + count * stride);
790         for (; pts < end; pts = (const SkPoint*)((intptr_t)pts + stride)) {
791             this->growToInclude(pts->fX, pts->fY);
792         }
793     }
794 
795     /**
796      *  Return true if this rectangle contains r, and if both rectangles are
797      *  not empty.
798      */
containsSkRect799     bool contains(const SkRect& r) const {
800         // todo: can we eliminate the this->isEmpty check?
801         return  !r.isEmpty() && !this->isEmpty() &&
802                 fLeft <= r.fLeft && fTop <= r.fTop &&
803                 fRight >= r.fRight && fBottom >= r.fBottom;
804     }
805 
806     /**
807      * Returns true if the specified rectangle r is inside or equal to this rectangle.
808      */
containsSkRect809     bool contains(const SkIRect& r) const {
810         // todo: can we eliminate the this->isEmpty check?
811         return  !r.isEmpty() && !this->isEmpty() &&
812                 fLeft <= SkIntToScalar(r.fLeft) && fTop <= SkIntToScalar(r.fTop) &&
813                 fRight >= SkIntToScalar(r.fRight) && fBottom >= SkIntToScalar(r.fBottom);
814     }
815 
816     /**
817      *  Set the dst rectangle by rounding this rectangle's coordinates to their
818      *  nearest integer values using SkScalarRoundToInt.
819      */
roundSkRect820     void round(SkIRect* dst) const {
821         SkASSERT(dst);
822         dst->set(SkScalarRoundToInt(fLeft), SkScalarRoundToInt(fTop),
823                  SkScalarRoundToInt(fRight), SkScalarRoundToInt(fBottom));
824     }
825 
826     /**
827      *  Set the dst rectangle by rounding "out" this rectangle, choosing the
828      *  SkScalarFloor of top and left, and the SkScalarCeil of right and bottom.
829      */
roundOutSkRect830     void roundOut(SkIRect* dst) const {
831         SkASSERT(dst);
832         dst->set(SkScalarFloorToInt(fLeft), SkScalarFloorToInt(fTop),
833                  SkScalarCeilToInt(fRight), SkScalarCeilToInt(fBottom));
834     }
835 
836     /**
837      *  Set the dst rectangle by rounding "out" this rectangle, choosing the
838      *  SkScalarFloorToScalar of top and left, and the SkScalarCeilToScalar of right and bottom.
839      *
840      *  It is safe for this == dst
841      */
roundOutSkRect842     void roundOut(SkRect* dst) const {
843         dst->set(SkScalarFloorToScalar(fLeft),
844                  SkScalarFloorToScalar(fTop),
845                  SkScalarCeilToScalar(fRight),
846                  SkScalarCeilToScalar(fBottom));
847     }
848 
849     /**
850      *  Set the dst rectangle by rounding "in" this rectangle, choosing the
851      *  ceil of top and left, and the floor of right and bottom. This does *not*
852      *  call sort(), so it is possible that the resulting rect is inverted...
853      *  e.g. left >= right or top >= bottom. Call isEmpty() to detect that.
854      */
roundInSkRect855     void roundIn(SkIRect* dst) const {
856         SkASSERT(dst);
857         dst->set(SkScalarCeilToInt(fLeft), SkScalarCeilToInt(fTop),
858                  SkScalarFloorToInt(fRight), SkScalarFloorToInt(fBottom));
859     }
860 
861     //! Returns the result of calling round(&dst)
roundSkRect862     SkIRect round() const {
863         SkIRect ir;
864         this->round(&ir);
865         return ir;
866     }
867 
868     //! Returns the result of calling roundOut(&dst)
roundOutSkRect869     SkIRect roundOut() const {
870         SkIRect ir;
871         this->roundOut(&ir);
872         return ir;
873     }
874 
875     /**
876      *  Swap top/bottom or left/right if there are flipped (i.e. if width()
877      *  or height() would have returned a negative value.) This should be called
878      *  if the edges are computed separately, and may have crossed over each
879      *  other. When this returns, left <= right && top <= bottom
880      */
sortSkRect881     void sort() {
882         if (fLeft > fRight) {
883             SkTSwap<SkScalar>(fLeft, fRight);
884         }
885 
886         if (fTop > fBottom) {
887             SkTSwap<SkScalar>(fTop, fBottom);
888         }
889     }
890 
891     /**
892      *  cast-safe way to treat the rect as an array of (4) SkScalars.
893      */
asScalarsSkRect894     const SkScalar* asScalars() const { return &fLeft; }
895 
896     void dump(bool asHex) const;
dumpSkRect897     void dump() const { this->dump(false); }
dumpHexSkRect898     void dumpHex() const { this->dump(true); }
899 };
900 
contains(const SkRect & r)901 inline bool SkIRect::contains(const SkRect& r) const {
902     return  !r.isEmpty() && !this->isEmpty() &&     // check for empties
903             (SkScalar)fLeft <= r.fLeft && (SkScalar)fTop <= r.fTop &&
904             (SkScalar)fRight >= r.fRight && (SkScalar)fBottom >= r.fBottom;
905 }
906 
907 #endif
908